Compare commits
4 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 600c8b2e30 | |||
| 7469ded0de | |||
| 47cd3b338e | |||
| f44f0c2cea |
1
.gitignore
vendored
1
.gitignore
vendored
@ -5,3 +5,4 @@ yarn-debug.log*
|
|||||||
yarn-error.log*
|
yarn-error.log*
|
||||||
.DS_Store
|
.DS_Store
|
||||||
Thumbs.db
|
Thumbs.db
|
||||||
|
*.sqlite3
|
||||||
|
|||||||
@ -1,11 +1,12 @@
|
|||||||
## API Proxy Mock
|
## API Proxy Mock
|
||||||
|
|
||||||
基于 Node.js 的轻量级 **HTTP 代理 + 本地 Mock**:在 `config.json` 里配置要拦截的路径和本地文件,其余请求按配置(HTTP/HTTPS)转发到真实后端。
|
基于 Node.js 的轻量级 **HTTP 代理 + 本地 Mock**:路由与 mock 文件路径映射存储在 `sqlite3`,mock 内容仍保存在 `mock/` 目录文件中,其余请求按配置(HTTP/HTTPS)转发到真实后端。
|
||||||
|
|
||||||
### 功能说明
|
### 功能说明
|
||||||
|
|
||||||
- **代理转发**:未命中 Mock 的请求会转发到 `config.targetHost`(由 `targetHttps` 决定 HTTP/HTTPS,`targetPort` 可配)。
|
- **代理转发**:未命中 Mock 的请求会转发到 `config.targetHost`(由 `targetHttps` 决定 HTTP/HTTPS,`targetPort` 可配)。
|
||||||
- **本地 Mock**:命中路由时直接读取 `mock` 目录下的文件作为响应体。
|
- **本地 Mock**:命中路由时直接读取 `mock` 目录下的文件作为响应体。
|
||||||
|
- **SQLite 映射**:路由列表、接口列表存储在 `mock/mock-mappings.sqlite3`,数据库只保存路由与文件路径/目录关系。
|
||||||
- **Mock 总开关**:`mockEnabled` 为 `false` 时**不拦截**任何 Mock 路由,全部走代理。
|
- **Mock 总开关**:`mockEnabled` 为 `false` 时**不拦截**任何 Mock 路由,全部走代理。
|
||||||
- **热更新**:`reloadOnChange` 为 `true` 时监听 `config.json` 变更并自动重载;也可通过管理接口手动重载。
|
- **热更新**:`reloadOnChange` 为 `true` 时监听 `config.json` 变更并自动重载;也可通过管理接口手动重载。
|
||||||
- **Mock 响应**:带简单 CORS 头,以及 `X-Mock-Source`、`X-Mock-Timestamp` 便于排查。
|
- **Mock 响应**:带简单 CORS 头,以及 `X-Mock-Source`、`X-Mock-Timestamp` 便于排查。
|
||||||
@ -69,6 +70,8 @@ npm run typecheck
|
|||||||
|
|
||||||
#### `routes`
|
#### `routes`
|
||||||
|
|
||||||
|
`routes` 仍会出现在接口返回中用于兼容旧面板;实际持久化以 sqlite 为准。
|
||||||
|
|
||||||
| 说明 | |
|
| 说明 | |
|
||||||
| --- | --- |
|
| --- | --- |
|
||||||
| **key** | 请求路径,只匹配 **pathname**(不含域名;查询串不参与匹配),例如 `"/api/user/info"`。 |
|
| **key** | 请求路径,只匹配 **pathname**(不含域名;查询串不参与匹配),例如 `"/api/user/info"`。 |
|
||||||
@ -121,3 +124,4 @@ npm run typecheck
|
|||||||
- `index.api.ts`:服务入口。
|
- `index.api.ts`:服务入口。
|
||||||
- `config.json`:路由与运行参数。
|
- `config.json`:路由与运行参数。
|
||||||
- `mock/`:Mock 响应文件(文本内容原样返回,按需自行写成 JSON 等)。
|
- `mock/`:Mock 响应文件(文本内容原样返回,按需自行写成 JSON 等)。
|
||||||
|
- `mock/mock-mappings.sqlite3`:路由/接口与 mock 文件路径映射(不保存 mock 内容)。
|
||||||
|
|||||||
107
admin.html
107
admin.html
@ -14,7 +14,7 @@
|
|||||||
background: #f5f7fa;
|
background: #f5f7fa;
|
||||||
}
|
}
|
||||||
.container {
|
.container {
|
||||||
max-width: 1200px;
|
max-width: 1320px;
|
||||||
margin: 20px auto;
|
margin: 20px auto;
|
||||||
padding: 0 16px 24px;
|
padding: 0 16px 24px;
|
||||||
}
|
}
|
||||||
@ -51,6 +51,9 @@
|
|||||||
margin-top: 12px;
|
margin-top: 12px;
|
||||||
text-align: right;
|
text-align: right;
|
||||||
}
|
}
|
||||||
|
.route-table .el-table__cell .cell {
|
||||||
|
white-space: nowrap;
|
||||||
|
}
|
||||||
</style>
|
</style>
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
@ -72,28 +75,33 @@
|
|||||||
<strong>路由配置(routes)</strong>
|
<strong>路由配置(routes)</strong>
|
||||||
<el-button size="mini" type="primary" @click="openRouteDialogForCreate">新增路由</el-button>
|
<el-button size="mini" type="primary" @click="openRouteDialogForCreate">新增路由</el-button>
|
||||||
</div>
|
</div>
|
||||||
<el-table :data="pagedRoutes" border>
|
<el-table :data="pagedRoutes" border class="route-table" style="width: 100%;">
|
||||||
<el-table-column label="接口名称" min-width="220">
|
<el-table-column label="接口名称" min-width="180" show-overflow-tooltip>
|
||||||
<template slot-scope="scope">
|
<template slot-scope="scope">
|
||||||
<span>{{ scope.row.apiName || "-" }}</span>
|
<span>{{ scope.row.apiName || "-" }}</span>
|
||||||
</template>
|
</template>
|
||||||
</el-table-column>
|
</el-table-column>
|
||||||
<el-table-column label="请求路径" min-width="260">
|
<el-table-column label="请求路径" min-width="220" show-overflow-tooltip>
|
||||||
<template slot-scope="scope">
|
<template slot-scope="scope">
|
||||||
<span>{{ scope.row.route }}</span>
|
<span>{{ scope.row.route }}</span>
|
||||||
</template>
|
</template>
|
||||||
</el-table-column>
|
</el-table-column>
|
||||||
<el-table-column label="Mock 文件路径" min-width="300">
|
<el-table-column label="Mock 文件路径" min-width="240" show-overflow-tooltip>
|
||||||
<template slot-scope="scope">
|
<template slot-scope="scope">
|
||||||
<span>{{ scope.row.filePath }}</span>
|
<span>{{ scope.row.filePath }}</span>
|
||||||
</template>
|
</template>
|
||||||
</el-table-column>
|
</el-table-column>
|
||||||
|
<el-table-column label="返回状态码" width="120" align="center">
|
||||||
|
<template slot-scope="scope">
|
||||||
|
<span>{{ scope.row.statusCode || 200 }}</span>
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
<el-table-column label="启用" width="90" align="center">
|
<el-table-column label="启用" width="90" align="center">
|
||||||
<template slot-scope="scope">
|
<template slot-scope="scope">
|
||||||
<el-switch v-model="scope.row.enabled"></el-switch>
|
<el-switch v-model="scope.row.enabled"></el-switch>
|
||||||
</template>
|
</template>
|
||||||
</el-table-column>
|
</el-table-column>
|
||||||
<el-table-column label="操作" width="180">
|
<el-table-column label="操作" width="160">
|
||||||
<template slot-scope="scope">
|
<template slot-scope="scope">
|
||||||
<el-button
|
<el-button
|
||||||
size="mini"
|
size="mini"
|
||||||
@ -129,6 +137,11 @@
|
|||||||
<el-button size="mini" type="primary" @click="openMockFileDialogForCreate">新增 Mock 文件</el-button>
|
<el-button size="mini" type="primary" @click="openMockFileDialogForCreate">新增 Mock 文件</el-button>
|
||||||
</div>
|
</div>
|
||||||
<el-table :data="pagedMockFiles" border>
|
<el-table :data="pagedMockFiles" border>
|
||||||
|
<el-table-column label="别名" min-width="180">
|
||||||
|
<template slot-scope="scope">
|
||||||
|
<span>{{ scope.row.alias || "-" }}</span>
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
<el-table-column label="Mock 文件路径" min-width="320">
|
<el-table-column label="Mock 文件路径" min-width="320">
|
||||||
<template slot-scope="scope">
|
<template slot-scope="scope">
|
||||||
<span>{{ scope.row.filePath }}</span>
|
<span>{{ scope.row.filePath }}</span>
|
||||||
@ -258,11 +271,29 @@
|
|||||||
<el-option
|
<el-option
|
||||||
v-for="item in mockFiles"
|
v-for="item in mockFiles"
|
||||||
:key="item.filePath"
|
:key="item.filePath"
|
||||||
:label="item.filePath"
|
:label="item.alias ? item.alias + ' (' + item.filePath + ')' : item.filePath"
|
||||||
:value="item.filePath"
|
:value="item.filePath"
|
||||||
></el-option>
|
></el-option>
|
||||||
</el-select>
|
</el-select>
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
|
<el-form-item label="返回状态码">
|
||||||
|
<el-select
|
||||||
|
v-model="routeDialog.form.statusCode"
|
||||||
|
filterable
|
||||||
|
allow-create
|
||||||
|
default-first-option
|
||||||
|
clearable
|
||||||
|
placeholder="先选常用状态码,也可直接输入"
|
||||||
|
style="width: 100%;"
|
||||||
|
>
|
||||||
|
<el-option
|
||||||
|
v-for="item in commonStatusCodes"
|
||||||
|
:key="item.value"
|
||||||
|
:label="item.label"
|
||||||
|
:value="item.value"
|
||||||
|
></el-option>
|
||||||
|
</el-select>
|
||||||
|
</el-form-item>
|
||||||
<el-form-item label="是否启用 Mock">
|
<el-form-item label="是否启用 Mock">
|
||||||
<el-switch v-model="routeDialog.form.enabled"></el-switch>
|
<el-switch v-model="routeDialog.form.enabled"></el-switch>
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
@ -281,10 +312,16 @@
|
|||||||
width="760px"
|
width="760px"
|
||||||
>
|
>
|
||||||
<el-form :model="mockFileDialog.form" label-width="180px">
|
<el-form :model="mockFileDialog.form" label-width="180px">
|
||||||
|
<el-form-item label="别名">
|
||||||
|
<el-input
|
||||||
|
v-model="mockFileDialog.form.alias"
|
||||||
|
placeholder="可选:用于展示,支持任意文本"
|
||||||
|
></el-input>
|
||||||
|
</el-form-item>
|
||||||
<el-form-item label="Mock 文件路径">
|
<el-form-item label="Mock 文件路径">
|
||||||
<el-input
|
<el-input
|
||||||
v-model="mockFileDialog.form.filePath"
|
v-model="mockFileDialog.form.filePath"
|
||||||
placeholder="mock/new-api.json"
|
placeholder="mock/test123.json(路径仅英文和数字)"
|
||||||
:disabled="mockFileDialog.mode === 'edit'"
|
:disabled="mockFileDialog.mode === 'edit'"
|
||||||
></el-input>
|
></el-input>
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
@ -315,6 +352,22 @@
|
|||||||
activeMainTab: "routes",
|
activeMainTab: "routes",
|
||||||
apiList: [],
|
apiList: [],
|
||||||
mockFiles: [],
|
mockFiles: [],
|
||||||
|
commonStatusCodes: [
|
||||||
|
{ value: 200, label: "200 OK" },
|
||||||
|
{ value: 201, label: "201 Created" },
|
||||||
|
{ value: 204, label: "204 No Content" },
|
||||||
|
{ value: 400, label: "400 Bad Request" },
|
||||||
|
{ value: 401, label: "401 Unauthorized" },
|
||||||
|
{ value: 403, label: "403 Forbidden" },
|
||||||
|
{ value: 404, label: "404 Not Found" },
|
||||||
|
{ value: 409, label: "409 Conflict" },
|
||||||
|
{ value: 422, label: "422 Unprocessable Entity" },
|
||||||
|
{ value: 429, label: "429 Too Many Requests" },
|
||||||
|
{ value: 500, label: "500 Internal Server Error" },
|
||||||
|
{ value: 501, label: "501 Not Implemented" },
|
||||||
|
{ value: 502, label: "502 Bad Gateway" },
|
||||||
|
{ value: 503, label: "503 Service Unavailable" },
|
||||||
|
],
|
||||||
routePagination: {
|
routePagination: {
|
||||||
currentPage: 1,
|
currentPage: 1,
|
||||||
pageSize: 10,
|
pageSize: 10,
|
||||||
@ -347,6 +400,7 @@
|
|||||||
originalRawRoute: "",
|
originalRawRoute: "",
|
||||||
route: "",
|
route: "",
|
||||||
filePath: "",
|
filePath: "",
|
||||||
|
statusCode: 200,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
@ -355,6 +409,7 @@
|
|||||||
mode: "create",
|
mode: "create",
|
||||||
form: {
|
form: {
|
||||||
filePath: "mock/",
|
filePath: "mock/",
|
||||||
|
alias: "",
|
||||||
content: "",
|
content: "",
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
@ -383,6 +438,13 @@
|
|||||||
var start = (currentPage - 1) * pageSize;
|
var start = (currentPage - 1) * pageSize;
|
||||||
return source.slice(start, start + pageSize);
|
return source.slice(start, start + pageSize);
|
||||||
},
|
},
|
||||||
|
normalizeStatusCode: function (value) {
|
||||||
|
var numeric = Number(value);
|
||||||
|
if (Number.isInteger(numeric) && numeric >= 100 && numeric <= 599) {
|
||||||
|
return numeric;
|
||||||
|
}
|
||||||
|
return 200;
|
||||||
|
},
|
||||||
handleRoutePageChange: function (page) {
|
handleRoutePageChange: function (page) {
|
||||||
this.routePagination.currentPage = page;
|
this.routePagination.currentPage = page;
|
||||||
},
|
},
|
||||||
@ -394,7 +456,7 @@
|
|||||||
return item.route === route;
|
return item.route === route;
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
toRouteArray: function (routesObj) {
|
toRouteArray: function (routesObj, routeStatusesObj) {
|
||||||
var self = this;
|
var self = this;
|
||||||
return Object.keys(routesObj || {}).map(function (rawRoute) {
|
return Object.keys(routesObj || {}).map(function (rawRoute) {
|
||||||
var enabled = !rawRoute.startsWith("#");
|
var enabled = !rawRoute.startsWith("#");
|
||||||
@ -404,6 +466,9 @@
|
|||||||
rawRoute: rawRoute,
|
rawRoute: rawRoute,
|
||||||
route: route,
|
route: route,
|
||||||
filePath: routesObj[rawRoute],
|
filePath: routesObj[rawRoute],
|
||||||
|
statusCode: self.normalizeStatusCode(
|
||||||
|
routeStatusesObj && routeStatusesObj[rawRoute],
|
||||||
|
),
|
||||||
apiName: matched ? matched.name : "",
|
apiName: matched ? matched.name : "",
|
||||||
selectedApiRoute: matched ? matched.route : "",
|
selectedApiRoute: matched ? matched.route : "",
|
||||||
enabled: enabled,
|
enabled: enabled,
|
||||||
@ -422,6 +487,18 @@
|
|||||||
});
|
});
|
||||||
return obj;
|
return obj;
|
||||||
},
|
},
|
||||||
|
toRouteStatusObject: function (routeArray) {
|
||||||
|
var self = this;
|
||||||
|
var obj = {};
|
||||||
|
(routeArray || []).forEach(function (item) {
|
||||||
|
var route = (item.route || "").trim();
|
||||||
|
if (!route) return;
|
||||||
|
var routeKey = item.enabled === false ? "#" + route : route;
|
||||||
|
var statusCode = self.normalizeStatusCode(item.statusCode);
|
||||||
|
obj[routeKey] = statusCode;
|
||||||
|
});
|
||||||
|
return obj;
|
||||||
|
},
|
||||||
removeRoute: function (index) {
|
removeRoute: function (index) {
|
||||||
var actualIndex =
|
var actualIndex =
|
||||||
(this.routePagination.currentPage - 1) * this.routePagination.pageSize +
|
(this.routePagination.currentPage - 1) * this.routePagination.pageSize +
|
||||||
@ -436,6 +513,7 @@
|
|||||||
originalRawRoute: "",
|
originalRawRoute: "",
|
||||||
route: "",
|
route: "",
|
||||||
filePath: "",
|
filePath: "",
|
||||||
|
statusCode: 200,
|
||||||
enabled: true,
|
enabled: true,
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
@ -474,6 +552,7 @@
|
|||||||
originalRawRoute: item.rawRoute || item.route || "",
|
originalRawRoute: item.rawRoute || item.route || "",
|
||||||
route: item.route || "",
|
route: item.route || "",
|
||||||
filePath: item.filePath || "mock/",
|
filePath: item.filePath || "mock/",
|
||||||
|
statusCode: this.normalizeStatusCode(item.statusCode),
|
||||||
enabled: item.enabled !== false,
|
enabled: item.enabled !== false,
|
||||||
};
|
};
|
||||||
this.routeDialog.visible = true;
|
this.routeDialog.visible = true;
|
||||||
@ -481,6 +560,7 @@
|
|||||||
getDefaultMockFileForm: function () {
|
getDefaultMockFileForm: function () {
|
||||||
return {
|
return {
|
||||||
filePath: "mock/",
|
filePath: "mock/",
|
||||||
|
alias: "",
|
||||||
content: "",
|
content: "",
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
@ -493,6 +573,7 @@
|
|||||||
this.mockFileDialog.mode = "edit";
|
this.mockFileDialog.mode = "edit";
|
||||||
this.mockFileDialog.form = {
|
this.mockFileDialog.form = {
|
||||||
filePath: item.filePath || "mock/",
|
filePath: item.filePath || "mock/",
|
||||||
|
alias: String(item.alias || ""),
|
||||||
content: String(item.content || ""),
|
content: String(item.content || ""),
|
||||||
};
|
};
|
||||||
this.mockFileDialog.visible = true;
|
this.mockFileDialog.visible = true;
|
||||||
@ -522,6 +603,7 @@
|
|||||||
headers: { "Content-Type": "application/json" },
|
headers: { "Content-Type": "application/json" },
|
||||||
body: JSON.stringify({
|
body: JSON.stringify({
|
||||||
filePath: this.mockFileDialog.form.filePath,
|
filePath: this.mockFileDialog.form.filePath,
|
||||||
|
alias: this.mockFileDialog.form.alias,
|
||||||
content: this.mockFileDialog.form.content,
|
content: this.mockFileDialog.form.content,
|
||||||
}),
|
}),
|
||||||
});
|
});
|
||||||
@ -578,7 +660,10 @@
|
|||||||
var resp = await fetch("/__config");
|
var resp = await fetch("/__config");
|
||||||
var data = await resp.json();
|
var data = await resp.json();
|
||||||
this.form.config = Object.assign({}, this.form.config, data.config || {});
|
this.form.config = Object.assign({}, this.form.config, data.config || {});
|
||||||
this.form.routes = this.toRouteArray(data.routes || {});
|
this.form.routes = this.toRouteArray(
|
||||||
|
data.routes || {},
|
||||||
|
data.routeStatuses || {},
|
||||||
|
);
|
||||||
this.routePagination.currentPage = 1;
|
this.routePagination.currentPage = 1;
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
this.$message.error("加载配置失败: " + err.message);
|
this.$message.error("加载配置失败: " + err.message);
|
||||||
@ -588,6 +673,7 @@
|
|||||||
var payload = {
|
var payload = {
|
||||||
config: this.form.config,
|
config: this.form.config,
|
||||||
routes: this.toRouteObject(this.form.routes),
|
routes: this.toRouteObject(this.form.routes),
|
||||||
|
routeStatuses: this.toRouteStatusObject(this.form.routes),
|
||||||
};
|
};
|
||||||
try {
|
try {
|
||||||
var resp = await fetch("/__config", {
|
var resp = await fetch("/__config", {
|
||||||
@ -628,6 +714,7 @@
|
|||||||
}
|
}
|
||||||
var payload = Object.assign({}, this.routeDialog.form, {
|
var payload = Object.assign({}, this.routeDialog.form, {
|
||||||
useExistingFile: true,
|
useExistingFile: true,
|
||||||
|
statusCode: this.normalizeStatusCode(this.routeDialog.form.statusCode),
|
||||||
});
|
});
|
||||||
try {
|
try {
|
||||||
var resp = await fetch("/__routes", {
|
var resp = await fetch("/__routes", {
|
||||||
|
|||||||
@ -1,14 +1,20 @@
|
|||||||
{
|
{
|
||||||
"routes": {
|
"routes": {
|
||||||
|
"#/api1/account/bind": "mock/basic-error.json",
|
||||||
"/api2/admin/banner/list": "mock/banner.txt",
|
"/api2/admin/banner/list": "mock/banner.txt",
|
||||||
"/api2/home/ad/list": "mock/basic-error.json"
|
"/api2/home/ad/list": "mock/basic-error.json"
|
||||||
},
|
},
|
||||||
|
"routeStatuses": {
|
||||||
|
"#/api1/account/bind": 500,
|
||||||
|
"/api2/admin/banner/list": 200,
|
||||||
|
"/api2/home/ad/list": 200
|
||||||
|
},
|
||||||
"config": {
|
"config": {
|
||||||
"mockEnabled": true,
|
"mockEnabled": true,
|
||||||
"cacheConfig": true,
|
"cacheConfig": true,
|
||||||
"reloadOnChange": true,
|
"reloadOnChange": true,
|
||||||
"defaultContentType": "application/json",
|
"defaultContentType": "application/json",
|
||||||
"proxyPort": 8877,
|
"proxyPort": 8879,
|
||||||
"targetHost": "192.168.3.9",
|
"targetHost": "192.168.3.9",
|
||||||
"targetPort": 8092,
|
"targetPort": 8092,
|
||||||
"targetHttps": false
|
"targetHttps": false
|
||||||
|
|||||||
376
index.api.ts
376
index.api.ts
@ -3,17 +3,25 @@ import * as https from "https";
|
|||||||
import * as fs from "fs";
|
import * as fs from "fs";
|
||||||
import * as path from "path";
|
import * as path from "path";
|
||||||
import * as zlib from "zlib";
|
import * as zlib from "zlib";
|
||||||
|
import sqlite3 from "sqlite3";
|
||||||
|
|
||||||
// 配置文件路径
|
// 配置文件路径
|
||||||
const CONFIG_FILE = path.join(__dirname, "config.json");
|
const CONFIG_FILE = path.join(__dirname, "config.json");
|
||||||
const API_LIST_FILE = path.join(__dirname, "mock", "api-list.json");
|
const API_LIST_FILE = path.join(__dirname, "mock", "api-list.json");
|
||||||
const MOCK_DIR = path.join(__dirname, "mock");
|
const MOCK_DIR = path.join(__dirname, "mock");
|
||||||
|
const DATA_DIR = path.join(__dirname, "data");
|
||||||
|
const DB_FILE = path.join(DATA_DIR, "mock-mappings.sqlite3");
|
||||||
|
const LEGACY_DB_FILE = path.join(MOCK_DIR, "mock-mappings.sqlite3");
|
||||||
|
|
||||||
// 定义类型
|
// 定义类型
|
||||||
interface RouteConfig {
|
interface RouteConfig {
|
||||||
[route: string]: string;
|
[route: string]: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
interface RouteStatusConfig {
|
||||||
|
[route: string]: number;
|
||||||
|
}
|
||||||
|
|
||||||
interface AppConfig {
|
interface AppConfig {
|
||||||
/** 为 false 时所有请求走代理,不命中 mock 路由;缺省为 true */
|
/** 为 false 时所有请求走代理,不命中 mock 路由;缺省为 true */
|
||||||
mockEnabled?: boolean;
|
mockEnabled?: boolean;
|
||||||
@ -30,6 +38,7 @@ interface AppConfig {
|
|||||||
|
|
||||||
interface ConfigFile {
|
interface ConfigFile {
|
||||||
routes: RouteConfig;
|
routes: RouteConfig;
|
||||||
|
routeStatuses?: RouteStatusConfig;
|
||||||
config: AppConfig;
|
config: AppConfig;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -40,12 +49,26 @@ interface ApiListItem {
|
|||||||
|
|
||||||
interface MockFileItem {
|
interface MockFileItem {
|
||||||
filePath: string;
|
filePath: string;
|
||||||
|
alias: string;
|
||||||
content: string;
|
content: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
interface RouteRow {
|
||||||
|
route_key: string;
|
||||||
|
file_path: string;
|
||||||
|
status_code: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface MockFileRow {
|
||||||
|
file_path: string;
|
||||||
|
alias: string;
|
||||||
|
}
|
||||||
|
|
||||||
// 存储当前的路由配置
|
// 存储当前的路由配置
|
||||||
let MOCK_ROUTES: RouteConfig = {};
|
let MOCK_ROUTES: RouteConfig = {};
|
||||||
let RAW_ROUTES: RouteConfig = {};
|
let RAW_ROUTES: RouteConfig = {};
|
||||||
|
let MOCK_ROUTE_STATUSES: RouteStatusConfig = {};
|
||||||
|
let RAW_ROUTE_STATUSES: RouteStatusConfig = {};
|
||||||
let CONFIG: AppConfig = {
|
let CONFIG: AppConfig = {
|
||||||
cacheConfig: true,
|
cacheConfig: true,
|
||||||
reloadOnChange: true,
|
reloadOnChange: true,
|
||||||
@ -54,6 +77,7 @@ let CONFIG: AppConfig = {
|
|||||||
targetHost: "localhost",
|
targetHost: "localhost",
|
||||||
targetPort: 443,
|
targetPort: 443,
|
||||||
};
|
};
|
||||||
|
let DB: sqlite3.Database;
|
||||||
|
|
||||||
// 过滤掉以 # 开头的路由(视为注释,不参与 mock)
|
// 过滤掉以 # 开头的路由(视为注释,不参与 mock)
|
||||||
function filterActiveRoutes(routes: RouteConfig): RouteConfig {
|
function filterActiveRoutes(routes: RouteConfig): RouteConfig {
|
||||||
@ -65,6 +89,24 @@ function filterActiveRoutes(routes: RouteConfig): RouteConfig {
|
|||||||
return filtered;
|
return filtered;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function filterActiveRouteStatuses(statuses: RouteStatusConfig): RouteStatusConfig {
|
||||||
|
const filtered: RouteStatusConfig = {};
|
||||||
|
for (const [route, statusCode] of Object.entries(statuses)) {
|
||||||
|
if (route.startsWith("#")) continue;
|
||||||
|
filtered[route] = statusCode;
|
||||||
|
}
|
||||||
|
return filtered;
|
||||||
|
}
|
||||||
|
|
||||||
|
function normalizeStatusCode(input: unknown, fallback = 200): number {
|
||||||
|
const numeric =
|
||||||
|
typeof input === "number" ? input : Number.parseInt(String(input ?? ""), 10);
|
||||||
|
if (Number.isInteger(numeric) && numeric >= 100 && numeric <= 599) {
|
||||||
|
return numeric;
|
||||||
|
}
|
||||||
|
return fallback;
|
||||||
|
}
|
||||||
|
|
||||||
function isTargetHttps(): boolean {
|
function isTargetHttps(): boolean {
|
||||||
return CONFIG.targetHttps !== false;
|
return CONFIG.targetHttps !== false;
|
||||||
}
|
}
|
||||||
@ -83,8 +125,164 @@ function upstreamRequest(
|
|||||||
: http.request(options, callback);
|
: http.request(options, callback);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function dbRun(sql: string, params: unknown[] = []): Promise<void> {
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
DB.run(sql, params, (error) => {
|
||||||
|
if (error) {
|
||||||
|
reject(error);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
resolve();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function dbAll<T = unknown>(sql: string, params: unknown[] = []): Promise<T[]> {
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
DB.all(sql, params, (error, rows) => {
|
||||||
|
if (error) {
|
||||||
|
reject(error);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
resolve((rows as T[]) || []);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function openDatabase(): Promise<void> {
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
DB = new sqlite3.Database(DB_FILE, (error) => {
|
||||||
|
if (error) {
|
||||||
|
reject(error);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
resolve();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
async function initDatabase(): Promise<void> {
|
||||||
|
fs.mkdirSync(MOCK_DIR, { recursive: true });
|
||||||
|
fs.mkdirSync(DATA_DIR, { recursive: true });
|
||||||
|
if (!fs.existsSync(DB_FILE) && fs.existsSync(LEGACY_DB_FILE)) {
|
||||||
|
fs.copyFileSync(LEGACY_DB_FILE, DB_FILE);
|
||||||
|
}
|
||||||
|
await openDatabase();
|
||||||
|
await dbRun(`
|
||||||
|
CREATE TABLE IF NOT EXISTS route_mappings (
|
||||||
|
route_key TEXT PRIMARY KEY,
|
||||||
|
file_path TEXT NOT NULL,
|
||||||
|
status_code INTEGER NOT NULL DEFAULT 200
|
||||||
|
)
|
||||||
|
`);
|
||||||
|
await dbRun(
|
||||||
|
"ALTER TABLE route_mappings ADD COLUMN status_code INTEGER NOT NULL DEFAULT 200",
|
||||||
|
).catch(() => {
|
||||||
|
// ignore when column already exists
|
||||||
|
});
|
||||||
|
await dbRun(`
|
||||||
|
CREATE TABLE IF NOT EXISTS api_list (
|
||||||
|
route TEXT PRIMARY KEY,
|
||||||
|
name TEXT NOT NULL
|
||||||
|
)
|
||||||
|
`);
|
||||||
|
await dbRun(`
|
||||||
|
CREATE TABLE IF NOT EXISTS mock_files (
|
||||||
|
file_path TEXT PRIMARY KEY,
|
||||||
|
alias TEXT NOT NULL DEFAULT ''
|
||||||
|
)
|
||||||
|
`);
|
||||||
|
await dbRun("ALTER TABLE mock_files ADD COLUMN alias TEXT NOT NULL DEFAULT ''").catch(
|
||||||
|
() => {
|
||||||
|
// ignore when column already exists
|
||||||
|
},
|
||||||
|
);
|
||||||
|
// 清理不该出现在 mock 列表中的系统文件
|
||||||
|
await dbRun(
|
||||||
|
"DELETE FROM mock_files WHERE file_path = ? OR file_path LIKE ?",
|
||||||
|
["mock/api-list.json", "%.sqlite3"],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
async function saveRoutesToDb(
|
||||||
|
routes: RouteConfig,
|
||||||
|
routeStatuses: RouteStatusConfig = {},
|
||||||
|
): Promise<void> {
|
||||||
|
await dbRun("DELETE FROM route_mappings");
|
||||||
|
for (const [routeKey, filePath] of Object.entries(routes)) {
|
||||||
|
const statusCode = normalizeStatusCode(routeStatuses[routeKey], 200);
|
||||||
|
await dbRun(
|
||||||
|
"INSERT INTO route_mappings(route_key, file_path, status_code) VALUES(?, ?, ?)",
|
||||||
|
[routeKey, filePath, statusCode],
|
||||||
|
);
|
||||||
|
await dbRun("INSERT OR IGNORE INTO mock_files(file_path) VALUES(?)", [
|
||||||
|
filePath,
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function loadRoutesFromDb(): Promise<{
|
||||||
|
routes: RouteConfig;
|
||||||
|
routeStatuses: RouteStatusConfig;
|
||||||
|
}> {
|
||||||
|
const rows = await dbAll<RouteRow>(
|
||||||
|
"SELECT route_key, file_path, status_code FROM route_mappings ORDER BY route_key ASC",
|
||||||
|
);
|
||||||
|
const routes: RouteConfig = {};
|
||||||
|
const routeStatuses: RouteStatusConfig = {};
|
||||||
|
for (const row of rows) {
|
||||||
|
routes[row.route_key] = row.file_path;
|
||||||
|
routeStatuses[row.route_key] = normalizeStatusCode(row.status_code, 200);
|
||||||
|
}
|
||||||
|
return { routes, routeStatuses };
|
||||||
|
}
|
||||||
|
|
||||||
|
async function upsertApiListToDb(list: ApiListItem[]): Promise<void> {
|
||||||
|
await dbRun("DELETE FROM api_list");
|
||||||
|
for (const item of list) {
|
||||||
|
await dbRun("INSERT INTO api_list(route, name) VALUES(?, ?)", [
|
||||||
|
item.route,
|
||||||
|
item.name,
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function loadApiListFromDb(): Promise<ApiListItem[]> {
|
||||||
|
const rows = await dbAll<{ route: string; name: string }>(
|
||||||
|
"SELECT route, name FROM api_list ORDER BY route ASC",
|
||||||
|
);
|
||||||
|
return rows.map((row) => ({ route: row.route, name: row.name }));
|
||||||
|
}
|
||||||
|
|
||||||
|
async function upsertMockFilePathToDb(filePath: string, alias?: string): Promise<void> {
|
||||||
|
if (typeof alias === "string") {
|
||||||
|
await dbRun(
|
||||||
|
"INSERT INTO mock_files(file_path, alias) VALUES(?, ?) ON CONFLICT(file_path) DO UPDATE SET alias = excluded.alias",
|
||||||
|
[filePath, alias],
|
||||||
|
);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
await dbRun("INSERT OR IGNORE INTO mock_files(file_path, alias) VALUES(?, '')", [
|
||||||
|
filePath,
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
async function removeMockFilePathFromDb(filePath: string): Promise<void> {
|
||||||
|
await dbRun("DELETE FROM mock_files WHERE file_path = ?", [filePath]);
|
||||||
|
}
|
||||||
|
|
||||||
|
async function loadMockFilePathsFromDb(): Promise<MockFileRow[]> {
|
||||||
|
const rows = await dbAll<MockFileRow>(
|
||||||
|
"SELECT file_path, alias FROM mock_files ORDER BY file_path ASC",
|
||||||
|
);
|
||||||
|
return rows.map((row) => ({
|
||||||
|
file_path: row.file_path,
|
||||||
|
alias: String(row.alias || ""),
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
// 加载配置文件
|
// 加载配置文件
|
||||||
function loadConfig() {
|
async function loadConfig() {
|
||||||
try {
|
try {
|
||||||
if (!fs.existsSync(CONFIG_FILE)) {
|
if (!fs.existsSync(CONFIG_FILE)) {
|
||||||
console.warn(`[CONFIG] 配置文件不存在: ${CONFIG_FILE}`);
|
console.warn(`[CONFIG] 配置文件不存在: ${CONFIG_FILE}`);
|
||||||
@ -119,25 +317,32 @@ function loadConfig() {
|
|||||||
);
|
);
|
||||||
console.log(`[CONFIG] 已创建默认配置文件: ${CONFIG_FILE}`);
|
console.log(`[CONFIG] 已创建默认配置文件: ${CONFIG_FILE}`);
|
||||||
|
|
||||||
// 加载配置(# 开头的路由视为注释,不参与 mock)
|
// 加载配置,路由最终以 sqlite 为准
|
||||||
const configData: ConfigFile = JSON.parse(
|
const configData: ConfigFile = JSON.parse(
|
||||||
fs.readFileSync(CONFIG_FILE, "utf-8"),
|
fs.readFileSync(CONFIG_FILE, "utf-8"),
|
||||||
);
|
);
|
||||||
RAW_ROUTES = configData.routes || {};
|
|
||||||
MOCK_ROUTES = filterActiveRoutes(RAW_ROUTES);
|
|
||||||
CONFIG = configData.config || CONFIG;
|
CONFIG = configData.config || CONFIG;
|
||||||
|
await saveRoutesToDb(configData.routes || {}, configData.routeStatuses || {});
|
||||||
} else {
|
} else {
|
||||||
const configData: ConfigFile = JSON.parse(
|
const configData: ConfigFile = JSON.parse(
|
||||||
fs.readFileSync(CONFIG_FILE, "utf-8"),
|
fs.readFileSync(CONFIG_FILE, "utf-8"),
|
||||||
);
|
);
|
||||||
RAW_ROUTES = configData.routes || {};
|
|
||||||
MOCK_ROUTES = filterActiveRoutes(RAW_ROUTES);
|
|
||||||
CONFIG = configData.config || CONFIG;
|
CONFIG = configData.config || CONFIG;
|
||||||
console.log(
|
const dbMappings = await loadRoutesFromDb();
|
||||||
`[CONFIG] 配置文件已加载,共 ${Object.keys(MOCK_ROUTES).length} 个路由${CONFIG.mockEnabled !== false ? "" : "(mock 已关闭,全部走代理)"}`,
|
if (Object.keys(dbMappings.routes).length === 0 && configData.routes) {
|
||||||
);
|
await saveRoutesToDb(configData.routes, configData.routeStatuses || {});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const dbMappings = await loadRoutesFromDb();
|
||||||
|
RAW_ROUTES = dbMappings.routes;
|
||||||
|
RAW_ROUTE_STATUSES = dbMappings.routeStatuses;
|
||||||
|
MOCK_ROUTES = filterActiveRoutes(RAW_ROUTES);
|
||||||
|
MOCK_ROUTE_STATUSES = filterActiveRouteStatuses(RAW_ROUTE_STATUSES);
|
||||||
|
console.log(
|
||||||
|
`[CONFIG] 配置已加载(sqlite 路由 ${Object.keys(MOCK_ROUTES).length} 条)${CONFIG.mockEnabled !== false ? "" : "(mock 已关闭,全部走代理)"}`,
|
||||||
|
);
|
||||||
|
|
||||||
// 验证mock文件是否存在
|
// 验证mock文件是否存在
|
||||||
validateMockFiles();
|
validateMockFiles();
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
@ -147,6 +352,8 @@ function loadConfig() {
|
|||||||
"/api2/user/list": "mock/user_list.txt",
|
"/api2/user/list": "mock/user_list.txt",
|
||||||
};
|
};
|
||||||
RAW_ROUTES = { ...MOCK_ROUTES };
|
RAW_ROUTES = { ...MOCK_ROUTES };
|
||||||
|
RAW_ROUTE_STATUSES = {};
|
||||||
|
MOCK_ROUTE_STATUSES = {};
|
||||||
CONFIG = {
|
CONFIG = {
|
||||||
cacheConfig: true,
|
cacheConfig: true,
|
||||||
reloadOnChange: true,
|
reloadOnChange: true,
|
||||||
@ -161,18 +368,24 @@ function loadConfig() {
|
|||||||
function buildCurrentConfigFile(): ConfigFile {
|
function buildCurrentConfigFile(): ConfigFile {
|
||||||
return {
|
return {
|
||||||
routes: { ...RAW_ROUTES },
|
routes: { ...RAW_ROUTES },
|
||||||
|
routeStatuses: { ...RAW_ROUTE_STATUSES },
|
||||||
config: { ...CONFIG },
|
config: { ...CONFIG },
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
function saveConfigFile(nextConfig: ConfigFile): void {
|
async function saveConfigFile(nextConfig: ConfigFile): Promise<void> {
|
||||||
|
// routes 持久化到 sqlite,config 仍使用 config.json
|
||||||
|
await saveRoutesToDb(nextConfig.routes || {}, nextConfig.routeStatuses || {});
|
||||||
fs.writeFileSync(CONFIG_FILE, JSON.stringify(nextConfig, null, 2), "utf-8");
|
fs.writeFileSync(CONFIG_FILE, JSON.stringify(nextConfig, null, 2), "utf-8");
|
||||||
RAW_ROUTES = nextConfig.routes || {};
|
const dbMappings = await loadRoutesFromDb();
|
||||||
|
RAW_ROUTES = dbMappings.routes;
|
||||||
|
RAW_ROUTE_STATUSES = dbMappings.routeStatuses;
|
||||||
MOCK_ROUTES = filterActiveRoutes(RAW_ROUTES);
|
MOCK_ROUTES = filterActiveRoutes(RAW_ROUTES);
|
||||||
|
MOCK_ROUTE_STATUSES = filterActiveRouteStatuses(RAW_ROUTE_STATUSES);
|
||||||
CONFIG = nextConfig.config || CONFIG;
|
CONFIG = nextConfig.config || CONFIG;
|
||||||
}
|
}
|
||||||
|
|
||||||
function loadApiList(): ApiListItem[] {
|
function loadApiListFromJson(): ApiListItem[] {
|
||||||
if (!fs.existsSync(API_LIST_FILE)) {
|
if (!fs.existsSync(API_LIST_FILE)) {
|
||||||
return [];
|
return [];
|
||||||
}
|
}
|
||||||
@ -203,6 +416,18 @@ function loadApiList(): ApiListItem[] {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async function loadApiList(): Promise<ApiListItem[]> {
|
||||||
|
const dbList = await loadApiListFromDb();
|
||||||
|
if (dbList.length > 0) {
|
||||||
|
return dbList;
|
||||||
|
}
|
||||||
|
const jsonList = loadApiListFromJson();
|
||||||
|
if (jsonList.length > 0) {
|
||||||
|
await upsertApiListToDb(jsonList);
|
||||||
|
}
|
||||||
|
return jsonList;
|
||||||
|
}
|
||||||
|
|
||||||
function normalizeMockFilePath(filePath: string): string {
|
function normalizeMockFilePath(filePath: string): string {
|
||||||
const trimmed = filePath.trim().replace(/\\/g, "/");
|
const trimmed = filePath.trim().replace(/\\/g, "/");
|
||||||
if (!trimmed) {
|
if (!trimmed) {
|
||||||
@ -238,25 +463,55 @@ function walkMockFiles(dir: string, baseDir: string, result: string[]): void {
|
|||||||
}
|
}
|
||||||
if (!entry.isFile()) continue;
|
if (!entry.isFile()) continue;
|
||||||
if (entry.name === "api-list.json") continue;
|
if (entry.name === "api-list.json") continue;
|
||||||
|
if (entry.name.endsWith(".sqlite3")) continue;
|
||||||
const relative = path.relative(baseDir, fullPath).replace(/\\/g, "/");
|
const relative = path.relative(baseDir, fullPath).replace(/\\/g, "/");
|
||||||
result.push(`mock/${relative}`);
|
result.push(`mock/${relative}`);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function loadMockFiles(): MockFileItem[] {
|
async function loadMockFiles(): Promise<MockFileItem[]> {
|
||||||
|
let files = await loadMockFilePathsFromDb();
|
||||||
|
files = files.filter(
|
||||||
|
(item) =>
|
||||||
|
item.file_path !== "mock/api-list.json" &&
|
||||||
|
!item.file_path.endsWith(".sqlite3"),
|
||||||
|
);
|
||||||
|
if (files.length === 0 && fs.existsSync(MOCK_DIR)) {
|
||||||
|
const fsFiles: string[] = [];
|
||||||
|
walkMockFiles(MOCK_DIR, MOCK_DIR, fsFiles);
|
||||||
|
fsFiles.sort((a, b) => a.localeCompare(b));
|
||||||
|
for (const filePath of fsFiles) {
|
||||||
|
await upsertMockFilePathToDb(filePath);
|
||||||
|
}
|
||||||
|
files = fsFiles.map((filePath) => ({ file_path: filePath, alias: "" }));
|
||||||
|
}
|
||||||
|
|
||||||
|
const result: MockFileItem[] = [];
|
||||||
|
for (const item of files) {
|
||||||
|
const filePath = item.file_path;
|
||||||
|
const fullPath = path.join(__dirname, filePath);
|
||||||
|
if (!fs.existsSync(fullPath)) continue;
|
||||||
|
result.push({
|
||||||
|
filePath,
|
||||||
|
alias: String(item.alias || ""),
|
||||||
|
content: fs.readFileSync(fullPath, "utf-8"),
|
||||||
|
});
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
async function initMockFilesFromFsIfNeeded(): Promise<void> {
|
||||||
|
const filesInDb = await loadMockFilePathsFromDb();
|
||||||
|
if (filesInDb.length > 0 || !fs.existsSync(MOCK_DIR)) return;
|
||||||
if (!fs.existsSync(MOCK_DIR)) {
|
if (!fs.existsSync(MOCK_DIR)) {
|
||||||
return [];
|
return;
|
||||||
}
|
}
|
||||||
const files: string[] = [];
|
const files: string[] = [];
|
||||||
walkMockFiles(MOCK_DIR, MOCK_DIR, files);
|
walkMockFiles(MOCK_DIR, MOCK_DIR, files);
|
||||||
files.sort((a, b) => a.localeCompare(b));
|
files.sort((a, b) => a.localeCompare(b));
|
||||||
return files.map((filePath) => {
|
for (const filePath of files) {
|
||||||
const fullPath = path.join(__dirname, filePath);
|
await upsertMockFilePathToDb(filePath);
|
||||||
return {
|
}
|
||||||
filePath,
|
|
||||||
content: fs.readFileSync(fullPath, "utf-8"),
|
|
||||||
};
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function readBody(req: http.IncomingMessage): Promise<string> {
|
function readBody(req: http.IncomingMessage): Promise<string> {
|
||||||
@ -309,6 +564,10 @@ function getMockFilePath(requestPath: string): string {
|
|||||||
return MOCK_ROUTES[requestPath];
|
return MOCK_ROUTES[requestPath];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function getMockStatusCode(requestPath: string): number {
|
||||||
|
return normalizeStatusCode(MOCK_ROUTE_STATUSES[requestPath], 200);
|
||||||
|
}
|
||||||
|
|
||||||
function decodeBodyByEncoding(
|
function decodeBodyByEncoding(
|
||||||
bodyBuffer: Buffer,
|
bodyBuffer: Buffer,
|
||||||
contentEncoding?: string,
|
contentEncoding?: string,
|
||||||
@ -331,7 +590,7 @@ function decodeBodyByEncoding(
|
|||||||
}
|
}
|
||||||
|
|
||||||
// 代理服务器
|
// 代理服务器
|
||||||
const proxyServer = http.createServer((clientReq, clientRes) => {
|
const proxyServer = http.createServer(async (clientReq, clientRes) => {
|
||||||
// 解析客户端请求的 URL
|
// 解析客户端请求的 URL
|
||||||
const parsedUrl = new URL(`http://localhost${clientReq.url!}`);
|
const parsedUrl = new URL(`http://localhost${clientReq.url!}`);
|
||||||
const requestPath = parsedUrl.pathname;
|
const requestPath = parsedUrl.pathname;
|
||||||
@ -355,7 +614,7 @@ const proxyServer = http.createServer((clientReq, clientRes) => {
|
|||||||
|
|
||||||
try {
|
try {
|
||||||
const oldCount = Object.keys(MOCK_ROUTES).length;
|
const oldCount = Object.keys(MOCK_ROUTES).length;
|
||||||
loadConfig();
|
await loadConfig();
|
||||||
const newCount = Object.keys(MOCK_ROUTES).length;
|
const newCount = Object.keys(MOCK_ROUTES).length;
|
||||||
|
|
||||||
clientRes.writeHead(200, { "Content-Type": "application/json" });
|
clientRes.writeHead(200, { "Content-Type": "application/json" });
|
||||||
@ -400,13 +659,14 @@ const proxyServer = http.createServer((clientReq, clientRes) => {
|
|||||||
|
|
||||||
if (clientReq.method === "POST") {
|
if (clientReq.method === "POST") {
|
||||||
readBody(clientReq)
|
readBody(clientReq)
|
||||||
.then((bodyText) => {
|
.then(async (bodyText) => {
|
||||||
const body = bodyText ? JSON.parse(bodyText) : {};
|
const body = bodyText ? JSON.parse(bodyText) : {};
|
||||||
const nextConfig: ConfigFile = {
|
const nextConfig: ConfigFile = {
|
||||||
routes: body.routes || {},
|
routes: body.routes ?? RAW_ROUTES,
|
||||||
|
routeStatuses: body.routeStatuses ?? RAW_ROUTE_STATUSES,
|
||||||
config: body.config || CONFIG,
|
config: body.config || CONFIG,
|
||||||
};
|
};
|
||||||
saveConfigFile(nextConfig);
|
await saveConfigFile(nextConfig);
|
||||||
validateMockFiles();
|
validateMockFiles();
|
||||||
clientRes.writeHead(200, { "Content-Type": "application/json" });
|
clientRes.writeHead(200, { "Content-Type": "application/json" });
|
||||||
clientRes.end(
|
clientRes.end(
|
||||||
@ -435,6 +695,7 @@ const proxyServer = http.createServer((clientReq, clientRes) => {
|
|||||||
JSON.stringify(
|
JSON.stringify(
|
||||||
{
|
{
|
||||||
routes: RAW_ROUTES,
|
routes: RAW_ROUTES,
|
||||||
|
routeStatuses: RAW_ROUTE_STATUSES,
|
||||||
config: CONFIG,
|
config: CONFIG,
|
||||||
timestamp: new Date().toISOString(),
|
timestamp: new Date().toISOString(),
|
||||||
totalRoutes: Object.keys(MOCK_ROUTES).length,
|
totalRoutes: Object.keys(MOCK_ROUTES).length,
|
||||||
@ -472,7 +733,7 @@ const proxyServer = http.createServer((clientReq, clientRes) => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
readBody(clientReq)
|
readBody(clientReq)
|
||||||
.then((bodyText) => {
|
.then(async (bodyText) => {
|
||||||
const body = bodyText ? JSON.parse(bodyText) : {};
|
const body = bodyText ? JSON.parse(bodyText) : {};
|
||||||
let route = String(body.route || "").trim();
|
let route = String(body.route || "").trim();
|
||||||
const filePath = String(body.filePath || "").trim();
|
const filePath = String(body.filePath || "").trim();
|
||||||
@ -483,10 +744,11 @@ const proxyServer = http.createServer((clientReq, clientRes) => {
|
|||||||
const selectedApiRoute = String(body.selectedApiRoute || "").trim();
|
const selectedApiRoute = String(body.selectedApiRoute || "").trim();
|
||||||
const originalRoute = String(body.originalRoute || "").trim();
|
const originalRoute = String(body.originalRoute || "").trim();
|
||||||
const originalRawRoute = String(body.originalRawRoute || "").trim();
|
const originalRawRoute = String(body.originalRawRoute || "").trim();
|
||||||
|
const statusCode = normalizeStatusCode(body.statusCode, 200);
|
||||||
const enabled = body.enabled !== false;
|
const enabled = body.enabled !== false;
|
||||||
const useExistingFile =
|
const useExistingFile =
|
||||||
body.useExistingFile === true || template === "basicError";
|
body.useExistingFile === true || template === "basicError";
|
||||||
const apiList = loadApiList();
|
const apiList = await loadApiList();
|
||||||
const selectedApi = selectedApiRoute
|
const selectedApi = selectedApiRoute
|
||||||
? apiList.find((item) => item.route === selectedApiRoute)
|
? apiList.find((item) => item.route === selectedApiRoute)
|
||||||
: undefined;
|
: undefined;
|
||||||
@ -543,10 +805,16 @@ const proxyServer = http.createServer((clientReq, clientRes) => {
|
|||||||
oldRouteCandidates.forEach((key) => {
|
oldRouteCandidates.forEach((key) => {
|
||||||
if (key && key !== nextRouteKey) {
|
if (key && key !== nextRouteKey) {
|
||||||
delete nextConfig.routes[key];
|
delete nextConfig.routes[key];
|
||||||
|
delete nextConfig.routeStatuses?.[key];
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
nextConfig.routes[nextRouteKey] = normalizedFilePath;
|
nextConfig.routes[nextRouteKey] = normalizedFilePath;
|
||||||
saveConfigFile(nextConfig);
|
if (!nextConfig.routeStatuses) {
|
||||||
|
nextConfig.routeStatuses = {};
|
||||||
|
}
|
||||||
|
nextConfig.routeStatuses[nextRouteKey] = statusCode;
|
||||||
|
await saveConfigFile(nextConfig);
|
||||||
|
await upsertMockFilePathToDb(normalizedFilePath);
|
||||||
|
|
||||||
clientRes.writeHead(200, { "Content-Type": "application/json" });
|
clientRes.writeHead(200, { "Content-Type": "application/json" });
|
||||||
clientRes.end(
|
clientRes.end(
|
||||||
@ -556,6 +824,7 @@ const proxyServer = http.createServer((clientReq, clientRes) => {
|
|||||||
route,
|
route,
|
||||||
routeKey: nextRouteKey,
|
routeKey: nextRouteKey,
|
||||||
filePath: normalizedFilePath,
|
filePath: normalizedFilePath,
|
||||||
|
statusCode,
|
||||||
enabled,
|
enabled,
|
||||||
}),
|
}),
|
||||||
);
|
);
|
||||||
@ -591,7 +860,7 @@ const proxyServer = http.createServer((clientReq, clientRes) => {
|
|||||||
clientRes.end(
|
clientRes.end(
|
||||||
JSON.stringify({
|
JSON.stringify({
|
||||||
success: true,
|
success: true,
|
||||||
list: loadApiList(),
|
list: await loadApiList(),
|
||||||
}),
|
}),
|
||||||
);
|
);
|
||||||
return;
|
return;
|
||||||
@ -622,16 +891,17 @@ const proxyServer = http.createServer((clientReq, clientRes) => {
|
|||||||
clientRes.end(
|
clientRes.end(
|
||||||
JSON.stringify({
|
JSON.stringify({
|
||||||
success: true,
|
success: true,
|
||||||
list: loadMockFiles(),
|
list: await loadMockFiles(),
|
||||||
}),
|
}),
|
||||||
);
|
);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
readBody(clientReq)
|
readBody(clientReq)
|
||||||
.then((bodyText) => {
|
.then(async (bodyText) => {
|
||||||
const body = bodyText ? JSON.parse(bodyText) : {};
|
const body = bodyText ? JSON.parse(bodyText) : {};
|
||||||
const filePath = String(body.filePath || "").trim();
|
const filePath = String(body.filePath || "").trim();
|
||||||
|
const alias = String(body.alias || "");
|
||||||
const fullPath = resolveMockFullPath(filePath);
|
const fullPath = resolveMockFullPath(filePath);
|
||||||
const normalizedPath = normalizeMockFilePath(filePath);
|
const normalizedPath = normalizeMockFilePath(filePath);
|
||||||
|
|
||||||
@ -639,11 +909,13 @@ const proxyServer = http.createServer((clientReq, clientRes) => {
|
|||||||
const content = String(body.content || "");
|
const content = String(body.content || "");
|
||||||
fs.mkdirSync(path.dirname(fullPath), { recursive: true });
|
fs.mkdirSync(path.dirname(fullPath), { recursive: true });
|
||||||
fs.writeFileSync(fullPath, content, "utf-8");
|
fs.writeFileSync(fullPath, content, "utf-8");
|
||||||
|
await upsertMockFilePathToDb(normalizedPath, alias);
|
||||||
clientRes.writeHead(200, { "Content-Type": "application/json" });
|
clientRes.writeHead(200, { "Content-Type": "application/json" });
|
||||||
clientRes.end(
|
clientRes.end(
|
||||||
JSON.stringify({
|
JSON.stringify({
|
||||||
success: true,
|
success: true,
|
||||||
filePath: normalizedPath,
|
filePath: normalizedPath,
|
||||||
|
alias,
|
||||||
}),
|
}),
|
||||||
);
|
);
|
||||||
return;
|
return;
|
||||||
@ -654,6 +926,7 @@ const proxyServer = http.createServer((clientReq, clientRes) => {
|
|||||||
throw new Error("mock file does not exist");
|
throw new Error("mock file does not exist");
|
||||||
}
|
}
|
||||||
fs.unlinkSync(fullPath);
|
fs.unlinkSync(fullPath);
|
||||||
|
await removeMockFilePathFromDb(normalizedPath);
|
||||||
clientRes.writeHead(200, { "Content-Type": "application/json" });
|
clientRes.writeHead(200, { "Content-Type": "application/json" });
|
||||||
clientRes.end(
|
clientRes.end(
|
||||||
JSON.stringify({
|
JSON.stringify({
|
||||||
@ -695,6 +968,7 @@ const proxyServer = http.createServer((clientReq, clientRes) => {
|
|||||||
// 检查是否为需要mock的路由
|
// 检查是否为需要mock的路由
|
||||||
if (isMockRoute(requestPath)) {
|
if (isMockRoute(requestPath)) {
|
||||||
const mockFile = getMockFilePath(requestPath);
|
const mockFile = getMockFilePath(requestPath);
|
||||||
|
const mockStatusCode = getMockStatusCode(requestPath);
|
||||||
console.log(`[MOCK] 拦截路由: ${requestPath} -> 使用文件: ${mockFile}`);
|
console.log(`[MOCK] 拦截路由: ${requestPath} -> 使用文件: ${mockFile}`);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
@ -748,10 +1022,11 @@ const proxyServer = http.createServer((clientReq, clientRes) => {
|
|||||||
console.error(`[MOCK] 自动写入mock文件失败: ${mockFilePath}`, writeErr);
|
console.error(`[MOCK] 自动写入mock文件失败: ${mockFilePath}`, writeErr);
|
||||||
}
|
}
|
||||||
|
|
||||||
clientRes.writeHead(proxyRes.statusCode || 200, {
|
clientRes.writeHead(mockStatusCode, {
|
||||||
...proxyRes.headers,
|
...proxyRes.headers,
|
||||||
"X-Mock-Autogenerated": "true",
|
"X-Mock-Autogenerated": "true",
|
||||||
"X-Mock-Source": mockFile,
|
"X-Mock-Source": mockFile,
|
||||||
|
"X-Mock-Status-Code": String(mockStatusCode),
|
||||||
});
|
});
|
||||||
clientRes.end(bodyBuffer);
|
clientRes.end(bodyBuffer);
|
||||||
});
|
});
|
||||||
@ -780,12 +1055,13 @@ const proxyServer = http.createServer((clientReq, clientRes) => {
|
|||||||
|
|
||||||
// 设置响应头
|
// 设置响应头
|
||||||
const contentType = CONFIG.defaultContentType || "application/json";
|
const contentType = CONFIG.defaultContentType || "application/json";
|
||||||
clientRes.writeHead(200, {
|
clientRes.writeHead(mockStatusCode, {
|
||||||
"Content-Type": contentType,
|
"Content-Type": contentType,
|
||||||
"Access-Control-Allow-Origin": "*",
|
"Access-Control-Allow-Origin": "*",
|
||||||
"Access-Control-Allow-Methods": "GET, POST, PUT, DELETE, OPTIONS",
|
"Access-Control-Allow-Methods": "GET, POST, PUT, DELETE, OPTIONS",
|
||||||
"Access-Control-Allow-Headers": "Content-Type",
|
"Access-Control-Allow-Headers": "Content-Type",
|
||||||
"X-Mock-Source": mockFile,
|
"X-Mock-Source": mockFile,
|
||||||
|
"X-Mock-Status-Code": String(mockStatusCode),
|
||||||
"X-Mock-Timestamp": new Date().toISOString(),
|
"X-Mock-Timestamp": new Date().toISOString(),
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -849,16 +1125,13 @@ const proxyServer = http.createServer((clientReq, clientRes) => {
|
|||||||
clientReq.pipe(proxyReq);
|
clientReq.pipe(proxyReq);
|
||||||
});
|
});
|
||||||
|
|
||||||
// 初始化:第一次加载配置
|
function setupConfigWatcher(): void {
|
||||||
loadConfig();
|
if (!CONFIG.reloadOnChange) return;
|
||||||
|
fs.watchFile(CONFIG_FILE, async () => {
|
||||||
// 如果配置了文件监视,则监听配置文件变化
|
|
||||||
if (CONFIG.reloadOnChange) {
|
|
||||||
fs.watchFile(CONFIG_FILE, (curr, prev) => {
|
|
||||||
console.log(`[CONFIG] 配置文件已修改,重新加载...`);
|
console.log(`[CONFIG] 配置文件已修改,重新加载...`);
|
||||||
try {
|
try {
|
||||||
const oldRoutesCount = Object.keys(MOCK_ROUTES).length;
|
const oldRoutesCount = Object.keys(MOCK_ROUTES).length;
|
||||||
loadConfig();
|
await loadConfig();
|
||||||
const newRoutesCount = Object.keys(MOCK_ROUTES).length;
|
const newRoutesCount = Object.keys(MOCK_ROUTES).length;
|
||||||
console.log(
|
console.log(
|
||||||
`[CONFIG] 配置重载完成 (路由数: ${oldRoutesCount} -> ${newRoutesCount})`,
|
`[CONFIG] 配置重载完成 (路由数: ${oldRoutesCount} -> ${newRoutesCount})`,
|
||||||
@ -870,7 +1143,8 @@ if (CONFIG.reloadOnChange) {
|
|||||||
console.log(`[CONFIG] 已启用配置文件监视: ${CONFIG_FILE}`);
|
console.log(`[CONFIG] 已启用配置文件监视: ${CONFIG_FILE}`);
|
||||||
}
|
}
|
||||||
|
|
||||||
proxyServer.listen(CONFIG.proxyPort, "0.0.0.0", () => {
|
function startServer(): void {
|
||||||
|
proxyServer.listen(CONFIG.proxyPort, "0.0.0.0", () => {
|
||||||
const targetPort = getTargetPort();
|
const targetPort = getTargetPort();
|
||||||
const proto = isTargetHttps() ? "https" : "http";
|
const proto = isTargetHttps() ? "https" : "http";
|
||||||
const defaultPort = isTargetHttps() ? 443 : 80;
|
const defaultPort = isTargetHttps() ? 443 : 80;
|
||||||
@ -900,4 +1174,22 @@ proxyServer.listen(CONFIG.proxyPort, "0.0.0.0", () => {
|
|||||||
console.log(` ${exists} ${route} -> ${file}`);
|
console.log(` ${exists} ${route} -> ${file}`);
|
||||||
}
|
}
|
||||||
console.log(`========================================`);
|
console.log(`========================================`);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
async function bootstrap(): Promise<void> {
|
||||||
|
await initDatabase();
|
||||||
|
await initMockFilesFromFsIfNeeded();
|
||||||
|
await loadConfig();
|
||||||
|
const apiListFromDb = await loadApiListFromDb();
|
||||||
|
if (apiListFromDb.length === 0) {
|
||||||
|
await upsertApiListToDb(loadApiListFromJson());
|
||||||
|
}
|
||||||
|
setupConfigWatcher();
|
||||||
|
startServer();
|
||||||
|
}
|
||||||
|
|
||||||
|
bootstrap().catch((error) => {
|
||||||
|
console.error("[BOOTSTRAP] 启动失败:", error);
|
||||||
|
process.exit(1);
|
||||||
});
|
});
|
||||||
|
|||||||
713
package-lock.json
generated
713
package-lock.json
generated
@ -4,6 +4,9 @@
|
|||||||
"requires": true,
|
"requires": true,
|
||||||
"packages": {
|
"packages": {
|
||||||
"": {
|
"": {
|
||||||
|
"dependencies": {
|
||||||
|
"sqlite3": "^6.0.1"
|
||||||
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@types/node": "^25.5.0",
|
"@types/node": "^25.5.0",
|
||||||
"ts-node": "^10.9.2",
|
"ts-node": "^10.9.2",
|
||||||
@ -23,6 +26,18 @@
|
|||||||
"node": ">=12"
|
"node": ">=12"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/@isaacs/fs-minipass": {
|
||||||
|
"version": "4.0.1",
|
||||||
|
"resolved": "https://registry.npmmirror.com/@isaacs/fs-minipass/-/fs-minipass-4.0.1.tgz",
|
||||||
|
"integrity": "sha512-wgm9Ehl2jpeqP3zw/7mo3kRHFp5MEDhqAdwy1fTGkHAwnkGOVsgpvQhL8B5n1qlb01jV3n/bI0ZfZp5lWA1k4w==",
|
||||||
|
"license": "ISC",
|
||||||
|
"dependencies": {
|
||||||
|
"minipass": "^7.0.4"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=18.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/@jridgewell/resolve-uri": {
|
"node_modules/@jridgewell/resolve-uri": {
|
||||||
"version": "3.1.2",
|
"version": "3.1.2",
|
||||||
"resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.1.2.tgz",
|
"resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.1.2.tgz",
|
||||||
@ -89,6 +104,16 @@
|
|||||||
"undici-types": "~7.18.0"
|
"undici-types": "~7.18.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/abbrev": {
|
||||||
|
"version": "4.0.0",
|
||||||
|
"resolved": "https://registry.npmmirror.com/abbrev/-/abbrev-4.0.0.tgz",
|
||||||
|
"integrity": "sha512-a1wflyaL0tHtJSmLSOVybYhy22vRih4eduhhrkcjgrWGnRfrZtovJ2FRjxuTtkkj47O/baf0R86QU5OuYpz8fA==",
|
||||||
|
"license": "ISC",
|
||||||
|
"optional": true,
|
||||||
|
"engines": {
|
||||||
|
"node": "^20.17.0 || >=22.9.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/acorn": {
|
"node_modules/acorn": {
|
||||||
"version": "8.16.0",
|
"version": "8.16.0",
|
||||||
"resolved": "https://registry.npmjs.org/acorn/-/acorn-8.16.0.tgz",
|
"resolved": "https://registry.npmjs.org/acorn/-/acorn-8.16.0.tgz",
|
||||||
@ -122,6 +147,79 @@
|
|||||||
"dev": true,
|
"dev": true,
|
||||||
"license": "MIT"
|
"license": "MIT"
|
||||||
},
|
},
|
||||||
|
"node_modules/base64-js": {
|
||||||
|
"version": "1.5.1",
|
||||||
|
"resolved": "https://registry.npmmirror.com/base64-js/-/base64-js-1.5.1.tgz",
|
||||||
|
"integrity": "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==",
|
||||||
|
"funding": [
|
||||||
|
{
|
||||||
|
"type": "github",
|
||||||
|
"url": "https://github.com/sponsors/feross"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "patreon",
|
||||||
|
"url": "https://www.patreon.com/feross"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "consulting",
|
||||||
|
"url": "https://feross.org/support"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"license": "MIT"
|
||||||
|
},
|
||||||
|
"node_modules/bindings": {
|
||||||
|
"version": "1.5.0",
|
||||||
|
"resolved": "https://registry.npmmirror.com/bindings/-/bindings-1.5.0.tgz",
|
||||||
|
"integrity": "sha512-p2q/t/mhvuOj/UeLlV6566GD/guowlr0hHxClI0W9m7MWYkL1F0hLo+0Aexs9HSPCtR1SXQ0TD3MMKrXZajbiQ==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"file-uri-to-path": "1.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/bl": {
|
||||||
|
"version": "4.1.0",
|
||||||
|
"resolved": "https://registry.npmmirror.com/bl/-/bl-4.1.0.tgz",
|
||||||
|
"integrity": "sha512-1W07cM9gS6DcLperZfFSj+bWLtaPGSOHWhPiGzXmvVJbRLdG82sH/Kn8EtW1VqWVA54AKf2h5k5BbnIbwF3h6w==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"buffer": "^5.5.0",
|
||||||
|
"inherits": "^2.0.4",
|
||||||
|
"readable-stream": "^3.4.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/buffer": {
|
||||||
|
"version": "5.7.1",
|
||||||
|
"resolved": "https://registry.npmmirror.com/buffer/-/buffer-5.7.1.tgz",
|
||||||
|
"integrity": "sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ==",
|
||||||
|
"funding": [
|
||||||
|
{
|
||||||
|
"type": "github",
|
||||||
|
"url": "https://github.com/sponsors/feross"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "patreon",
|
||||||
|
"url": "https://www.patreon.com/feross"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "consulting",
|
||||||
|
"url": "https://feross.org/support"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"base64-js": "^1.3.1",
|
||||||
|
"ieee754": "^1.1.13"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/chownr": {
|
||||||
|
"version": "3.0.0",
|
||||||
|
"resolved": "https://registry.npmmirror.com/chownr/-/chownr-3.0.0.tgz",
|
||||||
|
"integrity": "sha512-+IxzY9BZOQd/XuYPRmrvEVjF/nqj5kgT4kEq7VofrDoM1MxoRjEWkrCC3EtLi59TVawxTAn+orJwFQcrqEN1+g==",
|
||||||
|
"license": "BlueOak-1.0.0",
|
||||||
|
"engines": {
|
||||||
|
"node": ">=18"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/create-require": {
|
"node_modules/create-require": {
|
||||||
"version": "1.1.1",
|
"version": "1.1.1",
|
||||||
"resolved": "https://registry.npmjs.org/create-require/-/create-require-1.1.1.tgz",
|
"resolved": "https://registry.npmjs.org/create-require/-/create-require-1.1.1.tgz",
|
||||||
@ -129,6 +227,39 @@
|
|||||||
"dev": true,
|
"dev": true,
|
||||||
"license": "MIT"
|
"license": "MIT"
|
||||||
},
|
},
|
||||||
|
"node_modules/decompress-response": {
|
||||||
|
"version": "6.0.0",
|
||||||
|
"resolved": "https://registry.npmmirror.com/decompress-response/-/decompress-response-6.0.0.tgz",
|
||||||
|
"integrity": "sha512-aW35yZM6Bb/4oJlZncMH2LCoZtJXTRxES17vE3hoRiowU2kWHaJKFkSBDnDR+cm9J+9QhXmREyIfv0pji9ejCQ==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"mimic-response": "^3.1.0"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=10"
|
||||||
|
},
|
||||||
|
"funding": {
|
||||||
|
"url": "https://github.com/sponsors/sindresorhus"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/deep-extend": {
|
||||||
|
"version": "0.6.0",
|
||||||
|
"resolved": "https://registry.npmmirror.com/deep-extend/-/deep-extend-0.6.0.tgz",
|
||||||
|
"integrity": "sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==",
|
||||||
|
"license": "MIT",
|
||||||
|
"engines": {
|
||||||
|
"node": ">=4.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/detect-libc": {
|
||||||
|
"version": "2.1.2",
|
||||||
|
"resolved": "https://registry.npmmirror.com/detect-libc/-/detect-libc-2.1.2.tgz",
|
||||||
|
"integrity": "sha512-Btj2BOOO83o3WyH59e8MgXsxEQVcarkUOpEYrubB0urwnN10yQ364rsiByU11nZlqWYZm05i/of7io4mzihBtQ==",
|
||||||
|
"license": "Apache-2.0",
|
||||||
|
"engines": {
|
||||||
|
"node": ">=8"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/diff": {
|
"node_modules/diff": {
|
||||||
"version": "4.0.4",
|
"version": "4.0.4",
|
||||||
"resolved": "https://registry.npmjs.org/diff/-/diff-4.0.4.tgz",
|
"resolved": "https://registry.npmjs.org/diff/-/diff-4.0.4.tgz",
|
||||||
@ -139,6 +270,126 @@
|
|||||||
"node": ">=0.3.1"
|
"node": ">=0.3.1"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/end-of-stream": {
|
||||||
|
"version": "1.4.5",
|
||||||
|
"resolved": "https://registry.npmmirror.com/end-of-stream/-/end-of-stream-1.4.5.tgz",
|
||||||
|
"integrity": "sha512-ooEGc6HP26xXq/N+GCGOT0JKCLDGrq2bQUZrQ7gyrJiZANJ/8YDTxTpQBXGMn+WbIQXNVpyWymm7KYVICQnyOg==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"once": "^1.4.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/env-paths": {
|
||||||
|
"version": "2.2.1",
|
||||||
|
"resolved": "https://registry.npmmirror.com/env-paths/-/env-paths-2.2.1.tgz",
|
||||||
|
"integrity": "sha512-+h1lkLKhZMTYjog1VEpJNG7NZJWcuc2DDk/qsqSTRRCOXiLjeQ1d1/udrUGhqMxUgAlwKNZ0cf2uqan5GLuS2A==",
|
||||||
|
"license": "MIT",
|
||||||
|
"optional": true,
|
||||||
|
"engines": {
|
||||||
|
"node": ">=6"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/expand-template": {
|
||||||
|
"version": "2.0.3",
|
||||||
|
"resolved": "https://registry.npmmirror.com/expand-template/-/expand-template-2.0.3.tgz",
|
||||||
|
"integrity": "sha512-XYfuKMvj4O35f/pOXLObndIRvyQ+/+6AhODh+OKWj9S9498pHHn/IMszH+gt0fBCRWMNfk1ZSp5x3AifmnI2vg==",
|
||||||
|
"license": "(MIT OR WTFPL)",
|
||||||
|
"engines": {
|
||||||
|
"node": ">=6"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/exponential-backoff": {
|
||||||
|
"version": "3.1.3",
|
||||||
|
"resolved": "https://registry.npmmirror.com/exponential-backoff/-/exponential-backoff-3.1.3.tgz",
|
||||||
|
"integrity": "sha512-ZgEeZXj30q+I0EN+CbSSpIyPaJ5HVQD18Z1m+u1FXbAeT94mr1zw50q4q6jiiC447Nl/YTcIYSAftiGqetwXCA==",
|
||||||
|
"license": "Apache-2.0",
|
||||||
|
"optional": true
|
||||||
|
},
|
||||||
|
"node_modules/fdir": {
|
||||||
|
"version": "6.5.0",
|
||||||
|
"resolved": "https://registry.npmmirror.com/fdir/-/fdir-6.5.0.tgz",
|
||||||
|
"integrity": "sha512-tIbYtZbucOs0BRGqPJkshJUYdL+SDH7dVM8gjy+ERp3WAUjLEFJE+02kanyHtwjWOnwrKYBiwAmM0p4kLJAnXg==",
|
||||||
|
"license": "MIT",
|
||||||
|
"optional": true,
|
||||||
|
"engines": {
|
||||||
|
"node": ">=12.0.0"
|
||||||
|
},
|
||||||
|
"peerDependencies": {
|
||||||
|
"picomatch": "^3 || ^4"
|
||||||
|
},
|
||||||
|
"peerDependenciesMeta": {
|
||||||
|
"picomatch": {
|
||||||
|
"optional": true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/file-uri-to-path": {
|
||||||
|
"version": "1.0.0",
|
||||||
|
"resolved": "https://registry.npmmirror.com/file-uri-to-path/-/file-uri-to-path-1.0.0.tgz",
|
||||||
|
"integrity": "sha512-0Zt+s3L7Vf1biwWZ29aARiVYLx7iMGnEUl9x33fbB/j3jR81u/O2LbqK+Bm1CDSNDKVtJ/YjwY7TUd5SkeLQLw==",
|
||||||
|
"license": "MIT"
|
||||||
|
},
|
||||||
|
"node_modules/fs-constants": {
|
||||||
|
"version": "1.0.0",
|
||||||
|
"resolved": "https://registry.npmmirror.com/fs-constants/-/fs-constants-1.0.0.tgz",
|
||||||
|
"integrity": "sha512-y6OAwoSIf7FyjMIv94u+b5rdheZEjzR63GTyZJm5qh4Bi+2YgwLCcI/fPFZkL5PSixOt6ZNKm+w+Hfp/Bciwow==",
|
||||||
|
"license": "MIT"
|
||||||
|
},
|
||||||
|
"node_modules/github-from-package": {
|
||||||
|
"version": "0.0.0",
|
||||||
|
"resolved": "https://registry.npmmirror.com/github-from-package/-/github-from-package-0.0.0.tgz",
|
||||||
|
"integrity": "sha512-SyHy3T1v2NUXn29OsWdxmK6RwHD+vkj3v8en8AOBZ1wBQ/hCAQ5bAQTD02kW4W9tUp/3Qh6J8r9EvntiyCmOOw==",
|
||||||
|
"license": "MIT"
|
||||||
|
},
|
||||||
|
"node_modules/graceful-fs": {
|
||||||
|
"version": "4.2.11",
|
||||||
|
"resolved": "https://registry.npmmirror.com/graceful-fs/-/graceful-fs-4.2.11.tgz",
|
||||||
|
"integrity": "sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==",
|
||||||
|
"license": "ISC",
|
||||||
|
"optional": true
|
||||||
|
},
|
||||||
|
"node_modules/ieee754": {
|
||||||
|
"version": "1.2.1",
|
||||||
|
"resolved": "https://registry.npmmirror.com/ieee754/-/ieee754-1.2.1.tgz",
|
||||||
|
"integrity": "sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==",
|
||||||
|
"funding": [
|
||||||
|
{
|
||||||
|
"type": "github",
|
||||||
|
"url": "https://github.com/sponsors/feross"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "patreon",
|
||||||
|
"url": "https://www.patreon.com/feross"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "consulting",
|
||||||
|
"url": "https://feross.org/support"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"license": "BSD-3-Clause"
|
||||||
|
},
|
||||||
|
"node_modules/inherits": {
|
||||||
|
"version": "2.0.4",
|
||||||
|
"resolved": "https://registry.npmmirror.com/inherits/-/inherits-2.0.4.tgz",
|
||||||
|
"integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==",
|
||||||
|
"license": "ISC"
|
||||||
|
},
|
||||||
|
"node_modules/ini": {
|
||||||
|
"version": "1.3.8",
|
||||||
|
"resolved": "https://registry.npmmirror.com/ini/-/ini-1.3.8.tgz",
|
||||||
|
"integrity": "sha512-JV/yugV2uzW5iMRSiZAyDtQd+nxtUnjeLt0acNdw98kKLrvuRVyB80tsREOE7yvGVgalhZ6RNXCmEHkUKBKxew==",
|
||||||
|
"license": "ISC"
|
||||||
|
},
|
||||||
|
"node_modules/isexe": {
|
||||||
|
"version": "4.0.0",
|
||||||
|
"resolved": "https://registry.npmmirror.com/isexe/-/isexe-4.0.0.tgz",
|
||||||
|
"integrity": "sha512-FFUtZMpoZ8RqHS3XeXEmHWLA4thH+ZxCv2lOiPIn1Xc7CxrqhWzNSDzD+/chS/zbYezmiwWLdQC09JdQKmthOw==",
|
||||||
|
"license": "BlueOak-1.0.0",
|
||||||
|
"optional": true,
|
||||||
|
"engines": {
|
||||||
|
"node": ">=20"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/make-error": {
|
"node_modules/make-error": {
|
||||||
"version": "1.3.6",
|
"version": "1.3.6",
|
||||||
"resolved": "https://registry.npmjs.org/make-error/-/make-error-1.3.6.tgz",
|
"resolved": "https://registry.npmjs.org/make-error/-/make-error-1.3.6.tgz",
|
||||||
@ -146,6 +397,409 @@
|
|||||||
"dev": true,
|
"dev": true,
|
||||||
"license": "ISC"
|
"license": "ISC"
|
||||||
},
|
},
|
||||||
|
"node_modules/mimic-response": {
|
||||||
|
"version": "3.1.0",
|
||||||
|
"resolved": "https://registry.npmmirror.com/mimic-response/-/mimic-response-3.1.0.tgz",
|
||||||
|
"integrity": "sha512-z0yWI+4FDrrweS8Zmt4Ej5HdJmky15+L2e6Wgn3+iK5fWzb6T3fhNFq2+MeTRb064c6Wr4N/wv0DzQTjNzHNGQ==",
|
||||||
|
"license": "MIT",
|
||||||
|
"engines": {
|
||||||
|
"node": ">=10"
|
||||||
|
},
|
||||||
|
"funding": {
|
||||||
|
"url": "https://github.com/sponsors/sindresorhus"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/minimist": {
|
||||||
|
"version": "1.2.8",
|
||||||
|
"resolved": "https://registry.npmmirror.com/minimist/-/minimist-1.2.8.tgz",
|
||||||
|
"integrity": "sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==",
|
||||||
|
"license": "MIT",
|
||||||
|
"funding": {
|
||||||
|
"url": "https://github.com/sponsors/ljharb"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/minipass": {
|
||||||
|
"version": "7.1.3",
|
||||||
|
"resolved": "https://registry.npmmirror.com/minipass/-/minipass-7.1.3.tgz",
|
||||||
|
"integrity": "sha512-tEBHqDnIoM/1rXME1zgka9g6Q2lcoCkxHLuc7ODJ5BxbP5d4c2Z5cGgtXAku59200Cx7diuHTOYfSBD8n6mm8A==",
|
||||||
|
"license": "BlueOak-1.0.0",
|
||||||
|
"engines": {
|
||||||
|
"node": ">=16 || 14 >=14.17"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/minizlib": {
|
||||||
|
"version": "3.1.0",
|
||||||
|
"resolved": "https://registry.npmmirror.com/minizlib/-/minizlib-3.1.0.tgz",
|
||||||
|
"integrity": "sha512-KZxYo1BUkWD2TVFLr0MQoM8vUUigWD3LlD83a/75BqC+4qE0Hb1Vo5v1FgcfaNXvfXzr+5EhQ6ing/CaBijTlw==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"minipass": "^7.1.2"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">= 18"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/mkdirp-classic": {
|
||||||
|
"version": "0.5.3",
|
||||||
|
"resolved": "https://registry.npmmirror.com/mkdirp-classic/-/mkdirp-classic-0.5.3.tgz",
|
||||||
|
"integrity": "sha512-gKLcREMhtuZRwRAfqP3RFW+TK4JqApVBtOIftVgjuABpAtpxhPGaDcfvbhNvD0B8iD1oUr/txX35NjcaY6Ns/A==",
|
||||||
|
"license": "MIT"
|
||||||
|
},
|
||||||
|
"node_modules/napi-build-utils": {
|
||||||
|
"version": "2.0.0",
|
||||||
|
"resolved": "https://registry.npmmirror.com/napi-build-utils/-/napi-build-utils-2.0.0.tgz",
|
||||||
|
"integrity": "sha512-GEbrYkbfF7MoNaoh2iGG84Mnf/WZfB0GdGEsM8wz7Expx/LlWf5U8t9nvJKXSp3qr5IsEbK04cBGhol/KwOsWA==",
|
||||||
|
"license": "MIT"
|
||||||
|
},
|
||||||
|
"node_modules/node-abi": {
|
||||||
|
"version": "3.89.0",
|
||||||
|
"resolved": "https://registry.npmmirror.com/node-abi/-/node-abi-3.89.0.tgz",
|
||||||
|
"integrity": "sha512-6u9UwL0HlAl21+agMN3YAMXcKByMqwGx+pq+P76vii5f7hTPtKDp08/H9py6DY+cfDw7kQNTGEj/rly3IgbNQA==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"semver": "^7.3.5"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=10"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/node-addon-api": {
|
||||||
|
"version": "8.7.0",
|
||||||
|
"resolved": "https://registry.npmmirror.com/node-addon-api/-/node-addon-api-8.7.0.tgz",
|
||||||
|
"integrity": "sha512-9MdFxmkKaOYVTV+XVRG8ArDwwQ77XIgIPyKASB1k3JPq3M8fGQQQE3YpMOrKm6g//Ktx8ivZr8xo1Qmtqub+GA==",
|
||||||
|
"license": "MIT",
|
||||||
|
"engines": {
|
||||||
|
"node": "^18 || ^20 || >= 21"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/node-gyp": {
|
||||||
|
"version": "12.3.0",
|
||||||
|
"resolved": "https://registry.npmmirror.com/node-gyp/-/node-gyp-12.3.0.tgz",
|
||||||
|
"integrity": "sha512-QNcUWM+HgJplcPzBvFBZ9VXacyGZ4+VTOb80PwWR+TlVzoHbRKULNEzpRsnaoxG3Wzr7Qh7BYxGDU3CbKib2Yg==",
|
||||||
|
"license": "MIT",
|
||||||
|
"optional": true,
|
||||||
|
"dependencies": {
|
||||||
|
"env-paths": "^2.2.0",
|
||||||
|
"exponential-backoff": "^3.1.1",
|
||||||
|
"graceful-fs": "^4.2.6",
|
||||||
|
"nopt": "^9.0.0",
|
||||||
|
"proc-log": "^6.0.0",
|
||||||
|
"semver": "^7.3.5",
|
||||||
|
"tar": "^7.5.4",
|
||||||
|
"tinyglobby": "^0.2.12",
|
||||||
|
"undici": "^6.25.0",
|
||||||
|
"which": "^6.0.0"
|
||||||
|
},
|
||||||
|
"bin": {
|
||||||
|
"node-gyp": "bin/node-gyp.js"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": "^20.17.0 || >=22.9.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/nopt": {
|
||||||
|
"version": "9.0.0",
|
||||||
|
"resolved": "https://registry.npmmirror.com/nopt/-/nopt-9.0.0.tgz",
|
||||||
|
"integrity": "sha512-Zhq3a+yFKrYwSBluL4H9XP3m3y5uvQkB/09CwDruCiRmR/UJYnn9W4R48ry0uGC70aeTPKLynBtscP9efFFcPw==",
|
||||||
|
"license": "ISC",
|
||||||
|
"optional": true,
|
||||||
|
"dependencies": {
|
||||||
|
"abbrev": "^4.0.0"
|
||||||
|
},
|
||||||
|
"bin": {
|
||||||
|
"nopt": "bin/nopt.js"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": "^20.17.0 || >=22.9.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/once": {
|
||||||
|
"version": "1.4.0",
|
||||||
|
"resolved": "https://registry.npmmirror.com/once/-/once-1.4.0.tgz",
|
||||||
|
"integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==",
|
||||||
|
"license": "ISC",
|
||||||
|
"dependencies": {
|
||||||
|
"wrappy": "1"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/picomatch": {
|
||||||
|
"version": "4.0.4",
|
||||||
|
"resolved": "https://registry.npmmirror.com/picomatch/-/picomatch-4.0.4.tgz",
|
||||||
|
"integrity": "sha512-QP88BAKvMam/3NxH6vj2o21R6MjxZUAd6nlwAS/pnGvN9IVLocLHxGYIzFhg6fUQ+5th6P4dv4eW9jX3DSIj7A==",
|
||||||
|
"license": "MIT",
|
||||||
|
"optional": true,
|
||||||
|
"engines": {
|
||||||
|
"node": ">=12"
|
||||||
|
},
|
||||||
|
"funding": {
|
||||||
|
"url": "https://github.com/sponsors/jonschlinkert"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/prebuild-install": {
|
||||||
|
"version": "7.1.3",
|
||||||
|
"resolved": "https://registry.npmmirror.com/prebuild-install/-/prebuild-install-7.1.3.tgz",
|
||||||
|
"integrity": "sha512-8Mf2cbV7x1cXPUILADGI3wuhfqWvtiLA1iclTDbFRZkgRQS0NqsPZphna9V+HyTEadheuPmjaJMsbzKQFOzLug==",
|
||||||
|
"deprecated": "No longer maintained. Please contact the author of the relevant native addon; alternatives are available.",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"detect-libc": "^2.0.0",
|
||||||
|
"expand-template": "^2.0.3",
|
||||||
|
"github-from-package": "0.0.0",
|
||||||
|
"minimist": "^1.2.3",
|
||||||
|
"mkdirp-classic": "^0.5.3",
|
||||||
|
"napi-build-utils": "^2.0.0",
|
||||||
|
"node-abi": "^3.3.0",
|
||||||
|
"pump": "^3.0.0",
|
||||||
|
"rc": "^1.2.7",
|
||||||
|
"simple-get": "^4.0.0",
|
||||||
|
"tar-fs": "^2.0.0",
|
||||||
|
"tunnel-agent": "^0.6.0"
|
||||||
|
},
|
||||||
|
"bin": {
|
||||||
|
"prebuild-install": "bin.js"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=10"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/proc-log": {
|
||||||
|
"version": "6.1.0",
|
||||||
|
"resolved": "https://registry.npmmirror.com/proc-log/-/proc-log-6.1.0.tgz",
|
||||||
|
"integrity": "sha512-iG+GYldRf2BQ0UDUAd6JQ/RwzaQy6mXmsk/IzlYyal4A4SNFw54MeH4/tLkF4I5WoWG9SQwuqWzS99jaFQHBuQ==",
|
||||||
|
"license": "ISC",
|
||||||
|
"optional": true,
|
||||||
|
"engines": {
|
||||||
|
"node": "^20.17.0 || >=22.9.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/pump": {
|
||||||
|
"version": "3.0.4",
|
||||||
|
"resolved": "https://registry.npmmirror.com/pump/-/pump-3.0.4.tgz",
|
||||||
|
"integrity": "sha512-VS7sjc6KR7e1ukRFhQSY5LM2uBWAUPiOPa/A3mkKmiMwSmRFUITt0xuj+/lesgnCv+dPIEYlkzrcyXgquIHMcA==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"end-of-stream": "^1.1.0",
|
||||||
|
"once": "^1.3.1"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/rc": {
|
||||||
|
"version": "1.2.8",
|
||||||
|
"resolved": "https://registry.npmmirror.com/rc/-/rc-1.2.8.tgz",
|
||||||
|
"integrity": "sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw==",
|
||||||
|
"license": "(BSD-2-Clause OR MIT OR Apache-2.0)",
|
||||||
|
"dependencies": {
|
||||||
|
"deep-extend": "^0.6.0",
|
||||||
|
"ini": "~1.3.0",
|
||||||
|
"minimist": "^1.2.0",
|
||||||
|
"strip-json-comments": "~2.0.1"
|
||||||
|
},
|
||||||
|
"bin": {
|
||||||
|
"rc": "cli.js"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/readable-stream": {
|
||||||
|
"version": "3.6.2",
|
||||||
|
"resolved": "https://registry.npmmirror.com/readable-stream/-/readable-stream-3.6.2.tgz",
|
||||||
|
"integrity": "sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"inherits": "^2.0.3",
|
||||||
|
"string_decoder": "^1.1.1",
|
||||||
|
"util-deprecate": "^1.0.1"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">= 6"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/safe-buffer": {
|
||||||
|
"version": "5.2.1",
|
||||||
|
"resolved": "https://registry.npmmirror.com/safe-buffer/-/safe-buffer-5.2.1.tgz",
|
||||||
|
"integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==",
|
||||||
|
"funding": [
|
||||||
|
{
|
||||||
|
"type": "github",
|
||||||
|
"url": "https://github.com/sponsors/feross"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "patreon",
|
||||||
|
"url": "https://www.patreon.com/feross"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "consulting",
|
||||||
|
"url": "https://feross.org/support"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"license": "MIT"
|
||||||
|
},
|
||||||
|
"node_modules/semver": {
|
||||||
|
"version": "7.7.4",
|
||||||
|
"resolved": "https://registry.npmmirror.com/semver/-/semver-7.7.4.tgz",
|
||||||
|
"integrity": "sha512-vFKC2IEtQnVhpT78h1Yp8wzwrf8CM+MzKMHGJZfBtzhZNycRFnXsHk6E5TxIkkMsgNS7mdX3AGB7x2QM2di4lA==",
|
||||||
|
"license": "ISC",
|
||||||
|
"bin": {
|
||||||
|
"semver": "bin/semver.js"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=10"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/simple-concat": {
|
||||||
|
"version": "1.0.1",
|
||||||
|
"resolved": "https://registry.npmmirror.com/simple-concat/-/simple-concat-1.0.1.tgz",
|
||||||
|
"integrity": "sha512-cSFtAPtRhljv69IK0hTVZQ+OfE9nePi/rtJmw5UjHeVyVroEqJXP1sFztKUy1qU+xvz3u/sfYJLa947b7nAN2Q==",
|
||||||
|
"funding": [
|
||||||
|
{
|
||||||
|
"type": "github",
|
||||||
|
"url": "https://github.com/sponsors/feross"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "patreon",
|
||||||
|
"url": "https://www.patreon.com/feross"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "consulting",
|
||||||
|
"url": "https://feross.org/support"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"license": "MIT"
|
||||||
|
},
|
||||||
|
"node_modules/simple-get": {
|
||||||
|
"version": "4.0.1",
|
||||||
|
"resolved": "https://registry.npmmirror.com/simple-get/-/simple-get-4.0.1.tgz",
|
||||||
|
"integrity": "sha512-brv7p5WgH0jmQJr1ZDDfKDOSeWWg+OVypG99A/5vYGPqJ6pxiaHLy8nxtFjBA7oMa01ebA9gfh1uMCFqOuXxvA==",
|
||||||
|
"funding": [
|
||||||
|
{
|
||||||
|
"type": "github",
|
||||||
|
"url": "https://github.com/sponsors/feross"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "patreon",
|
||||||
|
"url": "https://www.patreon.com/feross"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "consulting",
|
||||||
|
"url": "https://feross.org/support"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"decompress-response": "^6.0.0",
|
||||||
|
"once": "^1.3.1",
|
||||||
|
"simple-concat": "^1.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/sqlite3": {
|
||||||
|
"version": "6.0.1",
|
||||||
|
"resolved": "https://registry.npmmirror.com/sqlite3/-/sqlite3-6.0.1.tgz",
|
||||||
|
"integrity": "sha512-X0czUUMG2tmSqJpEQa3tCuZSHKIx8PwM53vLZzKp/o6Rpy25fiVfjdbnZ988M8+O3ZWR1ih0K255VumCb3MAnQ==",
|
||||||
|
"hasInstallScript": true,
|
||||||
|
"license": "BSD-3-Clause",
|
||||||
|
"dependencies": {
|
||||||
|
"bindings": "^1.5.0",
|
||||||
|
"node-addon-api": "^8.0.0",
|
||||||
|
"prebuild-install": "^7.1.3",
|
||||||
|
"tar": "^7.5.10"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=20.17.0"
|
||||||
|
},
|
||||||
|
"optionalDependencies": {
|
||||||
|
"node-gyp": "12.x"
|
||||||
|
},
|
||||||
|
"peerDependencies": {
|
||||||
|
"node-gyp": "12.x"
|
||||||
|
},
|
||||||
|
"peerDependenciesMeta": {
|
||||||
|
"node-gyp": {
|
||||||
|
"optional": true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/string_decoder": {
|
||||||
|
"version": "1.3.0",
|
||||||
|
"resolved": "https://registry.npmmirror.com/string_decoder/-/string_decoder-1.3.0.tgz",
|
||||||
|
"integrity": "sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"safe-buffer": "~5.2.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/strip-json-comments": {
|
||||||
|
"version": "2.0.1",
|
||||||
|
"resolved": "https://registry.npmmirror.com/strip-json-comments/-/strip-json-comments-2.0.1.tgz",
|
||||||
|
"integrity": "sha512-4gB8na07fecVVkOI6Rs4e7T6NOTki5EmL7TUduTs6bu3EdnSycntVJ4re8kgZA+wx9IueI2Y11bfbgwtzuE0KQ==",
|
||||||
|
"license": "MIT",
|
||||||
|
"engines": {
|
||||||
|
"node": ">=0.10.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/tar": {
|
||||||
|
"version": "7.5.13",
|
||||||
|
"resolved": "https://registry.npmmirror.com/tar/-/tar-7.5.13.tgz",
|
||||||
|
"integrity": "sha512-tOG/7GyXpFevhXVh8jOPJrmtRpOTsYqUIkVdVooZYJS/z8WhfQUX8RJILmeuJNinGAMSu1veBr4asSHFt5/hng==",
|
||||||
|
"license": "BlueOak-1.0.0",
|
||||||
|
"dependencies": {
|
||||||
|
"@isaacs/fs-minipass": "^4.0.0",
|
||||||
|
"chownr": "^3.0.0",
|
||||||
|
"minipass": "^7.1.2",
|
||||||
|
"minizlib": "^3.1.0",
|
||||||
|
"yallist": "^5.0.0"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=18"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/tar-fs": {
|
||||||
|
"version": "2.1.4",
|
||||||
|
"resolved": "https://registry.npmmirror.com/tar-fs/-/tar-fs-2.1.4.tgz",
|
||||||
|
"integrity": "sha512-mDAjwmZdh7LTT6pNleZ05Yt65HC3E+NiQzl672vQG38jIrehtJk/J3mNwIg+vShQPcLF/LV7CMnDW6vjj6sfYQ==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"chownr": "^1.1.1",
|
||||||
|
"mkdirp-classic": "^0.5.2",
|
||||||
|
"pump": "^3.0.0",
|
||||||
|
"tar-stream": "^2.1.4"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/tar-fs/node_modules/chownr": {
|
||||||
|
"version": "1.1.4",
|
||||||
|
"resolved": "https://registry.npmmirror.com/chownr/-/chownr-1.1.4.tgz",
|
||||||
|
"integrity": "sha512-jJ0bqzaylmJtVnNgzTeSOs8DPavpbYgEr/b0YL8/2GO3xJEhInFmhKMUnEJQjZumK7KXGFhUy89PrsJWlakBVg==",
|
||||||
|
"license": "ISC"
|
||||||
|
},
|
||||||
|
"node_modules/tar-stream": {
|
||||||
|
"version": "2.2.0",
|
||||||
|
"resolved": "https://registry.npmmirror.com/tar-stream/-/tar-stream-2.2.0.tgz",
|
||||||
|
"integrity": "sha512-ujeqbceABgwMZxEJnk2HDY2DlnUZ+9oEcb1KzTVfYHio0UE6dG71n60d8D2I4qNvleWrrXpmjpt7vZeF1LnMZQ==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"bl": "^4.0.3",
|
||||||
|
"end-of-stream": "^1.4.1",
|
||||||
|
"fs-constants": "^1.0.0",
|
||||||
|
"inherits": "^2.0.3",
|
||||||
|
"readable-stream": "^3.1.1"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=6"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/tinyglobby": {
|
||||||
|
"version": "0.2.16",
|
||||||
|
"resolved": "https://registry.npmmirror.com/tinyglobby/-/tinyglobby-0.2.16.tgz",
|
||||||
|
"integrity": "sha512-pn99VhoACYR8nFHhxqix+uvsbXineAasWm5ojXoN8xEwK5Kd3/TrhNn1wByuD52UxWRLy8pu+kRMniEi6Eq9Zg==",
|
||||||
|
"license": "MIT",
|
||||||
|
"optional": true,
|
||||||
|
"dependencies": {
|
||||||
|
"fdir": "^6.5.0",
|
||||||
|
"picomatch": "^4.0.4"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": ">=12.0.0"
|
||||||
|
},
|
||||||
|
"funding": {
|
||||||
|
"url": "https://github.com/sponsors/SuperchupuDev"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/ts-node": {
|
"node_modules/ts-node": {
|
||||||
"version": "10.9.2",
|
"version": "10.9.2",
|
||||||
"resolved": "https://registry.npmjs.org/ts-node/-/ts-node-10.9.2.tgz",
|
"resolved": "https://registry.npmjs.org/ts-node/-/ts-node-10.9.2.tgz",
|
||||||
@ -190,6 +844,18 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/tunnel-agent": {
|
||||||
|
"version": "0.6.0",
|
||||||
|
"resolved": "https://registry.npmmirror.com/tunnel-agent/-/tunnel-agent-0.6.0.tgz",
|
||||||
|
"integrity": "sha512-McnNiV1l8RYeY8tBgEpuodCC1mLUdbSN+CYBL7kJsJNInOP8UjDDEwdk6Mw60vdLLrr5NHKZhMAOSrR2NZuQ+w==",
|
||||||
|
"license": "Apache-2.0",
|
||||||
|
"dependencies": {
|
||||||
|
"safe-buffer": "^5.0.1"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": "*"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/typescript": {
|
"node_modules/typescript": {
|
||||||
"version": "5.9.3",
|
"version": "5.9.3",
|
||||||
"resolved": "https://registry.npmjs.org/typescript/-/typescript-5.9.3.tgz",
|
"resolved": "https://registry.npmjs.org/typescript/-/typescript-5.9.3.tgz",
|
||||||
@ -204,6 +870,16 @@
|
|||||||
"node": ">=14.17"
|
"node": ">=14.17"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/undici": {
|
||||||
|
"version": "6.25.0",
|
||||||
|
"resolved": "https://registry.npmmirror.com/undici/-/undici-6.25.0.tgz",
|
||||||
|
"integrity": "sha512-ZgpWDC5gmNiuY9CnLVXEH8rl50xhRCuLNA97fAUnKi8RRuV4E6KG31pDTsLVUKnohJE0I3XDrTeEydAXRw47xg==",
|
||||||
|
"license": "MIT",
|
||||||
|
"optional": true,
|
||||||
|
"engines": {
|
||||||
|
"node": ">=18.17"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/undici-types": {
|
"node_modules/undici-types": {
|
||||||
"version": "7.18.2",
|
"version": "7.18.2",
|
||||||
"resolved": "https://registry.npmmirror.com/undici-types/-/undici-types-7.18.2.tgz",
|
"resolved": "https://registry.npmmirror.com/undici-types/-/undici-types-7.18.2.tgz",
|
||||||
@ -211,6 +887,12 @@
|
|||||||
"dev": true,
|
"dev": true,
|
||||||
"license": "MIT"
|
"license": "MIT"
|
||||||
},
|
},
|
||||||
|
"node_modules/util-deprecate": {
|
||||||
|
"version": "1.0.2",
|
||||||
|
"resolved": "https://registry.npmmirror.com/util-deprecate/-/util-deprecate-1.0.2.tgz",
|
||||||
|
"integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==",
|
||||||
|
"license": "MIT"
|
||||||
|
},
|
||||||
"node_modules/v8-compile-cache-lib": {
|
"node_modules/v8-compile-cache-lib": {
|
||||||
"version": "3.0.1",
|
"version": "3.0.1",
|
||||||
"resolved": "https://registry.npmjs.org/v8-compile-cache-lib/-/v8-compile-cache-lib-3.0.1.tgz",
|
"resolved": "https://registry.npmjs.org/v8-compile-cache-lib/-/v8-compile-cache-lib-3.0.1.tgz",
|
||||||
@ -218,6 +900,37 @@
|
|||||||
"dev": true,
|
"dev": true,
|
||||||
"license": "MIT"
|
"license": "MIT"
|
||||||
},
|
},
|
||||||
|
"node_modules/which": {
|
||||||
|
"version": "6.0.1",
|
||||||
|
"resolved": "https://registry.npmmirror.com/which/-/which-6.0.1.tgz",
|
||||||
|
"integrity": "sha512-oGLe46MIrCRqX7ytPUf66EAYvdeMIZYn3WaocqqKZAxrBpkqHfL/qvTyJ/bTk5+AqHCjXmrv3CEWgy368zhRUg==",
|
||||||
|
"license": "ISC",
|
||||||
|
"optional": true,
|
||||||
|
"dependencies": {
|
||||||
|
"isexe": "^4.0.0"
|
||||||
|
},
|
||||||
|
"bin": {
|
||||||
|
"node-which": "bin/which.js"
|
||||||
|
},
|
||||||
|
"engines": {
|
||||||
|
"node": "^20.17.0 || >=22.9.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/wrappy": {
|
||||||
|
"version": "1.0.2",
|
||||||
|
"resolved": "https://registry.npmmirror.com/wrappy/-/wrappy-1.0.2.tgz",
|
||||||
|
"integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==",
|
||||||
|
"license": "ISC"
|
||||||
|
},
|
||||||
|
"node_modules/yallist": {
|
||||||
|
"version": "5.0.0",
|
||||||
|
"resolved": "https://registry.npmmirror.com/yallist/-/yallist-5.0.0.tgz",
|
||||||
|
"integrity": "sha512-YgvUTfwqyc7UXVMrB+SImsVYSmTS8X/tSrtdNZMImM+n7+QTriRXyXim0mBrTXNeqzVF0KWGgHPeiyViFFrNDw==",
|
||||||
|
"license": "BlueOak-1.0.0",
|
||||||
|
"engines": {
|
||||||
|
"node": ">=18"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/yn": {
|
"node_modules/yn": {
|
||||||
"version": "3.1.1",
|
"version": "3.1.1",
|
||||||
"resolved": "https://registry.npmjs.org/yn/-/yn-3.1.1.tgz",
|
"resolved": "https://registry.npmjs.org/yn/-/yn-3.1.1.tgz",
|
||||||
|
|||||||
@ -8,5 +8,13 @@
|
|||||||
"@types/node": "^25.5.0",
|
"@types/node": "^25.5.0",
|
||||||
"ts-node": "^10.9.2",
|
"ts-node": "^10.9.2",
|
||||||
"typescript": "^5.7.3"
|
"typescript": "^5.7.3"
|
||||||
|
},
|
||||||
|
"dependencies": {
|
||||||
|
"sqlite3": "^6.0.1"
|
||||||
|
},
|
||||||
|
"pnpm": {
|
||||||
|
"onlyBuiltDependencies": [
|
||||||
|
"sqlite3"
|
||||||
|
]
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
447
pnpm-lock.yaml
generated
447
pnpm-lock.yaml
generated
@ -7,6 +7,10 @@ settings:
|
|||||||
importers:
|
importers:
|
||||||
|
|
||||||
.:
|
.:
|
||||||
|
dependencies:
|
||||||
|
sqlite3:
|
||||||
|
specifier: ^6.0.1
|
||||||
|
version: 6.0.1
|
||||||
devDependencies:
|
devDependencies:
|
||||||
'@types/node':
|
'@types/node':
|
||||||
specifier: ^25.5.0
|
specifier: ^25.5.0
|
||||||
@ -24,6 +28,10 @@ packages:
|
|||||||
resolution: {integrity: sha512-IchNf6dN4tHoMFIn/7OE8LWZ19Y6q/67Bmf6vnGREv8RSbBVb9LPJxEcnwrcwX6ixSvaiGoomAUvu4YSxXrVgw==}
|
resolution: {integrity: sha512-IchNf6dN4tHoMFIn/7OE8LWZ19Y6q/67Bmf6vnGREv8RSbBVb9LPJxEcnwrcwX6ixSvaiGoomAUvu4YSxXrVgw==}
|
||||||
engines: {node: '>=12'}
|
engines: {node: '>=12'}
|
||||||
|
|
||||||
|
'@isaacs/fs-minipass@4.0.1':
|
||||||
|
resolution: {integrity: sha512-wgm9Ehl2jpeqP3zw/7mo3kRHFp5MEDhqAdwy1fTGkHAwnkGOVsgpvQhL8B5n1qlb01jV3n/bI0ZfZp5lWA1k4w==}
|
||||||
|
engines: {node: '>=18.0.0'}
|
||||||
|
|
||||||
'@jridgewell/resolve-uri@3.1.2':
|
'@jridgewell/resolve-uri@3.1.2':
|
||||||
resolution: {integrity: sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==}
|
resolution: {integrity: sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==}
|
||||||
engines: {node: '>=6.0.0'}
|
engines: {node: '>=6.0.0'}
|
||||||
@ -49,6 +57,10 @@ packages:
|
|||||||
'@types/node@25.6.0':
|
'@types/node@25.6.0':
|
||||||
resolution: {integrity: sha512-+qIYRKdNYJwY3vRCZMdJbPLJAtGjQBudzZzdzwQYkEPQd+PJGixUL5QfvCLDaULoLv+RhT3LDkwEfKaAkgSmNQ==}
|
resolution: {integrity: sha512-+qIYRKdNYJwY3vRCZMdJbPLJAtGjQBudzZzdzwQYkEPQd+PJGixUL5QfvCLDaULoLv+RhT3LDkwEfKaAkgSmNQ==}
|
||||||
|
|
||||||
|
abbrev@4.0.0:
|
||||||
|
resolution: {integrity: sha512-a1wflyaL0tHtJSmLSOVybYhy22vRih4eduhhrkcjgrWGnRfrZtovJ2FRjxuTtkkj47O/baf0R86QU5OuYpz8fA==}
|
||||||
|
engines: {node: ^20.17.0 || >=22.9.0}
|
||||||
|
|
||||||
acorn-walk@8.3.5:
|
acorn-walk@8.3.5:
|
||||||
resolution: {integrity: sha512-HEHNfbars9v4pgpW6SO1KSPkfoS0xVOM/9UzkJltjlsHZmJasxg8aXkuZa7SMf8vKGIBhpUsPluQSqhJFCqebw==}
|
resolution: {integrity: sha512-HEHNfbars9v4pgpW6SO1KSPkfoS0xVOM/9UzkJltjlsHZmJasxg8aXkuZa7SMf8vKGIBhpUsPluQSqhJFCqebw==}
|
||||||
engines: {node: '>=0.4.0'}
|
engines: {node: '>=0.4.0'}
|
||||||
@ -61,16 +73,202 @@ packages:
|
|||||||
arg@4.1.3:
|
arg@4.1.3:
|
||||||
resolution: {integrity: sha512-58S9QDqG0Xx27YwPSt9fJxivjYl432YCwfDMfZ+71RAqUrZef7LrKQZ3LHLOwCS4FLNBplP533Zx895SeOCHvA==}
|
resolution: {integrity: sha512-58S9QDqG0Xx27YwPSt9fJxivjYl432YCwfDMfZ+71RAqUrZef7LrKQZ3LHLOwCS4FLNBplP533Zx895SeOCHvA==}
|
||||||
|
|
||||||
|
base64-js@1.5.1:
|
||||||
|
resolution: {integrity: sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==}
|
||||||
|
|
||||||
|
bindings@1.5.0:
|
||||||
|
resolution: {integrity: sha512-p2q/t/mhvuOj/UeLlV6566GD/guowlr0hHxClI0W9m7MWYkL1F0hLo+0Aexs9HSPCtR1SXQ0TD3MMKrXZajbiQ==}
|
||||||
|
|
||||||
|
bl@4.1.0:
|
||||||
|
resolution: {integrity: sha512-1W07cM9gS6DcLperZfFSj+bWLtaPGSOHWhPiGzXmvVJbRLdG82sH/Kn8EtW1VqWVA54AKf2h5k5BbnIbwF3h6w==}
|
||||||
|
|
||||||
|
buffer@5.7.1:
|
||||||
|
resolution: {integrity: sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ==}
|
||||||
|
|
||||||
|
chownr@1.1.4:
|
||||||
|
resolution: {integrity: sha512-jJ0bqzaylmJtVnNgzTeSOs8DPavpbYgEr/b0YL8/2GO3xJEhInFmhKMUnEJQjZumK7KXGFhUy89PrsJWlakBVg==}
|
||||||
|
|
||||||
|
chownr@3.0.0:
|
||||||
|
resolution: {integrity: sha512-+IxzY9BZOQd/XuYPRmrvEVjF/nqj5kgT4kEq7VofrDoM1MxoRjEWkrCC3EtLi59TVawxTAn+orJwFQcrqEN1+g==}
|
||||||
|
engines: {node: '>=18'}
|
||||||
|
|
||||||
create-require@1.1.1:
|
create-require@1.1.1:
|
||||||
resolution: {integrity: sha512-dcKFX3jn0MpIaXjisoRvexIJVEKzaq7z2rZKxf+MSr9TkdmHmsU4m2lcLojrj/FHl8mk5VxMmYA+ftRkP/3oKQ==}
|
resolution: {integrity: sha512-dcKFX3jn0MpIaXjisoRvexIJVEKzaq7z2rZKxf+MSr9TkdmHmsU4m2lcLojrj/FHl8mk5VxMmYA+ftRkP/3oKQ==}
|
||||||
|
|
||||||
|
decompress-response@6.0.0:
|
||||||
|
resolution: {integrity: sha512-aW35yZM6Bb/4oJlZncMH2LCoZtJXTRxES17vE3hoRiowU2kWHaJKFkSBDnDR+cm9J+9QhXmREyIfv0pji9ejCQ==}
|
||||||
|
engines: {node: '>=10'}
|
||||||
|
|
||||||
|
deep-extend@0.6.0:
|
||||||
|
resolution: {integrity: sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==}
|
||||||
|
engines: {node: '>=4.0.0'}
|
||||||
|
|
||||||
|
detect-libc@2.1.2:
|
||||||
|
resolution: {integrity: sha512-Btj2BOOO83o3WyH59e8MgXsxEQVcarkUOpEYrubB0urwnN10yQ364rsiByU11nZlqWYZm05i/of7io4mzihBtQ==}
|
||||||
|
engines: {node: '>=8'}
|
||||||
|
|
||||||
diff@4.0.4:
|
diff@4.0.4:
|
||||||
resolution: {integrity: sha512-X07nttJQkwkfKfvTPG/KSnE2OMdcUCao6+eXF3wmnIQRn2aPAHH3VxDbDOdegkd6JbPsXqShpvEOHfAT+nCNwQ==}
|
resolution: {integrity: sha512-X07nttJQkwkfKfvTPG/KSnE2OMdcUCao6+eXF3wmnIQRn2aPAHH3VxDbDOdegkd6JbPsXqShpvEOHfAT+nCNwQ==}
|
||||||
engines: {node: '>=0.3.1'}
|
engines: {node: '>=0.3.1'}
|
||||||
|
|
||||||
|
end-of-stream@1.4.5:
|
||||||
|
resolution: {integrity: sha512-ooEGc6HP26xXq/N+GCGOT0JKCLDGrq2bQUZrQ7gyrJiZANJ/8YDTxTpQBXGMn+WbIQXNVpyWymm7KYVICQnyOg==}
|
||||||
|
|
||||||
|
env-paths@2.2.1:
|
||||||
|
resolution: {integrity: sha512-+h1lkLKhZMTYjog1VEpJNG7NZJWcuc2DDk/qsqSTRRCOXiLjeQ1d1/udrUGhqMxUgAlwKNZ0cf2uqan5GLuS2A==}
|
||||||
|
engines: {node: '>=6'}
|
||||||
|
|
||||||
|
expand-template@2.0.3:
|
||||||
|
resolution: {integrity: sha512-XYfuKMvj4O35f/pOXLObndIRvyQ+/+6AhODh+OKWj9S9498pHHn/IMszH+gt0fBCRWMNfk1ZSp5x3AifmnI2vg==}
|
||||||
|
engines: {node: '>=6'}
|
||||||
|
|
||||||
|
exponential-backoff@3.1.3:
|
||||||
|
resolution: {integrity: sha512-ZgEeZXj30q+I0EN+CbSSpIyPaJ5HVQD18Z1m+u1FXbAeT94mr1zw50q4q6jiiC447Nl/YTcIYSAftiGqetwXCA==}
|
||||||
|
|
||||||
|
fdir@6.5.0:
|
||||||
|
resolution: {integrity: sha512-tIbYtZbucOs0BRGqPJkshJUYdL+SDH7dVM8gjy+ERp3WAUjLEFJE+02kanyHtwjWOnwrKYBiwAmM0p4kLJAnXg==}
|
||||||
|
engines: {node: '>=12.0.0'}
|
||||||
|
peerDependencies:
|
||||||
|
picomatch: ^3 || ^4
|
||||||
|
peerDependenciesMeta:
|
||||||
|
picomatch:
|
||||||
|
optional: true
|
||||||
|
|
||||||
|
file-uri-to-path@1.0.0:
|
||||||
|
resolution: {integrity: sha512-0Zt+s3L7Vf1biwWZ29aARiVYLx7iMGnEUl9x33fbB/j3jR81u/O2LbqK+Bm1CDSNDKVtJ/YjwY7TUd5SkeLQLw==}
|
||||||
|
|
||||||
|
fs-constants@1.0.0:
|
||||||
|
resolution: {integrity: sha512-y6OAwoSIf7FyjMIv94u+b5rdheZEjzR63GTyZJm5qh4Bi+2YgwLCcI/fPFZkL5PSixOt6ZNKm+w+Hfp/Bciwow==}
|
||||||
|
|
||||||
|
github-from-package@0.0.0:
|
||||||
|
resolution: {integrity: sha512-SyHy3T1v2NUXn29OsWdxmK6RwHD+vkj3v8en8AOBZ1wBQ/hCAQ5bAQTD02kW4W9tUp/3Qh6J8r9EvntiyCmOOw==}
|
||||||
|
|
||||||
|
graceful-fs@4.2.11:
|
||||||
|
resolution: {integrity: sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==}
|
||||||
|
|
||||||
|
ieee754@1.2.1:
|
||||||
|
resolution: {integrity: sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==}
|
||||||
|
|
||||||
|
inherits@2.0.4:
|
||||||
|
resolution: {integrity: sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ==}
|
||||||
|
|
||||||
|
ini@1.3.8:
|
||||||
|
resolution: {integrity: sha512-JV/yugV2uzW5iMRSiZAyDtQd+nxtUnjeLt0acNdw98kKLrvuRVyB80tsREOE7yvGVgalhZ6RNXCmEHkUKBKxew==}
|
||||||
|
|
||||||
|
isexe@4.0.0:
|
||||||
|
resolution: {integrity: sha512-FFUtZMpoZ8RqHS3XeXEmHWLA4thH+ZxCv2lOiPIn1Xc7CxrqhWzNSDzD+/chS/zbYezmiwWLdQC09JdQKmthOw==}
|
||||||
|
engines: {node: '>=20'}
|
||||||
|
|
||||||
make-error@1.3.6:
|
make-error@1.3.6:
|
||||||
resolution: {integrity: sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw==}
|
resolution: {integrity: sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw==}
|
||||||
|
|
||||||
|
mimic-response@3.1.0:
|
||||||
|
resolution: {integrity: sha512-z0yWI+4FDrrweS8Zmt4Ej5HdJmky15+L2e6Wgn3+iK5fWzb6T3fhNFq2+MeTRb064c6Wr4N/wv0DzQTjNzHNGQ==}
|
||||||
|
engines: {node: '>=10'}
|
||||||
|
|
||||||
|
minimist@1.2.8:
|
||||||
|
resolution: {integrity: sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==}
|
||||||
|
|
||||||
|
minipass@7.1.3:
|
||||||
|
resolution: {integrity: sha512-tEBHqDnIoM/1rXME1zgka9g6Q2lcoCkxHLuc7ODJ5BxbP5d4c2Z5cGgtXAku59200Cx7diuHTOYfSBD8n6mm8A==}
|
||||||
|
engines: {node: '>=16 || 14 >=14.17'}
|
||||||
|
|
||||||
|
minizlib@3.1.0:
|
||||||
|
resolution: {integrity: sha512-KZxYo1BUkWD2TVFLr0MQoM8vUUigWD3LlD83a/75BqC+4qE0Hb1Vo5v1FgcfaNXvfXzr+5EhQ6ing/CaBijTlw==}
|
||||||
|
engines: {node: '>= 18'}
|
||||||
|
|
||||||
|
mkdirp-classic@0.5.3:
|
||||||
|
resolution: {integrity: sha512-gKLcREMhtuZRwRAfqP3RFW+TK4JqApVBtOIftVgjuABpAtpxhPGaDcfvbhNvD0B8iD1oUr/txX35NjcaY6Ns/A==}
|
||||||
|
|
||||||
|
napi-build-utils@2.0.0:
|
||||||
|
resolution: {integrity: sha512-GEbrYkbfF7MoNaoh2iGG84Mnf/WZfB0GdGEsM8wz7Expx/LlWf5U8t9nvJKXSp3qr5IsEbK04cBGhol/KwOsWA==}
|
||||||
|
|
||||||
|
node-abi@3.89.0:
|
||||||
|
resolution: {integrity: sha512-6u9UwL0HlAl21+agMN3YAMXcKByMqwGx+pq+P76vii5f7hTPtKDp08/H9py6DY+cfDw7kQNTGEj/rly3IgbNQA==}
|
||||||
|
engines: {node: '>=10'}
|
||||||
|
|
||||||
|
node-addon-api@8.7.0:
|
||||||
|
resolution: {integrity: sha512-9MdFxmkKaOYVTV+XVRG8ArDwwQ77XIgIPyKASB1k3JPq3M8fGQQQE3YpMOrKm6g//Ktx8ivZr8xo1Qmtqub+GA==}
|
||||||
|
engines: {node: ^18 || ^20 || >= 21}
|
||||||
|
|
||||||
|
node-gyp@12.3.0:
|
||||||
|
resolution: {integrity: sha512-QNcUWM+HgJplcPzBvFBZ9VXacyGZ4+VTOb80PwWR+TlVzoHbRKULNEzpRsnaoxG3Wzr7Qh7BYxGDU3CbKib2Yg==}
|
||||||
|
engines: {node: ^20.17.0 || >=22.9.0}
|
||||||
|
hasBin: true
|
||||||
|
|
||||||
|
nopt@9.0.0:
|
||||||
|
resolution: {integrity: sha512-Zhq3a+yFKrYwSBluL4H9XP3m3y5uvQkB/09CwDruCiRmR/UJYnn9W4R48ry0uGC70aeTPKLynBtscP9efFFcPw==}
|
||||||
|
engines: {node: ^20.17.0 || >=22.9.0}
|
||||||
|
hasBin: true
|
||||||
|
|
||||||
|
once@1.4.0:
|
||||||
|
resolution: {integrity: sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==}
|
||||||
|
|
||||||
|
picomatch@4.0.4:
|
||||||
|
resolution: {integrity: sha512-QP88BAKvMam/3NxH6vj2o21R6MjxZUAd6nlwAS/pnGvN9IVLocLHxGYIzFhg6fUQ+5th6P4dv4eW9jX3DSIj7A==}
|
||||||
|
engines: {node: '>=12'}
|
||||||
|
|
||||||
|
prebuild-install@7.1.3:
|
||||||
|
resolution: {integrity: sha512-8Mf2cbV7x1cXPUILADGI3wuhfqWvtiLA1iclTDbFRZkgRQS0NqsPZphna9V+HyTEadheuPmjaJMsbzKQFOzLug==}
|
||||||
|
engines: {node: '>=10'}
|
||||||
|
deprecated: No longer maintained. Please contact the author of the relevant native addon; alternatives are available.
|
||||||
|
hasBin: true
|
||||||
|
|
||||||
|
proc-log@6.1.0:
|
||||||
|
resolution: {integrity: sha512-iG+GYldRf2BQ0UDUAd6JQ/RwzaQy6mXmsk/IzlYyal4A4SNFw54MeH4/tLkF4I5WoWG9SQwuqWzS99jaFQHBuQ==}
|
||||||
|
engines: {node: ^20.17.0 || >=22.9.0}
|
||||||
|
|
||||||
|
pump@3.0.4:
|
||||||
|
resolution: {integrity: sha512-VS7sjc6KR7e1ukRFhQSY5LM2uBWAUPiOPa/A3mkKmiMwSmRFUITt0xuj+/lesgnCv+dPIEYlkzrcyXgquIHMcA==}
|
||||||
|
|
||||||
|
rc@1.2.8:
|
||||||
|
resolution: {integrity: sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw==}
|
||||||
|
hasBin: true
|
||||||
|
|
||||||
|
readable-stream@3.6.2:
|
||||||
|
resolution: {integrity: sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA==}
|
||||||
|
engines: {node: '>= 6'}
|
||||||
|
|
||||||
|
safe-buffer@5.2.1:
|
||||||
|
resolution: {integrity: sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==}
|
||||||
|
|
||||||
|
semver@7.7.4:
|
||||||
|
resolution: {integrity: sha512-vFKC2IEtQnVhpT78h1Yp8wzwrf8CM+MzKMHGJZfBtzhZNycRFnXsHk6E5TxIkkMsgNS7mdX3AGB7x2QM2di4lA==}
|
||||||
|
engines: {node: '>=10'}
|
||||||
|
hasBin: true
|
||||||
|
|
||||||
|
simple-concat@1.0.1:
|
||||||
|
resolution: {integrity: sha512-cSFtAPtRhljv69IK0hTVZQ+OfE9nePi/rtJmw5UjHeVyVroEqJXP1sFztKUy1qU+xvz3u/sfYJLa947b7nAN2Q==}
|
||||||
|
|
||||||
|
simple-get@4.0.1:
|
||||||
|
resolution: {integrity: sha512-brv7p5WgH0jmQJr1ZDDfKDOSeWWg+OVypG99A/5vYGPqJ6pxiaHLy8nxtFjBA7oMa01ebA9gfh1uMCFqOuXxvA==}
|
||||||
|
|
||||||
|
sqlite3@6.0.1:
|
||||||
|
resolution: {integrity: sha512-X0czUUMG2tmSqJpEQa3tCuZSHKIx8PwM53vLZzKp/o6Rpy25fiVfjdbnZ988M8+O3ZWR1ih0K255VumCb3MAnQ==}
|
||||||
|
engines: {node: '>=20.17.0'}
|
||||||
|
|
||||||
|
string_decoder@1.3.0:
|
||||||
|
resolution: {integrity: sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA==}
|
||||||
|
|
||||||
|
strip-json-comments@2.0.1:
|
||||||
|
resolution: {integrity: sha512-4gB8na07fecVVkOI6Rs4e7T6NOTki5EmL7TUduTs6bu3EdnSycntVJ4re8kgZA+wx9IueI2Y11bfbgwtzuE0KQ==}
|
||||||
|
engines: {node: '>=0.10.0'}
|
||||||
|
|
||||||
|
tar-fs@2.1.4:
|
||||||
|
resolution: {integrity: sha512-mDAjwmZdh7LTT6pNleZ05Yt65HC3E+NiQzl672vQG38jIrehtJk/J3mNwIg+vShQPcLF/LV7CMnDW6vjj6sfYQ==}
|
||||||
|
|
||||||
|
tar-stream@2.2.0:
|
||||||
|
resolution: {integrity: sha512-ujeqbceABgwMZxEJnk2HDY2DlnUZ+9oEcb1KzTVfYHio0UE6dG71n60d8D2I4qNvleWrrXpmjpt7vZeF1LnMZQ==}
|
||||||
|
engines: {node: '>=6'}
|
||||||
|
|
||||||
|
tar@7.5.13:
|
||||||
|
resolution: {integrity: sha512-tOG/7GyXpFevhXVh8jOPJrmtRpOTsYqUIkVdVooZYJS/z8WhfQUX8RJILmeuJNinGAMSu1veBr4asSHFt5/hng==}
|
||||||
|
engines: {node: '>=18'}
|
||||||
|
|
||||||
|
tinyglobby@0.2.16:
|
||||||
|
resolution: {integrity: sha512-pn99VhoACYR8nFHhxqix+uvsbXineAasWm5ojXoN8xEwK5Kd3/TrhNn1wByuD52UxWRLy8pu+kRMniEi6Eq9Zg==}
|
||||||
|
engines: {node: '>=12.0.0'}
|
||||||
|
|
||||||
ts-node@10.9.2:
|
ts-node@10.9.2:
|
||||||
resolution: {integrity: sha512-f0FFpIdcHgn8zcPSbf1dRevwt047YMnaiJM3u2w2RewrB+fob/zePZcrOyQoLMMO7aBIddLcQIEK5dYjkLnGrQ==}
|
resolution: {integrity: sha512-f0FFpIdcHgn8zcPSbf1dRevwt047YMnaiJM3u2w2RewrB+fob/zePZcrOyQoLMMO7aBIddLcQIEK5dYjkLnGrQ==}
|
||||||
hasBin: true
|
hasBin: true
|
||||||
@ -85,6 +283,9 @@ packages:
|
|||||||
'@swc/wasm':
|
'@swc/wasm':
|
||||||
optional: true
|
optional: true
|
||||||
|
|
||||||
|
tunnel-agent@0.6.0:
|
||||||
|
resolution: {integrity: sha512-McnNiV1l8RYeY8tBgEpuodCC1mLUdbSN+CYBL7kJsJNInOP8UjDDEwdk6Mw60vdLLrr5NHKZhMAOSrR2NZuQ+w==}
|
||||||
|
|
||||||
typescript@5.9.3:
|
typescript@5.9.3:
|
||||||
resolution: {integrity: sha512-jl1vZzPDinLr9eUt3J/t7V6FgNEw9QjvBPdysz9KfQDD41fQrC2Y4vKQdiaUpFT4bXlb1RHhLpp8wtm6M5TgSw==}
|
resolution: {integrity: sha512-jl1vZzPDinLr9eUt3J/t7V6FgNEw9QjvBPdysz9KfQDD41fQrC2Y4vKQdiaUpFT4bXlb1RHhLpp8wtm6M5TgSw==}
|
||||||
engines: {node: '>=14.17'}
|
engines: {node: '>=14.17'}
|
||||||
@ -93,9 +294,28 @@ packages:
|
|||||||
undici-types@7.19.2:
|
undici-types@7.19.2:
|
||||||
resolution: {integrity: sha512-qYVnV5OEm2AW8cJMCpdV20CDyaN3g0AjDlOGf1OW4iaDEx8MwdtChUp4zu4H0VP3nDRF/8RKWH+IPp9uW0YGZg==}
|
resolution: {integrity: sha512-qYVnV5OEm2AW8cJMCpdV20CDyaN3g0AjDlOGf1OW4iaDEx8MwdtChUp4zu4H0VP3nDRF/8RKWH+IPp9uW0YGZg==}
|
||||||
|
|
||||||
|
undici@6.25.0:
|
||||||
|
resolution: {integrity: sha512-ZgpWDC5gmNiuY9CnLVXEH8rl50xhRCuLNA97fAUnKi8RRuV4E6KG31pDTsLVUKnohJE0I3XDrTeEydAXRw47xg==}
|
||||||
|
engines: {node: '>=18.17'}
|
||||||
|
|
||||||
|
util-deprecate@1.0.2:
|
||||||
|
resolution: {integrity: sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==}
|
||||||
|
|
||||||
v8-compile-cache-lib@3.0.1:
|
v8-compile-cache-lib@3.0.1:
|
||||||
resolution: {integrity: sha512-wa7YjyUGfNZngI/vtK0UHAN+lgDCxBPCylVXGp0zu59Fz5aiGtNXaq3DhIov063MorB+VfufLh3JlF2KdTK3xg==}
|
resolution: {integrity: sha512-wa7YjyUGfNZngI/vtK0UHAN+lgDCxBPCylVXGp0zu59Fz5aiGtNXaq3DhIov063MorB+VfufLh3JlF2KdTK3xg==}
|
||||||
|
|
||||||
|
which@6.0.1:
|
||||||
|
resolution: {integrity: sha512-oGLe46MIrCRqX7ytPUf66EAYvdeMIZYn3WaocqqKZAxrBpkqHfL/qvTyJ/bTk5+AqHCjXmrv3CEWgy368zhRUg==}
|
||||||
|
engines: {node: ^20.17.0 || >=22.9.0}
|
||||||
|
hasBin: true
|
||||||
|
|
||||||
|
wrappy@1.0.2:
|
||||||
|
resolution: {integrity: sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==}
|
||||||
|
|
||||||
|
yallist@5.0.0:
|
||||||
|
resolution: {integrity: sha512-YgvUTfwqyc7UXVMrB+SImsVYSmTS8X/tSrtdNZMImM+n7+QTriRXyXim0mBrTXNeqzVF0KWGgHPeiyViFFrNDw==}
|
||||||
|
engines: {node: '>=18'}
|
||||||
|
|
||||||
yn@3.1.1:
|
yn@3.1.1:
|
||||||
resolution: {integrity: sha512-Ux4ygGWsu2c7isFWe8Yu1YluJmqVhxqK2cLXNQA5AcC3QfbGNpM7fu0Y8b/z16pXLnFxZYvWhd3fhBY9DLmC6Q==}
|
resolution: {integrity: sha512-Ux4ygGWsu2c7isFWe8Yu1YluJmqVhxqK2cLXNQA5AcC3QfbGNpM7fu0Y8b/z16pXLnFxZYvWhd3fhBY9DLmC6Q==}
|
||||||
engines: {node: '>=6'}
|
engines: {node: '>=6'}
|
||||||
@ -106,6 +326,10 @@ snapshots:
|
|||||||
dependencies:
|
dependencies:
|
||||||
'@jridgewell/trace-mapping': 0.3.9
|
'@jridgewell/trace-mapping': 0.3.9
|
||||||
|
|
||||||
|
'@isaacs/fs-minipass@4.0.1':
|
||||||
|
dependencies:
|
||||||
|
minipass: 7.1.3
|
||||||
|
|
||||||
'@jridgewell/resolve-uri@3.1.2': {}
|
'@jridgewell/resolve-uri@3.1.2': {}
|
||||||
|
|
||||||
'@jridgewell/sourcemap-codec@1.5.5': {}
|
'@jridgewell/sourcemap-codec@1.5.5': {}
|
||||||
@ -127,6 +351,9 @@ snapshots:
|
|||||||
dependencies:
|
dependencies:
|
||||||
undici-types: 7.19.2
|
undici-types: 7.19.2
|
||||||
|
|
||||||
|
abbrev@4.0.0:
|
||||||
|
optional: true
|
||||||
|
|
||||||
acorn-walk@8.3.5:
|
acorn-walk@8.3.5:
|
||||||
dependencies:
|
dependencies:
|
||||||
acorn: 8.16.0
|
acorn: 8.16.0
|
||||||
@ -135,12 +362,214 @@ snapshots:
|
|||||||
|
|
||||||
arg@4.1.3: {}
|
arg@4.1.3: {}
|
||||||
|
|
||||||
|
base64-js@1.5.1: {}
|
||||||
|
|
||||||
|
bindings@1.5.0:
|
||||||
|
dependencies:
|
||||||
|
file-uri-to-path: 1.0.0
|
||||||
|
|
||||||
|
bl@4.1.0:
|
||||||
|
dependencies:
|
||||||
|
buffer: 5.7.1
|
||||||
|
inherits: 2.0.4
|
||||||
|
readable-stream: 3.6.2
|
||||||
|
|
||||||
|
buffer@5.7.1:
|
||||||
|
dependencies:
|
||||||
|
base64-js: 1.5.1
|
||||||
|
ieee754: 1.2.1
|
||||||
|
|
||||||
|
chownr@1.1.4: {}
|
||||||
|
|
||||||
|
chownr@3.0.0: {}
|
||||||
|
|
||||||
create-require@1.1.1: {}
|
create-require@1.1.1: {}
|
||||||
|
|
||||||
|
decompress-response@6.0.0:
|
||||||
|
dependencies:
|
||||||
|
mimic-response: 3.1.0
|
||||||
|
|
||||||
|
deep-extend@0.6.0: {}
|
||||||
|
|
||||||
|
detect-libc@2.1.2: {}
|
||||||
|
|
||||||
diff@4.0.4: {}
|
diff@4.0.4: {}
|
||||||
|
|
||||||
|
end-of-stream@1.4.5:
|
||||||
|
dependencies:
|
||||||
|
once: 1.4.0
|
||||||
|
|
||||||
|
env-paths@2.2.1:
|
||||||
|
optional: true
|
||||||
|
|
||||||
|
expand-template@2.0.3: {}
|
||||||
|
|
||||||
|
exponential-backoff@3.1.3:
|
||||||
|
optional: true
|
||||||
|
|
||||||
|
fdir@6.5.0(picomatch@4.0.4):
|
||||||
|
optionalDependencies:
|
||||||
|
picomatch: 4.0.4
|
||||||
|
optional: true
|
||||||
|
|
||||||
|
file-uri-to-path@1.0.0: {}
|
||||||
|
|
||||||
|
fs-constants@1.0.0: {}
|
||||||
|
|
||||||
|
github-from-package@0.0.0: {}
|
||||||
|
|
||||||
|
graceful-fs@4.2.11:
|
||||||
|
optional: true
|
||||||
|
|
||||||
|
ieee754@1.2.1: {}
|
||||||
|
|
||||||
|
inherits@2.0.4: {}
|
||||||
|
|
||||||
|
ini@1.3.8: {}
|
||||||
|
|
||||||
|
isexe@4.0.0:
|
||||||
|
optional: true
|
||||||
|
|
||||||
make-error@1.3.6: {}
|
make-error@1.3.6: {}
|
||||||
|
|
||||||
|
mimic-response@3.1.0: {}
|
||||||
|
|
||||||
|
minimist@1.2.8: {}
|
||||||
|
|
||||||
|
minipass@7.1.3: {}
|
||||||
|
|
||||||
|
minizlib@3.1.0:
|
||||||
|
dependencies:
|
||||||
|
minipass: 7.1.3
|
||||||
|
|
||||||
|
mkdirp-classic@0.5.3: {}
|
||||||
|
|
||||||
|
napi-build-utils@2.0.0: {}
|
||||||
|
|
||||||
|
node-abi@3.89.0:
|
||||||
|
dependencies:
|
||||||
|
semver: 7.7.4
|
||||||
|
|
||||||
|
node-addon-api@8.7.0: {}
|
||||||
|
|
||||||
|
node-gyp@12.3.0:
|
||||||
|
dependencies:
|
||||||
|
env-paths: 2.2.1
|
||||||
|
exponential-backoff: 3.1.3
|
||||||
|
graceful-fs: 4.2.11
|
||||||
|
nopt: 9.0.0
|
||||||
|
proc-log: 6.1.0
|
||||||
|
semver: 7.7.4
|
||||||
|
tar: 7.5.13
|
||||||
|
tinyglobby: 0.2.16
|
||||||
|
undici: 6.25.0
|
||||||
|
which: 6.0.1
|
||||||
|
optional: true
|
||||||
|
|
||||||
|
nopt@9.0.0:
|
||||||
|
dependencies:
|
||||||
|
abbrev: 4.0.0
|
||||||
|
optional: true
|
||||||
|
|
||||||
|
once@1.4.0:
|
||||||
|
dependencies:
|
||||||
|
wrappy: 1.0.2
|
||||||
|
|
||||||
|
picomatch@4.0.4:
|
||||||
|
optional: true
|
||||||
|
|
||||||
|
prebuild-install@7.1.3:
|
||||||
|
dependencies:
|
||||||
|
detect-libc: 2.1.2
|
||||||
|
expand-template: 2.0.3
|
||||||
|
github-from-package: 0.0.0
|
||||||
|
minimist: 1.2.8
|
||||||
|
mkdirp-classic: 0.5.3
|
||||||
|
napi-build-utils: 2.0.0
|
||||||
|
node-abi: 3.89.0
|
||||||
|
pump: 3.0.4
|
||||||
|
rc: 1.2.8
|
||||||
|
simple-get: 4.0.1
|
||||||
|
tar-fs: 2.1.4
|
||||||
|
tunnel-agent: 0.6.0
|
||||||
|
|
||||||
|
proc-log@6.1.0:
|
||||||
|
optional: true
|
||||||
|
|
||||||
|
pump@3.0.4:
|
||||||
|
dependencies:
|
||||||
|
end-of-stream: 1.4.5
|
||||||
|
once: 1.4.0
|
||||||
|
|
||||||
|
rc@1.2.8:
|
||||||
|
dependencies:
|
||||||
|
deep-extend: 0.6.0
|
||||||
|
ini: 1.3.8
|
||||||
|
minimist: 1.2.8
|
||||||
|
strip-json-comments: 2.0.1
|
||||||
|
|
||||||
|
readable-stream@3.6.2:
|
||||||
|
dependencies:
|
||||||
|
inherits: 2.0.4
|
||||||
|
string_decoder: 1.3.0
|
||||||
|
util-deprecate: 1.0.2
|
||||||
|
|
||||||
|
safe-buffer@5.2.1: {}
|
||||||
|
|
||||||
|
semver@7.7.4: {}
|
||||||
|
|
||||||
|
simple-concat@1.0.1: {}
|
||||||
|
|
||||||
|
simple-get@4.0.1:
|
||||||
|
dependencies:
|
||||||
|
decompress-response: 6.0.0
|
||||||
|
once: 1.4.0
|
||||||
|
simple-concat: 1.0.1
|
||||||
|
|
||||||
|
sqlite3@6.0.1:
|
||||||
|
dependencies:
|
||||||
|
bindings: 1.5.0
|
||||||
|
node-addon-api: 8.7.0
|
||||||
|
prebuild-install: 7.1.3
|
||||||
|
tar: 7.5.13
|
||||||
|
optionalDependencies:
|
||||||
|
node-gyp: 12.3.0
|
||||||
|
|
||||||
|
string_decoder@1.3.0:
|
||||||
|
dependencies:
|
||||||
|
safe-buffer: 5.2.1
|
||||||
|
|
||||||
|
strip-json-comments@2.0.1: {}
|
||||||
|
|
||||||
|
tar-fs@2.1.4:
|
||||||
|
dependencies:
|
||||||
|
chownr: 1.1.4
|
||||||
|
mkdirp-classic: 0.5.3
|
||||||
|
pump: 3.0.4
|
||||||
|
tar-stream: 2.2.0
|
||||||
|
|
||||||
|
tar-stream@2.2.0:
|
||||||
|
dependencies:
|
||||||
|
bl: 4.1.0
|
||||||
|
end-of-stream: 1.4.5
|
||||||
|
fs-constants: 1.0.0
|
||||||
|
inherits: 2.0.4
|
||||||
|
readable-stream: 3.6.2
|
||||||
|
|
||||||
|
tar@7.5.13:
|
||||||
|
dependencies:
|
||||||
|
'@isaacs/fs-minipass': 4.0.1
|
||||||
|
chownr: 3.0.0
|
||||||
|
minipass: 7.1.3
|
||||||
|
minizlib: 3.1.0
|
||||||
|
yallist: 5.0.0
|
||||||
|
|
||||||
|
tinyglobby@0.2.16:
|
||||||
|
dependencies:
|
||||||
|
fdir: 6.5.0(picomatch@4.0.4)
|
||||||
|
picomatch: 4.0.4
|
||||||
|
optional: true
|
||||||
|
|
||||||
ts-node@10.9.2(@types/node@25.6.0)(typescript@5.9.3):
|
ts-node@10.9.2(@types/node@25.6.0)(typescript@5.9.3):
|
||||||
dependencies:
|
dependencies:
|
||||||
'@cspotcode/source-map-support': 0.8.1
|
'@cspotcode/source-map-support': 0.8.1
|
||||||
@ -159,10 +588,28 @@ snapshots:
|
|||||||
v8-compile-cache-lib: 3.0.1
|
v8-compile-cache-lib: 3.0.1
|
||||||
yn: 3.1.1
|
yn: 3.1.1
|
||||||
|
|
||||||
|
tunnel-agent@0.6.0:
|
||||||
|
dependencies:
|
||||||
|
safe-buffer: 5.2.1
|
||||||
|
|
||||||
typescript@5.9.3: {}
|
typescript@5.9.3: {}
|
||||||
|
|
||||||
undici-types@7.19.2: {}
|
undici-types@7.19.2: {}
|
||||||
|
|
||||||
|
undici@6.25.0:
|
||||||
|
optional: true
|
||||||
|
|
||||||
|
util-deprecate@1.0.2: {}
|
||||||
|
|
||||||
v8-compile-cache-lib@3.0.1: {}
|
v8-compile-cache-lib@3.0.1: {}
|
||||||
|
|
||||||
|
which@6.0.1:
|
||||||
|
dependencies:
|
||||||
|
isexe: 4.0.0
|
||||||
|
optional: true
|
||||||
|
|
||||||
|
wrappy@1.0.2: {}
|
||||||
|
|
||||||
|
yallist@5.0.0: {}
|
||||||
|
|
||||||
yn@3.1.1: {}
|
yn@3.1.1: {}
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user