From 9c8724f0cb6e223cd0413e7f0221a7d324d9be1b Mon Sep 17 00:00:00 2001
From: Sun_sun <469361609@qq.com>
Date: Thu, 18 Sep 2025 17:12:46 +0800
Subject: [PATCH] =?UTF-8?q?2025-09-18=20app=E5=90=8E=E7=AB=AF=E6=90=AD?=
=?UTF-8?q?=E5=BB=BA?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
.gitignore | 1 +
.idea/.gitignore | 8 +
.idea/dataSources.xml | 12 +
.idea/data_source_mapping.xml | 6 +
.idea/jsLibraryMappings.xml | 6 +
.idea/middle_office_system.iml | 12 +
.idea/modules.xml | 8 +
.idea/runConfigurations/bin_www.xml | 11 +
.idea/sqldialects.xml | 13 +
.idea/vcs.xml | 6 +
app.js | 51 +
bin/www | 96 ++
config/config.js | 17 +
config/constants.js | 70 ++
config/database-init.js | 38 +
config/dbv2.js | 363 +++++++
config/logger.js | 73 ++
config/minio.js | 97 ++
config/swagger.js | 43 +
database.js | 159 +++
package-lock.json | 1559 +++++++++++++++++++++++++++
package.json | 21 +
public/index.html | 13 +
public/stylesheets/style.css | 8 +
routes/auth.js | 346 ++++++
routes/captcha.js | 220 ++++
routes/common.js | 77 ++
routes/index.js | 6 +
routes/users.js | 9 +
29 files changed, 3349 insertions(+)
create mode 100644 .gitignore
create mode 100644 .idea/.gitignore
create mode 100644 .idea/dataSources.xml
create mode 100644 .idea/data_source_mapping.xml
create mode 100644 .idea/jsLibraryMappings.xml
create mode 100644 .idea/middle_office_system.iml
create mode 100644 .idea/modules.xml
create mode 100644 .idea/runConfigurations/bin_www.xml
create mode 100644 .idea/sqldialects.xml
create mode 100644 .idea/vcs.xml
create mode 100644 app.js
create mode 100644 bin/www
create mode 100644 config/config.js
create mode 100644 config/constants.js
create mode 100644 config/database-init.js
create mode 100644 config/dbv2.js
create mode 100644 config/logger.js
create mode 100644 config/minio.js
create mode 100644 config/swagger.js
create mode 100644 database.js
create mode 100644 package-lock.json
create mode 100644 package.json
create mode 100644 public/index.html
create mode 100644 public/stylesheets/style.css
create mode 100644 routes/auth.js
create mode 100644 routes/captcha.js
create mode 100644 routes/common.js
create mode 100644 routes/index.js
create mode 100644 routes/users.js
diff --git a/.gitignore b/.gitignore
new file mode 100644
index 0000000..2ccbe46
--- /dev/null
+++ b/.gitignore
@@ -0,0 +1 @@
+/node_modules/
diff --git a/.idea/.gitignore b/.idea/.gitignore
new file mode 100644
index 0000000..35410ca
--- /dev/null
+++ b/.idea/.gitignore
@@ -0,0 +1,8 @@
+# 默认忽略的文件
+/shelf/
+/workspace.xml
+# 基于编辑器的 HTTP 客户端请求
+/httpRequests/
+# Datasource local storage ignored files
+/dataSources/
+/dataSources.local.xml
diff --git a/.idea/dataSources.xml b/.idea/dataSources.xml
new file mode 100644
index 0000000..947b8d8
--- /dev/null
+++ b/.idea/dataSources.xml
@@ -0,0 +1,12 @@
+
+
+
+
+ mysql.8
+ true
+ com.mysql.cj.jdbc.Driver
+ jdbc:mysql://114.55.111.44:3306/test_mao
+ $ProjectFileDir$
+
+
+
\ No newline at end of file
diff --git a/.idea/data_source_mapping.xml b/.idea/data_source_mapping.xml
new file mode 100644
index 0000000..4961f08
--- /dev/null
+++ b/.idea/data_source_mapping.xml
@@ -0,0 +1,6 @@
+
+
+
+
+
+
\ No newline at end of file
diff --git a/.idea/jsLibraryMappings.xml b/.idea/jsLibraryMappings.xml
new file mode 100644
index 0000000..d23208f
--- /dev/null
+++ b/.idea/jsLibraryMappings.xml
@@ -0,0 +1,6 @@
+
+
+
+
+
+
\ No newline at end of file
diff --git a/.idea/middle_office_system.iml b/.idea/middle_office_system.iml
new file mode 100644
index 0000000..24643cc
--- /dev/null
+++ b/.idea/middle_office_system.iml
@@ -0,0 +1,12 @@
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/.idea/modules.xml b/.idea/modules.xml
new file mode 100644
index 0000000..2e08d98
--- /dev/null
+++ b/.idea/modules.xml
@@ -0,0 +1,8 @@
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/.idea/runConfigurations/bin_www.xml b/.idea/runConfigurations/bin_www.xml
new file mode 100644
index 0000000..bdd328f
--- /dev/null
+++ b/.idea/runConfigurations/bin_www.xml
@@ -0,0 +1,11 @@
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/.idea/sqldialects.xml b/.idea/sqldialects.xml
new file mode 100644
index 0000000..f0a2a08
--- /dev/null
+++ b/.idea/sqldialects.xml
@@ -0,0 +1,13 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/.idea/vcs.xml b/.idea/vcs.xml
new file mode 100644
index 0000000..94a25f7
--- /dev/null
+++ b/.idea/vcs.xml
@@ -0,0 +1,6 @@
+
+
+
+
+
+
\ No newline at end of file
diff --git a/app.js b/app.js
new file mode 100644
index 0000000..29182d7
--- /dev/null
+++ b/app.js
@@ -0,0 +1,51 @@
+var express = require('express');
+var path = require('path');
+var cookieParser = require('cookie-parser');
+var logger = require('morgan');
+const {specs, swaggerUi} = require('./config/swagger'); // 引入 swagger 配置
+const cors = require('cors');
+const bodyParser = require('body-parser');
+
+var indexRouter = require('./routes/index');
+var usersRouter = require('./routes/users');
+var commonRouter = require('./routes/common');
+var captchaRouter = require('./routes/captcha');
+var authRouter = require('./routes/auth');
+
+var app = express();
+
+// 中间件配置
+// CORS配置 - 允许前端访问静态资源
+app.use(cors({
+ origin: [
+ 'http://localhost:5173',
+ 'http://localhost:5176',
+ 'http://localhost:5175',
+ 'http://localhost:5174',
+ 'http://localhost:3001',
+ 'https://www.zrbjr.com',
+ 'https://zrbjr.com'
+ ],
+ credentials: true,
+ methods: ['GET', 'POST', 'PUT', 'DELETE', 'OPTIONS'],
+ allowedHeaders: ['Content-Type', 'Authorization', 'X-Requested-With']
+}));
+app.use(bodyParser.json({ limit: '10mb' }));
+app.use(bodyParser.urlencoded({ extended: true, limit: '10mb' }));
+
+
+app.use('/api-docs', swaggerUi.serve, swaggerUi.setup(specs)); // swagger
+
+app.use(logger('dev'));
+app.use(express.json());
+app.use(express.urlencoded({extended: false}));
+app.use(cookieParser());
+app.use(express.static(path.join(__dirname, 'public')));
+
+app.use('/users', usersRouter);
+app.use('/api/common', commonRouter)
+app.use('/api/captcha', captchaRouter)
+app.use('/api/auth', authRouter)
+
+
+module.exports = app;
diff --git a/bin/www b/bin/www
new file mode 100644
index 0000000..b5b0257
--- /dev/null
+++ b/bin/www
@@ -0,0 +1,96 @@
+#!/usr/bin/env node
+
+/**
+ * Module dependencies.
+ */
+
+var app = require('../app');
+var debug = require('debug')('middle-office-system:server');
+var http = require('http');
+
+// 引入数据库初始化模块
+const {initDatabase} = require('../config/database-init');
+
+/**
+ * Get port from environment and store in Express.
+ */
+
+var port = normalizePort(process.env.PORT || '5001');
+app.set('port', port);
+
+/**
+ * Create HTTP server.
+ */
+
+var server = http.createServer(app);
+
+/**
+ * Listen on provided port, on all network interfaces.
+ */
+
+server.listen(port, async () => {
+ await initDatabase();
+ global.captchaStore = new Map();
+});
+server.on('error', onError);
+server.on('listening', onListening);
+
+/**
+ * Normalize a port into a number, string, or false.
+ */
+
+function normalizePort(val) {
+ var port = parseInt(val, 10);
+
+ if (isNaN(port)) {
+ // named pipe
+ return val;
+ }
+
+ if (port >= 0) {
+ // port number
+ return port;
+ }
+
+ return false;
+}
+
+/**
+ * Event listener for HTTP server "error" event.
+ */
+
+function onError(error) {
+ if (error.syscall !== 'listen') {
+ throw error;
+ }
+
+ var bind = typeof port === 'string'
+ ? 'Pipe ' + port
+ : 'Port ' + port;
+
+ // handle specific listen errors with friendly messages
+ switch (error.code) {
+ case 'EACCES':
+ console.error(bind + ' requires elevated privileges');
+ process.exit(1);
+ break;
+ case 'EADDRINUSE':
+ console.error(bind + ' is already in use');
+ process.exit(1);
+ break;
+ default:
+ throw error;
+ }
+}
+
+/**
+ * Event listener for HTTP server "listening" event.
+ */
+
+function onListening() {
+ var addr = server.address();
+ var bind = typeof addr === 'string'
+ ? 'pipe ' + addr
+ : 'port ' + addr.port;
+ debug('Listening on ' + bind);
+}
diff --git a/config/config.js b/config/config.js
new file mode 100644
index 0000000..3099ef2
--- /dev/null
+++ b/config/config.js
@@ -0,0 +1,17 @@
+const mysql = require('mysql2')
+
+const sql = {
+ createConnection() {
+ return mysql.createPool({
+ connectionLimit: 10,
+ host: '114.55.111.44',
+ user: 'test_mao',
+ password: 'nK2mPbWriBp25BRd',
+ database: 'test_mao',
+ charset: 'utf8mb4',
+ multipleStatements: true
+
+ })
+ }
+}
+module.exports = sql
diff --git a/config/constants.js b/config/constants.js
new file mode 100644
index 0000000..3272b44
--- /dev/null
+++ b/config/constants.js
@@ -0,0 +1,70 @@
+// 系统常量配置
+module.exports = {
+ // 转账类型
+ TRANSFER_TYPES: {
+ USER_TO_USER: 'user_to_user',
+ SYSTEM_TO_USER: 'system_to_user',
+ USER_TO_SYSTEM: 'user_to_system'
+ },
+
+ // 转账状态
+ TRANSFER_STATUS: {
+ PENDING: 'pending',
+ CONFIRMED: 'confirmed',
+ RECEIVED: 'received',
+ REJECTED: 'rejected',
+ CANCELLED: 'cancelled',
+ NOT_RECEIVED: 'not_received',
+ FAILED: 'failed'
+ },
+
+ // 用户角色
+ USER_ROLES: {
+ ADMIN: 'admin',
+ USER: 'user'
+ },
+
+ // 订单状态
+ ORDER_STATUS: {
+ PENDING: 'pending',
+ PAID: 'paid',
+ SHIPPED: 'shipped',
+ DELIVERED: 'delivered',
+ CANCELLED: 'cancelled'
+ },
+
+ // 错误代码
+ ERROR_CODES: {
+ VALIDATION_ERROR: 'VALIDATION_ERROR',
+ AUTHENTICATION_ERROR: 'AUTHENTICATION_ERROR',
+ AUTHORIZATION_ERROR: 'AUTHORIZATION_ERROR',
+ NOT_FOUND: 'NOT_FOUND',
+ DUPLICATE_ENTRY: 'DUPLICATE_ENTRY',
+ DATABASE_ERROR: 'DATABASE_ERROR',
+ INTERNAL_ERROR: 'INTERNAL_ERROR'
+ },
+
+ // HTTP状态码
+ HTTP_STATUS: {
+ OK: 200,
+ CREATED: 201,
+ BAD_REQUEST: 400,
+ UNAUTHORIZED: 401,
+ FORBIDDEN: 403,
+ NOT_FOUND: 404,
+ CONFLICT: 409,
+ INTERNAL_SERVER_ERROR: 500
+ },
+
+ // 分页默认值
+ PAGINATION: {
+ DEFAULT_PAGE: 1,
+ DEFAULT_LIMIT: 10,
+ MAX_LIMIT: 100
+ },
+
+ // JWT配置
+ JWT: {
+ EXPIRES_IN: '24h'
+ }
+};
\ No newline at end of file
diff --git a/config/database-init.js b/config/database-init.js
new file mode 100644
index 0000000..a0481bf
--- /dev/null
+++ b/config/database-init.js
@@ -0,0 +1,38 @@
+const mysql = require('mysql2/promise');
+const bcrypt = require('bcryptjs');
+const { initDB, getDB, dbConfig } = require('../database');
+
+/**
+ * 数据库初始化函数
+ * 创建所有必要的表结构和初始数据
+ */
+async function initDatabase() {
+ try {
+
+
+ // 初始化数据库连接池
+ await initDB();
+ console.log('数据库连接池初始化成功');
+
+ // 创建所有表
+ // await createTables();
+
+ // 添加字段(处理表结构升级)
+ // await addMissingFields();
+
+ // 创建默认数据
+ // await createDefaultData();
+
+ console.log('数据库初始化完成');
+ } catch (error) {
+ console.error('数据库初始化失败:', error);
+ throw error;
+ }
+}
+
+
+
+
+module.exports = {
+ initDatabase
+};
\ No newline at end of file
diff --git a/config/dbv2.js b/config/dbv2.js
new file mode 100644
index 0000000..e7e253b
--- /dev/null
+++ b/config/dbv2.js
@@ -0,0 +1,363 @@
+class QueryBuilder {
+ constructor() {
+ this.conditions = {};
+ this.limit = null;
+ this.offset = null;
+ this.groupBy = null;
+ }
+
+ where(condition, ...params) {
+ this.conditions[condition] = params;
+ return this;
+ }
+
+ setLimit(limit) {
+ this.limit = limit;
+ return this;
+ }
+
+ setOffset(offset) {
+ this.offset = offset;
+ return this;
+ }
+
+ setGroupBy(groupBy) {
+ this.groupBy = groupBy;
+ return this;
+ }
+
+ sqdata(sql, params) {
+ return new Promise((resolve, reject) => {
+ global.sqlReq.query(sql, params, (err, result) => {
+ if (err) {
+ reject(err);
+ }
+ resolve(result);
+ });
+ });
+ }
+
+ getParams() {
+ return Object.values(this.conditions).flat();
+ }
+
+ buildConditions() {
+ return Object.keys(this.conditions).map(condition => `${condition}`).join(' AND ');
+ }
+}
+
+class SelectBuilder extends QueryBuilder {
+ constructor() {
+ super();
+ this.selectFields = [];
+ this.tables = [];
+ this.orderByField = '';
+ this.orderByDirection = 'ASC';
+ this.subQueries = []; // 用于存储子查询
+ this.unions = []; // 存储UNION查询
+ }
+ // 添加UNION查询
+ union(queryBuilder, type = 'UNION') {
+ this.unions.push({queryBuilder, type});
+ return this;
+ }
+
+ // 添加UNION ALL查询
+ unionAll(queryBuilder) {
+ this.union(queryBuilder, 'UNION ALL');
+ return this;
+ }
+
+ // 构建主查询部分(不含ORDER BY/LIMIT/OFFSET)
+ buildMainQuery() {
+ const subQuerySQL = this.subQueries.map(({alias, subQuery}) => `(${subQuery}) AS ${alias}`);
+ const selectClause = this.selectFields.concat(subQuerySQL).join(', ');
+
+ let sql = `SELECT ${selectClause}
+ FROM ${this.tables.join(' ')}`;
+
+ const conditionClauses = this.buildConditions();
+ if (conditionClauses) {
+ sql += ` WHERE ${conditionClauses}`;
+ }
+
+ if (this.groupBy) {
+ sql += ` GROUP BY ${this.groupBy}`;
+ }
+
+ const params = this.getParams();
+ return {sql, params};
+ }
+
+ // 供UNION查询调用的构建方法
+ buildForUnion() {
+ return this.buildMainQuery();
+ }
+
+ select(fields) {
+ this.selectFields = fields.split(',').map(field => field.trim());
+ return this;
+ }
+
+ // 添加子查询
+ addSubQuery(alias, subQuery) {
+ this.subQueries.push({alias, subQuery});
+ return this;
+ }
+
+ whereLike(fields, keyword) {
+ const likeConditions = fields.map(field => `${field} LIKE ?`).join(' OR ');
+ this.conditions[likeConditions] = fields.map(() => `%${keyword}%`);
+ return this;
+ }
+
+ from(table) {
+ this.tables.push(table);
+ return this;
+ }
+
+ leftJoin(table, condition) {
+ this.tables.push(`LEFT JOIN ${table} ON ${condition}`);
+ return this;
+ }
+
+ orderBy(field, direction = 'ASC') {
+ this.orderByField = field;
+ this.orderByDirection = direction.toUpperCase();
+ return this;
+ }
+
+ paginate(page, pageSize) {
+ if (page <= 0 || pageSize <= 0) {
+ throw new Error('分页参数必须大于0');
+ }
+ this.limit = pageSize;
+ this.offset = (page - 1) * pageSize;
+ return this;
+ }
+
+ async chidBuild() {
+
+ let sql = `SELECT ${this.selectFields.join(', ')}
+ FROM ${this.tables.join(' ')}`;
+ let conditionClauses = this.buildConditions();
+ if (conditionClauses) {
+ sql += ` WHERE ${conditionClauses}`;
+ }
+ if (this.orderByField) {
+ sql += ` ORDER BY ${this.orderByField} ${this.orderByDirection}`;
+ }
+ if (this.limit !== null) {
+ sql += ` LIMIT ${this.limit}`;
+ }
+ if (this.offset !== null) {
+ sql += ` OFFSET ${this.offset}`;
+ }
+ return sql;
+ }
+
+ async build() {
+ const main = this.buildMainQuery();
+ let fullSql = `(${main.sql})`;
+ const allParams = [...main.params];
+
+ // 处理UNION部分
+ for (const union of this.unions) {
+ const unionBuilder = union.queryBuilder;
+ if (!(unionBuilder instanceof SelectBuilder)) {
+ throw new Error('UNION query must be a SelectBuilder instance');
+ }
+ const unionResult = unionBuilder.buildForUnion();
+ fullSql += ` ${union.type} (${unionResult.sql})`;
+ allParams.push(...unionResult.params);
+ }
+
+ // 添加ORDER BY、LIMIT、OFFSET
+ if (this.orderByField) {
+ fullSql += ` ORDER BY ${this.orderByField} ${this.orderByDirection}`;
+ }
+ if (this.limit !== null) {
+ fullSql += ` LIMIT ${this.limit}`;
+ }
+ if (this.offset !== null) {
+ fullSql += ` OFFSET ${this.offset}`;
+ }
+ console.log(fullSql,allParams);
+ return await this.sqdata(fullSql, allParams);
+ }
+}
+
+
+class UpdateBuilder extends QueryBuilder {
+ constructor() {
+ super();
+ this.table = '';
+ this.updateFields = {};
+ }
+
+ update(table) {
+ this.table = table;
+ return this;
+ }
+
+ set(field, value) {
+ if (value && value.increment && typeof value === 'object' ) {
+ this.updateFields[field] = {increment: value.increment};
+ } else {
+ this.updateFields[field] = value;
+ }
+ return this;
+ }
+
+ async build() {
+ let sql = `UPDATE ${this.table}
+ SET `;
+ let updateClauses = Object.keys(this.updateFields).map(field => {
+ const value = this.updateFields[field];
+ if (value && value.increment && typeof value === 'object' ) {
+ return `${field} = ${field} + ?`;
+ }
+ return `${field} = ?`;
+ }).join(', ');
+
+ sql += updateClauses;
+
+ let conditionClauses = this.buildConditions();
+ if (conditionClauses) {
+ sql += ` WHERE ${conditionClauses}`;
+ }
+ // 处理参数,确保自增字段也传入增量值
+ const params = [
+ ...Object.values(this.updateFields).map(value =>
+ (value && value.increment && typeof value === 'object' ) ? value.increment : value
+ ),
+ ...this.getParams()
+ ];
+ return await this.sqdata(sql, params);
+ }
+}
+
+class InsertBuilder extends QueryBuilder {
+ constructor() {
+ super();
+ this.table = '';
+ this.insertValues = [];
+ this.updateValues = {};
+ }
+
+ insertInto(table) {
+ this.table = table;
+ return this;
+ }
+
+ // 仍然保留单条记录的插入
+ values(values) {
+ if (Array.isArray(values)) {
+ this.insertValues = values;
+ } else {
+ this.insertValues = [values]; // 将单条记录包装成数组
+ }
+ return this;
+ }
+
+ // 新增方法,支持一次插入多条记录
+ valuesMultiple(records) {
+ if (!Array.isArray(records) || records.length === 0) {
+ throw new Error('Values must be a non-empty array');
+ }
+
+ // 确保每一条记录都是对象
+ records.forEach(record => {
+ if (typeof record !== 'object') {
+ throw new Error('Each record must be an object');
+ }
+ });
+
+ this.insertValues = records;
+ return this;
+ }
+
+ // 新增 upsert 方法,支持更新或插入
+ upsert(values, updateFields) {
+ // values: 要插入的记录
+ // updateFields: 如果记录存在时,需要更新的字段
+ if (!Array.isArray(values) || values.length === 0) {
+ throw new Error('Values must be a non-empty array');
+ }
+
+ // 检查每条记录是否是对象
+ values.forEach(record => {
+ if (typeof record !== 'object') {
+ throw new Error('Each record must be an object');
+ }
+ });
+
+ this.insertValues = values;
+ this.updateValues = updateFields || {};
+ return this;
+ }
+
+ async build() {
+ if (this.insertValues.length === 0) {
+ throw new Error("No values to insert");
+ }
+
+ // 获取表单列名,假设所有记录有相同的字段
+ const columns = Object.keys(this.insertValues[0]);
+
+ // 构建 VALUES 子句,支持批量插入
+ const valuePlaceholders = this.insertValues.map(() =>
+ `(${columns.map(() => '?').join(', ')})`
+ ).join(', ');
+
+ // 展平所有的插入值
+ const params = this.insertValues.flatMap(record =>
+ columns.map(column => record[column])
+ );
+
+ // 如果有 updateFields,构建 ON DUPLICATE KEY UPDATE 子句
+ let updateClause = '';
+ if (Object.keys(this.updateValues).length > 0) {
+ updateClause = ' ON DUPLICATE KEY UPDATE ' +
+ Object.keys(this.updateValues).map(field => {
+ return `${field} = VALUES(${field})`;
+ }).join(', ');
+ }
+
+ // 生成 SQL 语句
+ const sql = `INSERT INTO ${this.table} (${columns.join(', ')})
+ VALUES ${valuePlaceholders} ${updateClause}`;
+ // 执行查询
+ return await this.sqdata(sql, params);
+ }
+}
+
+
+class DeleteBuilder extends QueryBuilder {
+ constructor() {
+ super();
+ this.table = '';
+ }
+
+ deleteFrom(table) {
+ this.table = table;
+ return this;
+ }
+
+ async build() {
+ let sql = `DELETE
+ FROM ${this.table}`;
+ let conditionClauses = this.buildConditions();
+ if (conditionClauses) {
+ sql += ` WHERE ${conditionClauses}`;
+ }
+ return await this.sqdata(sql, this.getParams());
+ }
+}
+
+module.exports = {
+ SelectBuilder,
+ UpdateBuilder,
+ InsertBuilder,
+ DeleteBuilder,
+};
diff --git a/config/logger.js b/config/logger.js
new file mode 100644
index 0000000..45cda51
--- /dev/null
+++ b/config/logger.js
@@ -0,0 +1,73 @@
+const winston = require('winston');
+const path = require('path');
+
+// 创建日志目录
+const logDir = path.join(__dirname, '../logs');
+
+// 日志格式配置
+const logFormat = winston.format.combine(
+ winston.format.timestamp({
+ format: 'YYYY-MM-DD HH:mm:ss'
+ }),
+ winston.format.errors({ stack: true }),
+ winston.format.json()
+);
+
+// 控制台日志格式
+const consoleFormat = winston.format.combine(
+ winston.format.colorize(),
+ winston.format.timestamp({
+ format: 'YYYY-MM-DD HH:mm:ss'
+ }),
+ winston.format.printf(({ timestamp, level, message, ...meta }) => {
+ return `${timestamp} [${level}]: ${message} ${Object.keys(meta).length ? JSON.stringify(meta, null, 2) : ''}`;
+ })
+);
+
+// 创建logger实例
+const logger = winston.createLogger({
+ level: process.env.LOG_LEVEL || 'info',
+ format: logFormat,
+ defaultMeta: { service: 'integrated-system' },
+ transports: [
+ // 错误日志文件
+ new winston.transports.File({
+ filename: path.join(logDir, 'error.log'),
+ level: 'error',
+ maxsize: 5242880, // 5MB
+ maxFiles: 5
+ }),
+ // 所有日志文件
+ new winston.transports.File({
+ filename: path.join(logDir, 'combined.log'),
+ maxsize: 5242880, // 5MB
+ maxFiles: 5
+ })
+ ]
+});
+
+// 开发环境添加控制台输出
+if (process.env.NODE_ENV !== 'production') {
+ logger.add(new winston.transports.Console({
+ format: consoleFormat
+ }));
+}
+
+// 审计日志记录器
+const auditLogger = winston.createLogger({
+ level: 'info',
+ format: logFormat,
+ defaultMeta: { service: 'audit' },
+ transports: [
+ new winston.transports.File({
+ filename: path.join(logDir, 'audit.log'),
+ maxsize: 5242880, // 5MB
+ maxFiles: 10
+ })
+ ]
+});
+
+module.exports = {
+ logger,
+ auditLogger
+};
\ No newline at end of file
diff --git a/config/minio.js b/config/minio.js
new file mode 100644
index 0000000..f479bea
--- /dev/null
+++ b/config/minio.js
@@ -0,0 +1,97 @@
+const Minio = require('minio');
+require('dotenv').config();
+
+/**
+ * MinIO 配置
+ * 用于对象存储服务配置
+ */
+const minioConfig = {
+ // MinIO 服务器配置
+ endPoint: process.env.MINIO_ENDPOINT || 'localhost',
+ port: parseInt(process.env.MINIO_PORT) || 9000,
+ useSSL: process.env.MINIO_USE_SSL === 'true' || false,
+ accessKey: process.env.MINIO_ACCESS_KEY || 'minioadmin',
+ secretKey: process.env.MINIO_SECRET_KEY || 'minioadmin',
+
+ // 存储桶配置
+ buckets: {
+ uploads: process.env.MINIO_BUCKET_UPLOADS || 'uploads',
+ avatars: process.env.MINIO_BUCKET_AVATARS || 'avatars',
+ products: process.env.MINIO_BUCKET_PRODUCTS || 'products',
+ documents: process.env.MINIO_BUCKET_DOCUMENTS || 'documents'
+ },
+
+ // 文件访问配置
+ publicUrl: process.env.MINIO_PUBLIC_URL || `http://localhost:9000`
+};
+
+/**
+ * 创建 MinIO 客户端实例
+ */
+const createMinioClient = () => {
+ return new Minio.Client({
+ endPoint: minioConfig.endPoint,
+ port: minioConfig.port,
+ useSSL: minioConfig.useSSL,
+ accessKey: minioConfig.accessKey,
+ secretKey: minioConfig.secretKey
+ });
+};
+
+/**
+ * 初始化存储桶
+ * 确保所有需要的存储桶都存在
+ */
+const initializeBuckets = async () => {
+ const minioClient = createMinioClient();
+
+ try {
+ // 检查并创建存储桶
+ for (const [key, bucketName] of Object.entries(minioConfig.buckets)) {
+ const exists = await minioClient.bucketExists(bucketName);
+ if (!exists) {
+ await minioClient.makeBucket(bucketName, 'us-east-1');
+ console.log(`✅ 存储桶 '${bucketName}' 创建成功`);
+
+ // 设置存储桶策略为公开读取(可选)
+ const policy = {
+ Version: '2012-10-17',
+ Statement: [
+ {
+ Effect: 'Allow',
+ Principal: { AWS: ['*'] },
+ Action: ['s3:GetObject'],
+ Resource: [`arn:aws:s3:::${bucketName}/*`]
+ }
+ ]
+ };
+
+ try {
+ await minioClient.setBucketPolicy(bucketName, JSON.stringify(policy));
+ console.log(`✅ 存储桶 '${bucketName}' 策略设置成功`);
+ } catch (policyError) {
+ console.warn(`⚠️ 存储桶 '${bucketName}' 策略设置失败:`, policyError.message);
+ }
+ } else {
+ console.log(`✅ 存储桶 '${bucketName}' 已存在`);
+ }
+ }
+ } catch (error) {
+ console.error('❌ 初始化存储桶失败:', error);
+ throw error;
+ }
+};
+
+/**
+ * 获取文件的公开访问URL
+ */
+const getPublicUrl = (bucketName, objectName) => {
+ return `${minioConfig.publicUrl}/${bucketName}/${objectName}`;
+};
+
+module.exports = {
+ minioConfig,
+ createMinioClient,
+ initializeBuckets,
+ getPublicUrl
+};
\ No newline at end of file
diff --git a/config/swagger.js b/config/swagger.js
new file mode 100644
index 0000000..e8d70e9
--- /dev/null
+++ b/config/swagger.js
@@ -0,0 +1,43 @@
+const swaggerUi = require('swagger-ui-express');
+
+const swaggerJsdoc = require('swagger-jsdoc');
+
+// Swagger定义
+const options = {
+ definition: {
+ openapi: '3.0.0',
+ info: {
+ title: '融豆商城 API',
+ version: '1.0.0',
+ description: '融豆商城后端API文档',
+ contact: {
+ name: '技术支持',
+ email: 'support@example.com'
+ },
+ },
+ servers: [
+ {
+ url: '/api',
+ description: 'API服务器'
+ }
+ ],
+ components: {
+ securitySchemes: {
+ bearerAuth: {
+ type: 'http',
+ scheme: 'bearer',
+ bearerFormat: 'JWT'
+ }
+ }
+ },
+ security: [{
+ bearerAuth: []
+ }]
+ },
+ // API文档扫描路径
+ apis: ['./docs/schemas/*.js', './docs/apis/*.js', './routes/*.js', './admin/routes/*.js'],
+};
+
+const specs = swaggerJsdoc(options);
+
+module.exports = {specs,swaggerUi};
\ No newline at end of file
diff --git a/database.js b/database.js
new file mode 100644
index 0000000..949a3af
--- /dev/null
+++ b/database.js
@@ -0,0 +1,159 @@
+const mysql = require('mysql2/promise');
+
+// 数据库配置
+const dbConfig = {
+ // host: process.env.DB_HOST || '114.55.111.44',
+ // user: process.env.DB_USER || 'maov2',
+ // password: process.env.DB_PASSWORD || '5fYhw8z6T62b7heS',
+ // database: process.env.DB_NAME || 'maov2',
+ host: '114.55.111.44',
+ user: 'test_mao',
+ password: 'nK2mPbWriBp25BRd',
+ database: 'test_mao',
+ charset: 'utf8mb4',
+ dateStrings: true,
+ // 连接池配置
+ connectionLimit: 20, // 连接池最大连接数
+ queueLimit: 0, // 排队等待连接的最大数量,0表示无限制
+ // 连接超时配置
+ // acquireTimeout: 60000, // 获取连接超时时间 60秒
+ // timeout: 60000, // 查询超时时间 60秒
+ // reconnect: true, // 自动重连
+ // 连接保活配置
+ // multipleStatements: true,
+ // 空闲连接超时配置
+ // idleTimeout: 300000, // 5分钟空闲超时
+ // maxLifetime: 1800000, // 30分钟最大生命周期
+ // 连接保活设置
+ // keepAliveInitialDelay: 0, // 开始保活探测前的延迟时间
+ // enableKeepAlive: true, // 启用TCP保活
+ // 添加类型转换配置
+ typeCast: function (field, next) {
+ if (field.type === 'TINY' && field.length === 1) {
+ return (field.string() === '1'); // 1 = true, 0 = false
+ }
+ return next();
+ },
+ // 确保参数正确处理
+ supportBigNumbers: true,
+ bigNumberStrings: false
+};
+
+// 创建数据库连接池
+let pool;
+
+/**
+ * 初始化数据库连接池
+ * @returns {Promise} 数据库连接池
+ */
+async function initDB() {
+ if (!pool) {
+ try {
+ pool = mysql.createPool(dbConfig);
+
+ // 添加连接池事件监听
+ pool.on('connection', function (connection) {
+ console.log('新的数据库连接建立:', connection.threadId);
+ });
+
+ // 注释掉频繁的连接获取和释放日志,避免日志过多
+ // pool.on('acquire', function (connection) {
+ // console.log('连接池获取连接:', connection.threadId);
+ // });
+
+ // pool.on('release', function (connection) {
+ // console.log('连接池释放连接:', connection.threadId);
+ // });
+
+ pool.on('error', function (err) {
+ console.error('数据库连接池错误:', err);
+ if (err.code === 'PROTOCOL_CONNECTION_LOST') {
+ console.log('数据库连接丢失,尝试重新连接...');
+ } else if (err.code === 'ECONNRESET') {
+ console.log('数据库连接被重置,尝试重新连接...');
+ } else if (err.code === 'ETIMEDOUT') {
+ console.log('数据库连接超时,尝试重新连接...');
+ }
+ });
+
+ // 测试连接
+ const connection = await pool.getConnection();
+ console.log('数据库连接池初始化成功');
+ connection.release();
+
+ } catch (error) {
+ console.error('数据库连接池初始化失败:', error);
+ throw error;
+ }
+ }
+ return pool;
+}
+
+/**
+ * 获取数据库连接池
+ * @returns {mysql.Pool} 数据库连接池
+ */
+function getDB() {
+ if (!pool) {
+ throw new Error('数据库连接池未初始化,请先调用 initDB()');
+ }
+ return pool;
+}
+
+/**
+ * 执行数据库查询(带重试机制)
+ * @param {string} sql SQL查询语句
+ * @param {Array} params 查询参数
+ * @param {number} retries 重试次数
+ * @returns {Promise} 查询结果
+ */
+async function executeQuery(sql, params = [], retries = 3) {
+ for (let i = 0; i < retries; i++) {
+ try {
+ const connection = await pool.getConnection();
+ try {
+ const [results] = await connection.execute(sql, params);
+ connection.release();
+ return results;
+ } catch (error) {
+ connection.release();
+ throw error;
+ }
+ } catch (error) {
+ console.error(`数据库查询失败 (尝试 ${i + 1}/${retries}):`, error.message);
+
+ if (i === retries - 1) {
+ throw error;
+ }
+
+ // 如果是连接相关错误,等待后重试
+ if (error.code === 'PROTOCOL_CONNECTION_LOST' ||
+ error.code === 'ECONNRESET' ||
+ error.code === 'ETIMEDOUT') {
+ console.log(`等待 ${(i + 1) * 1000}ms 后重试...`);
+ await new Promise(resolve => setTimeout(resolve, (i + 1) * 1000));
+ } else {
+ throw error;
+ }
+ }
+ }
+}
+
+/**
+ * 关闭数据库连接池
+ */
+async function closeDB() {
+ if (pool) {
+ await pool.end();
+ pool = null;
+ console.log('数据库连接池已关闭');
+ }
+}
+
+module.exports = {
+ initDB,
+ getDB,
+ closeDB,
+ executeQuery,
+ dbConfig
+};
\ No newline at end of file
diff --git a/package-lock.json b/package-lock.json
new file mode 100644
index 0000000..c346e02
--- /dev/null
+++ b/package-lock.json
@@ -0,0 +1,1559 @@
+{
+ "name": "middle-office-system",
+ "version": "0.0.0",
+ "lockfileVersion": 3,
+ "requires": true,
+ "packages": {
+ "": {
+ "name": "middle-office-system",
+ "version": "0.0.0",
+ "dependencies": {
+ "bcryptjs": "^2.4.3",
+ "body-parser": "^2.2.0",
+ "cookie-parser": "~1.4.4",
+ "cors": "^2.8.5",
+ "debug": "~2.6.9",
+ "express": "~4.16.1",
+ "jsonwebtoken": "^9.0.2",
+ "morgan": "~1.9.1",
+ "mysql2": "^3.15.0",
+ "swagger-jsdoc": "^6.2.8",
+ "swagger-ui-express": "^5.0.1"
+ }
+ },
+ "node_modules/@apidevtools/json-schema-ref-parser": {
+ "version": "9.1.2",
+ "resolved": "https://registry.npmjs.org/@apidevtools/json-schema-ref-parser/-/json-schema-ref-parser-9.1.2.tgz",
+ "integrity": "sha512-r1w81DpR+KyRWd3f+rk6TNqMgedmAxZP5v5KWlXQWlgMUUtyEJch0DKEci1SorPMiSeM8XPl7MZ3miJ60JIpQg==",
+ "dependencies": {
+ "@jsdevtools/ono": "^7.1.3",
+ "@types/json-schema": "^7.0.6",
+ "call-me-maybe": "^1.0.1",
+ "js-yaml": "^4.1.0"
+ }
+ },
+ "node_modules/@apidevtools/openapi-schemas": {
+ "version": "2.1.0",
+ "resolved": "https://registry.npmjs.org/@apidevtools/openapi-schemas/-/openapi-schemas-2.1.0.tgz",
+ "integrity": "sha512-Zc1AlqrJlX3SlpupFGpiLi2EbteyP7fXmUOGup6/DnkRgjP9bgMM/ag+n91rsv0U1Gpz0H3VILA/o3bW7Ua6BQ==",
+ "engines": {
+ "node": ">=10"
+ }
+ },
+ "node_modules/@apidevtools/swagger-methods": {
+ "version": "3.0.2",
+ "resolved": "https://registry.npmjs.org/@apidevtools/swagger-methods/-/swagger-methods-3.0.2.tgz",
+ "integrity": "sha512-QAkD5kK2b1WfjDS/UQn/qQkbwF31uqRjPTrsCs5ZG9BQGAkjwvqGFjjPqAuzac/IYzpPtRzjCP1WrTuAIjMrXg=="
+ },
+ "node_modules/@apidevtools/swagger-parser": {
+ "version": "10.0.3",
+ "resolved": "https://registry.npmjs.org/@apidevtools/swagger-parser/-/swagger-parser-10.0.3.tgz",
+ "integrity": "sha512-sNiLY51vZOmSPFZA5TF35KZ2HbgYklQnTSDnkghamzLb3EkNtcQnrBQEj5AOCxHpTtXpqMCRM1CrmV2rG6nw4g==",
+ "dependencies": {
+ "@apidevtools/json-schema-ref-parser": "^9.0.6",
+ "@apidevtools/openapi-schemas": "^2.0.4",
+ "@apidevtools/swagger-methods": "^3.0.2",
+ "@jsdevtools/ono": "^7.1.3",
+ "call-me-maybe": "^1.0.1",
+ "z-schema": "^5.0.1"
+ },
+ "peerDependencies": {
+ "openapi-types": ">=7"
+ }
+ },
+ "node_modules/@jsdevtools/ono": {
+ "version": "7.1.3",
+ "resolved": "https://registry.npmjs.org/@jsdevtools/ono/-/ono-7.1.3.tgz",
+ "integrity": "sha512-4JQNk+3mVzK3xh2rqd6RB4J46qUR19azEHBneZyTZM+c456qOrbbM/5xcR8huNCCcbVt7+UmizG6GuUvPvKUYg=="
+ },
+ "node_modules/@scarf/scarf": {
+ "version": "1.4.0",
+ "resolved": "https://registry.npmjs.org/@scarf/scarf/-/scarf-1.4.0.tgz",
+ "integrity": "sha512-xxeapPiUXdZAE3che6f3xogoJPeZgig6omHEy1rIY5WVsB3H2BHNnZH+gHG6x91SCWyQCzWGsuL2Hh3ClO5/qQ==",
+ "hasInstallScript": true
+ },
+ "node_modules/@types/json-schema": {
+ "version": "7.0.15",
+ "resolved": "https://registry.npmjs.org/@types/json-schema/-/json-schema-7.0.15.tgz",
+ "integrity": "sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA=="
+ },
+ "node_modules/accepts": {
+ "version": "1.3.8",
+ "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.8.tgz",
+ "integrity": "sha512-PYAthTa2m2VKxuvSD3DPC/Gy+U+sOA1LAuT8mkmRuvw+NACSaeXEQ+NHcVF7rONl6qcaxV3Uuemwawk+7+SJLw==",
+ "dependencies": {
+ "mime-types": "~2.1.34",
+ "negotiator": "0.6.3"
+ },
+ "engines": {
+ "node": ">= 0.6"
+ }
+ },
+ "node_modules/argparse": {
+ "version": "2.0.1",
+ "resolved": "https://registry.npmjs.org/argparse/-/argparse-2.0.1.tgz",
+ "integrity": "sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q=="
+ },
+ "node_modules/array-flatten": {
+ "version": "1.1.1",
+ "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz",
+ "integrity": "sha512-PCVAQswWemu6UdxsDFFX/+gVeYqKAod3D3UVm91jHwynguOwAvYPhx8nNlM++NqRcK6CxxpUafjmhIdKiHibqg=="
+ },
+ "node_modules/aws-ssl-profiles": {
+ "version": "1.1.2",
+ "resolved": "https://registry.npmjs.org/aws-ssl-profiles/-/aws-ssl-profiles-1.1.2.tgz",
+ "integrity": "sha512-NZKeq9AfyQvEeNlN0zSYAaWrmBffJh3IELMZfRpJVWgrpEbtEpnjvzqBPf+mxoI287JohRDoa+/nsfqqiZmF6g==",
+ "engines": {
+ "node": ">= 6.0.0"
+ }
+ },
+ "node_modules/balanced-match": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz",
+ "integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw=="
+ },
+ "node_modules/basic-auth": {
+ "version": "2.0.1",
+ "resolved": "https://registry.npmjs.org/basic-auth/-/basic-auth-2.0.1.tgz",
+ "integrity": "sha512-NF+epuEdnUYVlGuhaxbbq+dvJttwLnGY+YixlXlME5KpQ5W3CnXA5cVTneY3SPbPDRkcjMbifrwmFYcClgOZeg==",
+ "dependencies": {
+ "safe-buffer": "5.1.2"
+ },
+ "engines": {
+ "node": ">= 0.8"
+ }
+ },
+ "node_modules/bcryptjs": {
+ "version": "2.4.3",
+ "resolved": "https://registry.npmjs.org/bcryptjs/-/bcryptjs-2.4.3.tgz",
+ "integrity": "sha512-V/Hy/X9Vt7f3BbPJEi8BdVFMByHi+jNXrYkW3huaybV/kQ0KJg0Y6PkEMbn+zeT+i+SiKZ/HMqJGIIt4LZDqNQ=="
+ },
+ "node_modules/body-parser": {
+ "version": "2.2.0",
+ "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-2.2.0.tgz",
+ "integrity": "sha512-02qvAaxv8tp7fBa/mw1ga98OGm+eCbqzJOKoRt70sLmfEEi+jyBYVTDGfCL/k06/4EMk/z01gCe7HoCH/f2LTg==",
+ "dependencies": {
+ "bytes": "^3.1.2",
+ "content-type": "^1.0.5",
+ "debug": "^4.4.0",
+ "http-errors": "^2.0.0",
+ "iconv-lite": "^0.6.3",
+ "on-finished": "^2.4.1",
+ "qs": "^6.14.0",
+ "raw-body": "^3.0.0",
+ "type-is": "^2.0.0"
+ },
+ "engines": {
+ "node": ">=18"
+ }
+ },
+ "node_modules/body-parser/node_modules/debug": {
+ "version": "4.4.3",
+ "resolved": "https://registry.npmjs.org/debug/-/debug-4.4.3.tgz",
+ "integrity": "sha512-RGwwWnwQvkVfavKVt22FGLw+xYSdzARwm0ru6DhTVA3umU5hZc28V3kO4stgYryrTlLpuvgI9GiijltAjNbcqA==",
+ "dependencies": {
+ "ms": "^2.1.3"
+ },
+ "engines": {
+ "node": ">=6.0"
+ },
+ "peerDependenciesMeta": {
+ "supports-color": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/body-parser/node_modules/depd": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/depd/-/depd-2.0.0.tgz",
+ "integrity": "sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw==",
+ "engines": {
+ "node": ">= 0.8"
+ }
+ },
+ "node_modules/body-parser/node_modules/http-errors": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-2.0.0.tgz",
+ "integrity": "sha512-FtwrG/euBzaEjYeRqOgly7G0qviiXoJWnvEH2Z1plBdXgbyjv34pHTSb9zoeHMyDy33+DWy5Wt9Wo+TURtOYSQ==",
+ "dependencies": {
+ "depd": "2.0.0",
+ "inherits": "2.0.4",
+ "setprototypeof": "1.2.0",
+ "statuses": "2.0.1",
+ "toidentifier": "1.0.1"
+ },
+ "engines": {
+ "node": ">= 0.8"
+ }
+ },
+ "node_modules/body-parser/node_modules/inherits": {
+ "version": "2.0.4",
+ "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz",
+ "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ=="
+ },
+ "node_modules/body-parser/node_modules/media-typer": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-1.1.0.tgz",
+ "integrity": "sha512-aisnrDP4GNe06UcKFnV5bfMNPBUw4jsLGaWwWfnH3v02GnBuXX2MCVn5RbrWo0j3pczUilYblq7fQ7Nw2t5XKw==",
+ "engines": {
+ "node": ">= 0.8"
+ }
+ },
+ "node_modules/body-parser/node_modules/mime-db": {
+ "version": "1.54.0",
+ "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.54.0.tgz",
+ "integrity": "sha512-aU5EJuIN2WDemCcAp2vFBfp/m4EAhWJnUNSSw0ixs7/kXbd6Pg64EmwJkNdFhB8aWt1sH2CTXrLxo/iAGV3oPQ==",
+ "engines": {
+ "node": ">= 0.6"
+ }
+ },
+ "node_modules/body-parser/node_modules/mime-types": {
+ "version": "3.0.1",
+ "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-3.0.1.tgz",
+ "integrity": "sha512-xRc4oEhT6eaBpU1XF7AjpOFD+xQmXNB5OVKwp4tqCuBpHLS/ZbBDrc07mYTDqVMg6PfxUjjNp85O6Cd2Z/5HWA==",
+ "dependencies": {
+ "mime-db": "^1.54.0"
+ },
+ "engines": {
+ "node": ">= 0.6"
+ }
+ },
+ "node_modules/body-parser/node_modules/ms": {
+ "version": "2.1.3",
+ "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz",
+ "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA=="
+ },
+ "node_modules/body-parser/node_modules/on-finished": {
+ "version": "2.4.1",
+ "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.4.1.tgz",
+ "integrity": "sha512-oVlzkg3ENAhCk2zdv7IJwd/QUD4z2RxRwpkcGY8psCVcCYZNq4wYnVWALHM+brtuJjePWiYF/ClmuDr8Ch5+kg==",
+ "dependencies": {
+ "ee-first": "1.1.1"
+ },
+ "engines": {
+ "node": ">= 0.8"
+ }
+ },
+ "node_modules/body-parser/node_modules/qs": {
+ "version": "6.14.0",
+ "resolved": "https://registry.npmjs.org/qs/-/qs-6.14.0.tgz",
+ "integrity": "sha512-YWWTjgABSKcvs/nWBi9PycY/JiPJqOD4JA6o9Sej2AtvSGarXxKC3OQSk4pAarbdQlKAh5D4FCQkJNkW+GAn3w==",
+ "dependencies": {
+ "side-channel": "^1.1.0"
+ },
+ "engines": {
+ "node": ">=0.6"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/body-parser/node_modules/setprototypeof": {
+ "version": "1.2.0",
+ "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.2.0.tgz",
+ "integrity": "sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw=="
+ },
+ "node_modules/body-parser/node_modules/statuses": {
+ "version": "2.0.1",
+ "resolved": "https://registry.npmjs.org/statuses/-/statuses-2.0.1.tgz",
+ "integrity": "sha512-RwNA9Z/7PrK06rYLIzFMlaF+l73iwpzsqRIFgbMLbTcLD6cOao82TaWefPXQvB2fOC4AjuYSEndS7N/mTCbkdQ==",
+ "engines": {
+ "node": ">= 0.8"
+ }
+ },
+ "node_modules/body-parser/node_modules/type-is": {
+ "version": "2.0.1",
+ "resolved": "https://registry.npmjs.org/type-is/-/type-is-2.0.1.tgz",
+ "integrity": "sha512-OZs6gsjF4vMp32qrCbiVSkrFmXtG/AZhY3t0iAMrMBiAZyV9oALtXO8hsrHbMXF9x6L3grlFuwW2oAz7cav+Gw==",
+ "dependencies": {
+ "content-type": "^1.0.5",
+ "media-typer": "^1.1.0",
+ "mime-types": "^3.0.0"
+ },
+ "engines": {
+ "node": ">= 0.6"
+ }
+ },
+ "node_modules/brace-expansion": {
+ "version": "1.1.12",
+ "resolved": "https://registry.npmjs.org/brace-expansion/-/brace-expansion-1.1.12.tgz",
+ "integrity": "sha512-9T9UjW3r0UW5c1Q7GTwllptXwhvYmEzFhzMfZ9H7FQWt+uZePjZPjBP/W1ZEyZ1twGWom5/56TF4lPcqjnDHcg==",
+ "dependencies": {
+ "balanced-match": "^1.0.0",
+ "concat-map": "0.0.1"
+ }
+ },
+ "node_modules/buffer-equal-constant-time": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/buffer-equal-constant-time/-/buffer-equal-constant-time-1.0.1.tgz",
+ "integrity": "sha512-zRpUiDwd/xk6ADqPMATG8vc9VPrkck7T07OIx0gnjmJAnHnTVXNQG3vfvWNuiZIkwu9KrKdA1iJKfsfTVxE6NA=="
+ },
+ "node_modules/bytes": {
+ "version": "3.1.2",
+ "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.2.tgz",
+ "integrity": "sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg==",
+ "engines": {
+ "node": ">= 0.8"
+ }
+ },
+ "node_modules/call-bind-apply-helpers": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/call-bind-apply-helpers/-/call-bind-apply-helpers-1.0.2.tgz",
+ "integrity": "sha512-Sp1ablJ0ivDkSzjcaJdxEunN5/XvksFJ2sMBFfq6x0ryhQV/2b/KwFe21cMpmHtPOSij8K99/wSfoEuTObmuMQ==",
+ "dependencies": {
+ "es-errors": "^1.3.0",
+ "function-bind": "^1.1.2"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ }
+ },
+ "node_modules/call-bound": {
+ "version": "1.0.4",
+ "resolved": "https://registry.npmjs.org/call-bound/-/call-bound-1.0.4.tgz",
+ "integrity": "sha512-+ys997U96po4Kx/ABpBCqhA9EuxJaQWDQg7295H4hBphv3IZg0boBKuwYpt4YXp6MZ5AmZQnU/tyMTlRpaSejg==",
+ "dependencies": {
+ "call-bind-apply-helpers": "^1.0.2",
+ "get-intrinsic": "^1.3.0"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/call-me-maybe": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/call-me-maybe/-/call-me-maybe-1.0.2.tgz",
+ "integrity": "sha512-HpX65o1Hnr9HH25ojC1YGs7HCQLq0GCOibSaWER0eNpgJ/Z1MZv2mTc7+xh6WOPxbRVcmgbv4hGU+uSQ/2xFZQ=="
+ },
+ "node_modules/commander": {
+ "version": "6.2.0",
+ "resolved": "https://registry.npmjs.org/commander/-/commander-6.2.0.tgz",
+ "integrity": "sha512-zP4jEKbe8SHzKJYQmq8Y9gYjtO/POJLgIdKgV7B9qNmABVFVc+ctqSX6iXh4mCpJfRBOabiZ2YKPg8ciDw6C+Q==",
+ "engines": {
+ "node": ">= 6"
+ }
+ },
+ "node_modules/concat-map": {
+ "version": "0.0.1",
+ "resolved": "https://registry.npmjs.org/concat-map/-/concat-map-0.0.1.tgz",
+ "integrity": "sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg=="
+ },
+ "node_modules/content-disposition": {
+ "version": "0.5.2",
+ "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.2.tgz",
+ "integrity": "sha512-kRGRZw3bLlFISDBgwTSA1TMBFN6J6GWDeubmDE3AF+3+yXL8hTWv8r5rkLbqYXY4RjPk/EzHnClI3zQf1cFmHA==",
+ "engines": {
+ "node": ">= 0.6"
+ }
+ },
+ "node_modules/content-type": {
+ "version": "1.0.5",
+ "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.5.tgz",
+ "integrity": "sha512-nTjqfcBFEipKdXCv4YDQWCfmcLZKm81ldF0pAopTvyrFGVbcR6P/VAAd5G7N+0tTr8QqiU0tFadD6FK4NtJwOA==",
+ "engines": {
+ "node": ">= 0.6"
+ }
+ },
+ "node_modules/cookie": {
+ "version": "0.7.2",
+ "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.7.2.tgz",
+ "integrity": "sha512-yki5XnKuf750l50uGTllt6kKILY4nQ1eNIQatoXEByZ5dWgnKqbnqmTrBE5B4N7lrMJKQ2ytWMiTO2o0v6Ew/w==",
+ "engines": {
+ "node": ">= 0.6"
+ }
+ },
+ "node_modules/cookie-parser": {
+ "version": "1.4.7",
+ "resolved": "https://registry.npmjs.org/cookie-parser/-/cookie-parser-1.4.7.tgz",
+ "integrity": "sha512-nGUvgXnotP3BsjiLX2ypbQnWoGUPIIfHQNZkkC668ntrzGWEZVW70HDEB1qnNGMicPje6EttlIgzo51YSwNQGw==",
+ "dependencies": {
+ "cookie": "0.7.2",
+ "cookie-signature": "1.0.6"
+ },
+ "engines": {
+ "node": ">= 0.8.0"
+ }
+ },
+ "node_modules/cookie-signature": {
+ "version": "1.0.6",
+ "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz",
+ "integrity": "sha512-QADzlaHc8icV8I7vbaJXJwod9HWYp8uCqf1xa4OfNu1T7JVxQIrUgOWtHdNDtPiywmFbiS12VjotIXLrKM3orQ=="
+ },
+ "node_modules/cors": {
+ "version": "2.8.5",
+ "resolved": "https://registry.npmjs.org/cors/-/cors-2.8.5.tgz",
+ "integrity": "sha512-KIHbLJqu73RGr/hnbrO9uBeixNGuvSQjul/jdFvS/KFSIH1hWVd1ng7zOHx+YrEfInLG7q4n6GHQ9cDtxv/P6g==",
+ "dependencies": {
+ "object-assign": "^4",
+ "vary": "^1"
+ },
+ "engines": {
+ "node": ">= 0.10"
+ }
+ },
+ "node_modules/debug": {
+ "version": "2.6.9",
+ "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz",
+ "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==",
+ "dependencies": {
+ "ms": "2.0.0"
+ }
+ },
+ "node_modules/denque": {
+ "version": "2.1.0",
+ "resolved": "https://registry.npmjs.org/denque/-/denque-2.1.0.tgz",
+ "integrity": "sha512-HVQE3AAb/pxF8fQAoiqpvg9i3evqug3hoiwakOyZAwJm+6vZehbkYXZ0l4JxS+I3QxM97v5aaRNhj8v5oBhekw==",
+ "engines": {
+ "node": ">=0.10"
+ }
+ },
+ "node_modules/depd": {
+ "version": "1.1.2",
+ "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz",
+ "integrity": "sha512-7emPTl6Dpo6JRXOXjLRxck+FlLRX5847cLKEn00PLAgc3g2hTZZgr+e4c2v6QpSmLeFP3n5yUo7ft6avBK/5jQ==",
+ "engines": {
+ "node": ">= 0.6"
+ }
+ },
+ "node_modules/destroy": {
+ "version": "1.0.4",
+ "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz",
+ "integrity": "sha512-3NdhDuEXnfun/z7x9GOElY49LoqVHoGScmOKwmxhsS8N5Y+Z8KyPPDnaSzqWgYt/ji4mqwfTS34Htrk0zPIXVg=="
+ },
+ "node_modules/doctrine": {
+ "version": "3.0.0",
+ "resolved": "https://registry.npmjs.org/doctrine/-/doctrine-3.0.0.tgz",
+ "integrity": "sha512-yS+Q5i3hBf7GBkd4KG8a7eBNNWNGLTaEwwYWUijIYM7zrlYDM0BFXHjjPWlWZ1Rg7UaddZeIDmi9jF3HmqiQ2w==",
+ "dependencies": {
+ "esutils": "^2.0.2"
+ },
+ "engines": {
+ "node": ">=6.0.0"
+ }
+ },
+ "node_modules/dunder-proto": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/dunder-proto/-/dunder-proto-1.0.1.tgz",
+ "integrity": "sha512-KIN/nDJBQRcXw0MLVhZE9iQHmG68qAVIBg9CqmUYjmQIhgij9U5MFvrqkUL5FbtyyzZuOeOt0zdeRe4UY7ct+A==",
+ "dependencies": {
+ "call-bind-apply-helpers": "^1.0.1",
+ "es-errors": "^1.3.0",
+ "gopd": "^1.2.0"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ }
+ },
+ "node_modules/ecdsa-sig-formatter": {
+ "version": "1.0.11",
+ "resolved": "https://registry.npmjs.org/ecdsa-sig-formatter/-/ecdsa-sig-formatter-1.0.11.tgz",
+ "integrity": "sha512-nagl3RYrbNv6kQkeJIpt6NJZy8twLB/2vtz6yN9Z4vRKHN4/QZJIEbqohALSgwKdnksuY3k5Addp5lg8sVoVcQ==",
+ "dependencies": {
+ "safe-buffer": "^5.0.1"
+ }
+ },
+ "node_modules/ee-first": {
+ "version": "1.1.1",
+ "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz",
+ "integrity": "sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow=="
+ },
+ "node_modules/encodeurl": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz",
+ "integrity": "sha512-TPJXq8JqFaVYm2CWmPvnP2Iyo4ZSM7/QKcSmuMLDObfpH5fi7RUGmd/rTDf+rut/saiDiQEeVTNgAmJEdAOx0w==",
+ "engines": {
+ "node": ">= 0.8"
+ }
+ },
+ "node_modules/es-define-property": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/es-define-property/-/es-define-property-1.0.1.tgz",
+ "integrity": "sha512-e3nRfgfUZ4rNGL232gUgX06QNyyez04KdjFrF+LTRoOXmrOgFKDg4BCdsjW8EnT69eqdYGmRpJwiPVYNrCaW3g==",
+ "engines": {
+ "node": ">= 0.4"
+ }
+ },
+ "node_modules/es-errors": {
+ "version": "1.3.0",
+ "resolved": "https://registry.npmjs.org/es-errors/-/es-errors-1.3.0.tgz",
+ "integrity": "sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==",
+ "engines": {
+ "node": ">= 0.4"
+ }
+ },
+ "node_modules/es-object-atoms": {
+ "version": "1.1.1",
+ "resolved": "https://registry.npmjs.org/es-object-atoms/-/es-object-atoms-1.1.1.tgz",
+ "integrity": "sha512-FGgH2h8zKNim9ljj7dankFPcICIK9Cp5bm+c2gQSYePhpaG5+esrLODihIorn+Pe6FGJzWhXQotPv73jTaldXA==",
+ "dependencies": {
+ "es-errors": "^1.3.0"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ }
+ },
+ "node_modules/escape-html": {
+ "version": "1.0.3",
+ "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz",
+ "integrity": "sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow=="
+ },
+ "node_modules/esutils": {
+ "version": "2.0.3",
+ "resolved": "https://registry.npmjs.org/esutils/-/esutils-2.0.3.tgz",
+ "integrity": "sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g==",
+ "engines": {
+ "node": ">=0.10.0"
+ }
+ },
+ "node_modules/etag": {
+ "version": "1.8.1",
+ "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz",
+ "integrity": "sha512-aIL5Fx7mawVa300al2BnEE4iNvo1qETxLrPI/o05L7z6go7fCw1J6EQmbK4FmJ2AS7kgVF/KEZWufBfdClMcPg==",
+ "engines": {
+ "node": ">= 0.6"
+ }
+ },
+ "node_modules/express": {
+ "version": "4.16.4",
+ "resolved": "https://registry.npmjs.org/express/-/express-4.16.4.tgz",
+ "integrity": "sha512-j12Uuyb4FMrd/qQAm6uCHAkPtO8FDTRJZBDd5D2KOL2eLaz1yUNdUB/NOIyq0iU4q4cFarsUCrnFDPBcnksuOg==",
+ "dependencies": {
+ "accepts": "~1.3.5",
+ "array-flatten": "1.1.1",
+ "body-parser": "1.18.3",
+ "content-disposition": "0.5.2",
+ "content-type": "~1.0.4",
+ "cookie": "0.3.1",
+ "cookie-signature": "1.0.6",
+ "debug": "2.6.9",
+ "depd": "~1.1.2",
+ "encodeurl": "~1.0.2",
+ "escape-html": "~1.0.3",
+ "etag": "~1.8.1",
+ "finalhandler": "1.1.1",
+ "fresh": "0.5.2",
+ "merge-descriptors": "1.0.1",
+ "methods": "~1.1.2",
+ "on-finished": "~2.3.0",
+ "parseurl": "~1.3.2",
+ "path-to-regexp": "0.1.7",
+ "proxy-addr": "~2.0.4",
+ "qs": "6.5.2",
+ "range-parser": "~1.2.0",
+ "safe-buffer": "5.1.2",
+ "send": "0.16.2",
+ "serve-static": "1.13.2",
+ "setprototypeof": "1.1.0",
+ "statuses": "~1.4.0",
+ "type-is": "~1.6.16",
+ "utils-merge": "1.0.1",
+ "vary": "~1.1.2"
+ },
+ "engines": {
+ "node": ">= 0.10.0"
+ }
+ },
+ "node_modules/express/node_modules/body-parser": {
+ "version": "1.18.3",
+ "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.18.3.tgz",
+ "integrity": "sha512-YQyoqQG3sO8iCmf8+hyVpgHHOv0/hCEFiS4zTGUwTA1HjAFX66wRcNQrVCeJq9pgESMRvUAOvSil5MJlmccuKQ==",
+ "dependencies": {
+ "bytes": "3.0.0",
+ "content-type": "~1.0.4",
+ "debug": "2.6.9",
+ "depd": "~1.1.2",
+ "http-errors": "~1.6.3",
+ "iconv-lite": "0.4.23",
+ "on-finished": "~2.3.0",
+ "qs": "6.5.2",
+ "raw-body": "2.3.3",
+ "type-is": "~1.6.16"
+ },
+ "engines": {
+ "node": ">= 0.8"
+ }
+ },
+ "node_modules/express/node_modules/bytes": {
+ "version": "3.0.0",
+ "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.0.0.tgz",
+ "integrity": "sha512-pMhOfFDPiv9t5jjIXkHosWmkSyQbvsgEVNkz0ERHbuLh2T/7j4Mqqpz523Fe8MVY89KC6Sh/QfS2sM+SjgFDcw==",
+ "engines": {
+ "node": ">= 0.8"
+ }
+ },
+ "node_modules/express/node_modules/cookie": {
+ "version": "0.3.1",
+ "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.3.1.tgz",
+ "integrity": "sha512-+IJOX0OqlHCszo2mBUq+SrEbCj6w7Kpffqx60zYbPTFaO4+yYgRjHwcZNpWvaTylDHaV7PPmBHzSecZiMhtPgw==",
+ "engines": {
+ "node": ">= 0.6"
+ }
+ },
+ "node_modules/express/node_modules/iconv-lite": {
+ "version": "0.4.23",
+ "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.23.tgz",
+ "integrity": "sha512-neyTUVFtahjf0mB3dZT77u+8O0QB89jFdnBkd5P1JgYPbPaia3gXXOVL2fq8VyU2gMMD7SaN7QukTB/pmXYvDA==",
+ "dependencies": {
+ "safer-buffer": ">= 2.1.2 < 3"
+ },
+ "engines": {
+ "node": ">=0.10.0"
+ }
+ },
+ "node_modules/express/node_modules/raw-body": {
+ "version": "2.3.3",
+ "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.3.3.tgz",
+ "integrity": "sha512-9esiElv1BrZoI3rCDuOuKCBRbuApGGaDPQfjSflGxdy4oyzqghxu6klEkkVIvBje+FF0BX9coEv8KqW6X/7njw==",
+ "dependencies": {
+ "bytes": "3.0.0",
+ "http-errors": "1.6.3",
+ "iconv-lite": "0.4.23",
+ "unpipe": "1.0.0"
+ },
+ "engines": {
+ "node": ">= 0.8"
+ }
+ },
+ "node_modules/finalhandler": {
+ "version": "1.1.1",
+ "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.1.1.tgz",
+ "integrity": "sha512-Y1GUDo39ez4aHAw7MysnUD5JzYX+WaIj8I57kO3aEPT1fFRL4sr7mjei97FgnwhAyyzRYmQZaTHb2+9uZ1dPtg==",
+ "dependencies": {
+ "debug": "2.6.9",
+ "encodeurl": "~1.0.2",
+ "escape-html": "~1.0.3",
+ "on-finished": "~2.3.0",
+ "parseurl": "~1.3.2",
+ "statuses": "~1.4.0",
+ "unpipe": "~1.0.0"
+ },
+ "engines": {
+ "node": ">= 0.8"
+ }
+ },
+ "node_modules/forwarded": {
+ "version": "0.2.0",
+ "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.2.0.tgz",
+ "integrity": "sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow==",
+ "engines": {
+ "node": ">= 0.6"
+ }
+ },
+ "node_modules/fresh": {
+ "version": "0.5.2",
+ "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz",
+ "integrity": "sha512-zJ2mQYM18rEFOudeV4GShTGIQ7RbzA7ozbU9I/XBpm7kqgMywgmylMwXHxZJmkVoYkna9d2pVXVXPdYTP9ej8Q==",
+ "engines": {
+ "node": ">= 0.6"
+ }
+ },
+ "node_modules/fs.realpath": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz",
+ "integrity": "sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw=="
+ },
+ "node_modules/function-bind": {
+ "version": "1.1.2",
+ "resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz",
+ "integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==",
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/generate-function": {
+ "version": "2.3.1",
+ "resolved": "https://registry.npmjs.org/generate-function/-/generate-function-2.3.1.tgz",
+ "integrity": "sha512-eeB5GfMNeevm/GRYq20ShmsaGcmI81kIX2K9XQx5miC8KdHaC6Jm0qQ8ZNeGOi7wYB8OsdxKs+Y2oVuTFuVwKQ==",
+ "dependencies": {
+ "is-property": "^1.0.2"
+ }
+ },
+ "node_modules/get-intrinsic": {
+ "version": "1.3.0",
+ "resolved": "https://registry.npmjs.org/get-intrinsic/-/get-intrinsic-1.3.0.tgz",
+ "integrity": "sha512-9fSjSaos/fRIVIp+xSJlE6lfwhES7LNtKaCBIamHsjr2na1BiABJPo0mOjjz8GJDURarmCPGqaiVg5mfjb98CQ==",
+ "dependencies": {
+ "call-bind-apply-helpers": "^1.0.2",
+ "es-define-property": "^1.0.1",
+ "es-errors": "^1.3.0",
+ "es-object-atoms": "^1.1.1",
+ "function-bind": "^1.1.2",
+ "get-proto": "^1.0.1",
+ "gopd": "^1.2.0",
+ "has-symbols": "^1.1.0",
+ "hasown": "^2.0.2",
+ "math-intrinsics": "^1.1.0"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/get-proto": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/get-proto/-/get-proto-1.0.1.tgz",
+ "integrity": "sha512-sTSfBjoXBp89JvIKIefqw7U2CCebsc74kiY6awiGogKtoSGbgjYE/G/+l9sF3MWFPNc9IcoOC4ODfKHfxFmp0g==",
+ "dependencies": {
+ "dunder-proto": "^1.0.1",
+ "es-object-atoms": "^1.0.0"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ }
+ },
+ "node_modules/glob": {
+ "version": "7.1.6",
+ "resolved": "https://registry.npmjs.org/glob/-/glob-7.1.6.tgz",
+ "integrity": "sha512-LwaxwyZ72Lk7vZINtNNrywX0ZuLyStrdDtabefZKAY5ZGJhVtgdznluResxNmPitE0SAO+O26sWTHeKSI2wMBA==",
+ "deprecated": "Glob versions prior to v9 are no longer supported",
+ "dependencies": {
+ "fs.realpath": "^1.0.0",
+ "inflight": "^1.0.4",
+ "inherits": "2",
+ "minimatch": "^3.0.4",
+ "once": "^1.3.0",
+ "path-is-absolute": "^1.0.0"
+ },
+ "engines": {
+ "node": "*"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/isaacs"
+ }
+ },
+ "node_modules/gopd": {
+ "version": "1.2.0",
+ "resolved": "https://registry.npmjs.org/gopd/-/gopd-1.2.0.tgz",
+ "integrity": "sha512-ZUKRh6/kUFoAiTAtTYPZJ3hw9wNxx+BIBOijnlG9PnrJsCcSjs1wyyD6vJpaYtgnzDrKYRSqf3OO6Rfa93xsRg==",
+ "engines": {
+ "node": ">= 0.4"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/has-symbols": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/has-symbols/-/has-symbols-1.1.0.tgz",
+ "integrity": "sha512-1cDNdwJ2Jaohmb3sg4OmKaMBwuC48sYni5HUw2DvsC8LjGTLK9h+eb1X6RyuOHe4hT0ULCW68iomhjUoKUqlPQ==",
+ "engines": {
+ "node": ">= 0.4"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/hasown": {
+ "version": "2.0.2",
+ "resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.2.tgz",
+ "integrity": "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ==",
+ "dependencies": {
+ "function-bind": "^1.1.2"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ }
+ },
+ "node_modules/http-errors": {
+ "version": "1.6.3",
+ "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.6.3.tgz",
+ "integrity": "sha512-lks+lVC8dgGyh97jxvxeYTWQFvh4uw4yC12gVl63Cg30sjPX4wuGcdkICVXDAESr6OJGjqGA8Iz5mkeN6zlD7A==",
+ "dependencies": {
+ "depd": "~1.1.2",
+ "inherits": "2.0.3",
+ "setprototypeof": "1.1.0",
+ "statuses": ">= 1.4.0 < 2"
+ },
+ "engines": {
+ "node": ">= 0.6"
+ }
+ },
+ "node_modules/iconv-lite": {
+ "version": "0.6.3",
+ "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.6.3.tgz",
+ "integrity": "sha512-4fCk79wshMdzMp2rH06qWrJE4iolqLhCUH+OiuIgU++RB0+94NlDL81atO7GX55uUKueo0txHNtvEyI6D7WdMw==",
+ "dependencies": {
+ "safer-buffer": ">= 2.1.2 < 3.0.0"
+ },
+ "engines": {
+ "node": ">=0.10.0"
+ }
+ },
+ "node_modules/inflight": {
+ "version": "1.0.6",
+ "resolved": "https://registry.npmjs.org/inflight/-/inflight-1.0.6.tgz",
+ "integrity": "sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA==",
+ "deprecated": "This module is not supported, and leaks memory. Do not use it. Check out lru-cache if you want a good and tested way to coalesce async requests by a key value, which is much more comprehensive and powerful.",
+ "dependencies": {
+ "once": "^1.3.0",
+ "wrappy": "1"
+ }
+ },
+ "node_modules/inherits": {
+ "version": "2.0.3",
+ "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz",
+ "integrity": "sha512-x00IRNXNy63jwGkJmzPigoySHbaqpNuzKbBOmzK+g2OdZpQ9w+sxCN+VSB3ja7IAge2OP2qpfxTjeNcyjmW1uw=="
+ },
+ "node_modules/ipaddr.js": {
+ "version": "1.9.1",
+ "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz",
+ "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==",
+ "engines": {
+ "node": ">= 0.10"
+ }
+ },
+ "node_modules/is-property": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/is-property/-/is-property-1.0.2.tgz",
+ "integrity": "sha512-Ks/IoX00TtClbGQr4TWXemAnktAQvYB7HzcCxDGqEZU6oCmb2INHuOoKxbtR+HFkmYWBKv/dOZtGRiAjDhj92g=="
+ },
+ "node_modules/js-yaml": {
+ "version": "4.1.0",
+ "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-4.1.0.tgz",
+ "integrity": "sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA==",
+ "dependencies": {
+ "argparse": "^2.0.1"
+ },
+ "bin": {
+ "js-yaml": "bin/js-yaml.js"
+ }
+ },
+ "node_modules/jsonwebtoken": {
+ "version": "9.0.2",
+ "resolved": "https://registry.npmjs.org/jsonwebtoken/-/jsonwebtoken-9.0.2.tgz",
+ "integrity": "sha512-PRp66vJ865SSqOlgqS8hujT5U4AOgMfhrwYIuIhfKaoSCZcirrmASQr8CX7cUg+RMih+hgznrjp99o+W4pJLHQ==",
+ "dependencies": {
+ "jws": "^3.2.2",
+ "lodash.includes": "^4.3.0",
+ "lodash.isboolean": "^3.0.3",
+ "lodash.isinteger": "^4.0.4",
+ "lodash.isnumber": "^3.0.3",
+ "lodash.isplainobject": "^4.0.6",
+ "lodash.isstring": "^4.0.1",
+ "lodash.once": "^4.0.0",
+ "ms": "^2.1.1",
+ "semver": "^7.5.4"
+ },
+ "engines": {
+ "node": ">=12",
+ "npm": ">=6"
+ }
+ },
+ "node_modules/jsonwebtoken/node_modules/ms": {
+ "version": "2.1.3",
+ "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz",
+ "integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA=="
+ },
+ "node_modules/jwa": {
+ "version": "1.4.2",
+ "resolved": "https://registry.npmjs.org/jwa/-/jwa-1.4.2.tgz",
+ "integrity": "sha512-eeH5JO+21J78qMvTIDdBXidBd6nG2kZjg5Ohz/1fpa28Z4CcsWUzJ1ZZyFq/3z3N17aZy+ZuBoHljASbL1WfOw==",
+ "dependencies": {
+ "buffer-equal-constant-time": "^1.0.1",
+ "ecdsa-sig-formatter": "1.0.11",
+ "safe-buffer": "^5.0.1"
+ }
+ },
+ "node_modules/jws": {
+ "version": "3.2.2",
+ "resolved": "https://registry.npmjs.org/jws/-/jws-3.2.2.tgz",
+ "integrity": "sha512-YHlZCB6lMTllWDtSPHz/ZXTsi8S00usEV6v1tjq8tOUZzw7DpSDWVXjXDre6ed1w/pd495ODpHZYSdkRTsa0HA==",
+ "dependencies": {
+ "jwa": "^1.4.1",
+ "safe-buffer": "^5.0.1"
+ }
+ },
+ "node_modules/lodash.get": {
+ "version": "4.4.2",
+ "resolved": "https://registry.npmjs.org/lodash.get/-/lodash.get-4.4.2.tgz",
+ "integrity": "sha512-z+Uw/vLuy6gQe8cfaFWD7p0wVv8fJl3mbzXh33RS+0oW2wvUqiRXiQ69gLWSLpgB5/6sU+r6BlQR0MBILadqTQ==",
+ "deprecated": "This package is deprecated. Use the optional chaining (?.) operator instead."
+ },
+ "node_modules/lodash.includes": {
+ "version": "4.3.0",
+ "resolved": "https://registry.npmjs.org/lodash.includes/-/lodash.includes-4.3.0.tgz",
+ "integrity": "sha512-W3Bx6mdkRTGtlJISOvVD/lbqjTlPPUDTMnlXZFnVwi9NKJ6tiAk6LVdlhZMm17VZisqhKcgzpO5Wz91PCt5b0w=="
+ },
+ "node_modules/lodash.isboolean": {
+ "version": "3.0.3",
+ "resolved": "https://registry.npmjs.org/lodash.isboolean/-/lodash.isboolean-3.0.3.tgz",
+ "integrity": "sha512-Bz5mupy2SVbPHURB98VAcw+aHh4vRV5IPNhILUCsOzRmsTmSQ17jIuqopAentWoehktxGd9e/hbIXq980/1QJg=="
+ },
+ "node_modules/lodash.isequal": {
+ "version": "4.5.0",
+ "resolved": "https://registry.npmjs.org/lodash.isequal/-/lodash.isequal-4.5.0.tgz",
+ "integrity": "sha512-pDo3lu8Jhfjqls6GkMgpahsF9kCyayhgykjyLMNFTKWrpVdAQtYyB4muAMWozBB4ig/dtWAmsMxLEI8wuz+DYQ==",
+ "deprecated": "This package is deprecated. Use require('node:util').isDeepStrictEqual instead."
+ },
+ "node_modules/lodash.isinteger": {
+ "version": "4.0.4",
+ "resolved": "https://registry.npmjs.org/lodash.isinteger/-/lodash.isinteger-4.0.4.tgz",
+ "integrity": "sha512-DBwtEWN2caHQ9/imiNeEA5ys1JoRtRfY3d7V9wkqtbycnAmTvRRmbHKDV4a0EYc678/dia0jrte4tjYwVBaZUA=="
+ },
+ "node_modules/lodash.isnumber": {
+ "version": "3.0.3",
+ "resolved": "https://registry.npmjs.org/lodash.isnumber/-/lodash.isnumber-3.0.3.tgz",
+ "integrity": "sha512-QYqzpfwO3/CWf3XP+Z+tkQsfaLL/EnUlXWVkIk5FUPc4sBdTehEqZONuyRt2P67PXAk+NXmTBcc97zw9t1FQrw=="
+ },
+ "node_modules/lodash.isplainobject": {
+ "version": "4.0.6",
+ "resolved": "https://registry.npmjs.org/lodash.isplainobject/-/lodash.isplainobject-4.0.6.tgz",
+ "integrity": "sha512-oSXzaWypCMHkPC3NvBEaPHf0KsA5mvPrOPgQWDsbg8n7orZ290M0BmC/jgRZ4vcJ6DTAhjrsSYgdsW/F+MFOBA=="
+ },
+ "node_modules/lodash.isstring": {
+ "version": "4.0.1",
+ "resolved": "https://registry.npmjs.org/lodash.isstring/-/lodash.isstring-4.0.1.tgz",
+ "integrity": "sha512-0wJxfxH1wgO3GrbuP+dTTk7op+6L41QCXbGINEmD+ny/G/eCqGzxyCsh7159S+mgDDcoarnBw6PC1PS5+wUGgw=="
+ },
+ "node_modules/lodash.mergewith": {
+ "version": "4.6.2",
+ "resolved": "https://registry.npmjs.org/lodash.mergewith/-/lodash.mergewith-4.6.2.tgz",
+ "integrity": "sha512-GK3g5RPZWTRSeLSpgP8Xhra+pnjBC56q9FZYe1d5RN3TJ35dbkGy3YqBSMbyCrlbi+CM9Z3Jk5yTL7RCsqboyQ=="
+ },
+ "node_modules/lodash.once": {
+ "version": "4.1.1",
+ "resolved": "https://registry.npmjs.org/lodash.once/-/lodash.once-4.1.1.tgz",
+ "integrity": "sha512-Sb487aTOCr9drQVL8pIxOzVhafOjZN9UU54hiN8PU3uAiSV7lx1yYNpbNmex2PK6dSJoNTSJUUswT651yww3Mg=="
+ },
+ "node_modules/long": {
+ "version": "5.3.2",
+ "resolved": "https://registry.npmjs.org/long/-/long-5.3.2.tgz",
+ "integrity": "sha512-mNAgZ1GmyNhD7AuqnTG3/VQ26o760+ZYBPKjPvugO8+nLbYfX6TVpJPseBvopbdY+qpZ/lKUnmEc1LeZYS3QAA=="
+ },
+ "node_modules/lru-cache": {
+ "version": "7.18.3",
+ "resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-7.18.3.tgz",
+ "integrity": "sha512-jumlc0BIUrS3qJGgIkWZsyfAM7NCWiBcCDhnd+3NNM5KbBmLTgHVfWBcg6W+rLUsIpzpERPsvwUP7CckAQSOoA==",
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/lru.min": {
+ "version": "1.1.2",
+ "resolved": "https://registry.npmjs.org/lru.min/-/lru.min-1.1.2.tgz",
+ "integrity": "sha512-Nv9KddBcQSlQopmBHXSsZVY5xsdlZkdH/Iey0BlcBYggMd4two7cZnKOK9vmy3nY0O5RGH99z1PCeTpPqszUYg==",
+ "engines": {
+ "bun": ">=1.0.0",
+ "deno": ">=1.30.0",
+ "node": ">=8.0.0"
+ },
+ "funding": {
+ "type": "github",
+ "url": "https://github.com/sponsors/wellwelwel"
+ }
+ },
+ "node_modules/math-intrinsics": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/math-intrinsics/-/math-intrinsics-1.1.0.tgz",
+ "integrity": "sha512-/IXtbwEk5HTPyEwyKX6hGkYXxM9nbj64B+ilVJnC/R6B0pH5G4V3b0pVbL7DBj4tkhBAppbQUlf6F6Xl9LHu1g==",
+ "engines": {
+ "node": ">= 0.4"
+ }
+ },
+ "node_modules/media-typer": {
+ "version": "0.3.0",
+ "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz",
+ "integrity": "sha512-dq+qelQ9akHpcOl/gUVRTxVIOkAJ1wR3QAvb4RsVjS8oVoFjDGTc679wJYmUmknUF5HwMLOgb5O+a3KxfWapPQ==",
+ "engines": {
+ "node": ">= 0.6"
+ }
+ },
+ "node_modules/merge-descriptors": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz",
+ "integrity": "sha512-cCi6g3/Zr1iqQi6ySbseM1Xvooa98N0w31jzUYrXPX2xqObmFGHJ0tQ5u74H3mVh7wLouTseZyYIq39g8cNp1w=="
+ },
+ "node_modules/methods": {
+ "version": "1.1.2",
+ "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz",
+ "integrity": "sha512-iclAHeNqNm68zFtnZ0e+1L2yUIdvzNoauKU4WBA3VvH/vPFieF7qfRlwUZU+DA9P9bPXIS90ulxoUoCH23sV2w==",
+ "engines": {
+ "node": ">= 0.6"
+ }
+ },
+ "node_modules/mime": {
+ "version": "1.4.1",
+ "resolved": "https://registry.npmjs.org/mime/-/mime-1.4.1.tgz",
+ "integrity": "sha512-KI1+qOZu5DcW6wayYHSzR/tXKCDC5Om4s1z2QJjDULzLcmf3DvzS7oluY4HCTrc+9FiKmWUgeNLg7W3uIQvxtQ==",
+ "bin": {
+ "mime": "cli.js"
+ }
+ },
+ "node_modules/mime-db": {
+ "version": "1.52.0",
+ "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz",
+ "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==",
+ "engines": {
+ "node": ">= 0.6"
+ }
+ },
+ "node_modules/mime-types": {
+ "version": "2.1.35",
+ "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz",
+ "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==",
+ "dependencies": {
+ "mime-db": "1.52.0"
+ },
+ "engines": {
+ "node": ">= 0.6"
+ }
+ },
+ "node_modules/minimatch": {
+ "version": "3.1.2",
+ "resolved": "https://registry.npmjs.org/minimatch/-/minimatch-3.1.2.tgz",
+ "integrity": "sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw==",
+ "dependencies": {
+ "brace-expansion": "^1.1.7"
+ },
+ "engines": {
+ "node": "*"
+ }
+ },
+ "node_modules/morgan": {
+ "version": "1.9.1",
+ "resolved": "https://registry.npmjs.org/morgan/-/morgan-1.9.1.tgz",
+ "integrity": "sha512-HQStPIV4y3afTiCYVxirakhlCfGkI161c76kKFca7Fk1JusM//Qeo1ej2XaMniiNeaZklMVrh3vTtIzpzwbpmA==",
+ "dependencies": {
+ "basic-auth": "~2.0.0",
+ "debug": "2.6.9",
+ "depd": "~1.1.2",
+ "on-finished": "~2.3.0",
+ "on-headers": "~1.0.1"
+ },
+ "engines": {
+ "node": ">= 0.8.0"
+ }
+ },
+ "node_modules/ms": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz",
+ "integrity": "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A=="
+ },
+ "node_modules/mysql2": {
+ "version": "3.15.0",
+ "resolved": "https://registry.npmjs.org/mysql2/-/mysql2-3.15.0.tgz",
+ "integrity": "sha512-tT6pomf5Z/I7Jzxu8sScgrYBMK9bUFWd7Kbo6Fs1L0M13OOIJ/ZobGKS3Z7tQ8Re4lj+LnLXIQVZZxa3fhYKzA==",
+ "dependencies": {
+ "aws-ssl-profiles": "^1.1.1",
+ "denque": "^2.1.0",
+ "generate-function": "^2.3.1",
+ "iconv-lite": "^0.7.0",
+ "long": "^5.2.1",
+ "lru.min": "^1.0.0",
+ "named-placeholders": "^1.1.3",
+ "seq-queue": "^0.0.5",
+ "sqlstring": "^2.3.2"
+ },
+ "engines": {
+ "node": ">= 8.0"
+ }
+ },
+ "node_modules/mysql2/node_modules/iconv-lite": {
+ "version": "0.7.0",
+ "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.7.0.tgz",
+ "integrity": "sha512-cf6L2Ds3h57VVmkZe+Pn+5APsT7FpqJtEhhieDCvrE2MK5Qk9MyffgQyuxQTm6BChfeZNtcOLHp9IcWRVcIcBQ==",
+ "dependencies": {
+ "safer-buffer": ">= 2.1.2 < 3.0.0"
+ },
+ "engines": {
+ "node": ">=0.10.0"
+ },
+ "funding": {
+ "type": "opencollective",
+ "url": "https://opencollective.com/express"
+ }
+ },
+ "node_modules/named-placeholders": {
+ "version": "1.1.3",
+ "resolved": "https://registry.npmjs.org/named-placeholders/-/named-placeholders-1.1.3.tgz",
+ "integrity": "sha512-eLoBxg6wE/rZkJPhU/xRX1WTpkFEwDJEN96oxFrTsqBdbT5ec295Q+CoHrL9IT0DipqKhmGcaZmwOt8OON5x1w==",
+ "dependencies": {
+ "lru-cache": "^7.14.1"
+ },
+ "engines": {
+ "node": ">=12.0.0"
+ }
+ },
+ "node_modules/negotiator": {
+ "version": "0.6.3",
+ "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.3.tgz",
+ "integrity": "sha512-+EUsqGPLsM+j/zdChZjsnX51g4XrHFOIXwfnCVPGlQk/k5giakcKsuxCObBRu6DSm9opw/O6slWbJdghQM4bBg==",
+ "engines": {
+ "node": ">= 0.6"
+ }
+ },
+ "node_modules/object-assign": {
+ "version": "4.1.1",
+ "resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz",
+ "integrity": "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==",
+ "engines": {
+ "node": ">=0.10.0"
+ }
+ },
+ "node_modules/object-inspect": {
+ "version": "1.13.4",
+ "resolved": "https://registry.npmjs.org/object-inspect/-/object-inspect-1.13.4.tgz",
+ "integrity": "sha512-W67iLl4J2EXEGTbfeHCffrjDfitvLANg0UlX3wFUUSTx92KXRFegMHUVgSqE+wvhAbi4WqjGg9czysTV2Epbew==",
+ "engines": {
+ "node": ">= 0.4"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/on-finished": {
+ "version": "2.3.0",
+ "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz",
+ "integrity": "sha512-ikqdkGAAyf/X/gPhXGvfgAytDZtDbr+bkNUJ0N9h5MI/dmdgCs3l6hoHrcUv41sRKew3jIwrp4qQDXiK99Utww==",
+ "dependencies": {
+ "ee-first": "1.1.1"
+ },
+ "engines": {
+ "node": ">= 0.8"
+ }
+ },
+ "node_modules/on-headers": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/on-headers/-/on-headers-1.0.2.tgz",
+ "integrity": "sha512-pZAE+FJLoyITytdqK0U5s+FIpjN0JP3OzFi/u8Rx+EV5/W+JTWGXG8xFzevE7AjBfDqHv/8vL8qQsIhHnqRkrA==",
+ "engines": {
+ "node": ">= 0.8"
+ }
+ },
+ "node_modules/once": {
+ "version": "1.4.0",
+ "resolved": "https://registry.npmjs.org/once/-/once-1.4.0.tgz",
+ "integrity": "sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w==",
+ "dependencies": {
+ "wrappy": "1"
+ }
+ },
+ "node_modules/openapi-types": {
+ "version": "12.1.3",
+ "resolved": "https://registry.npmjs.org/openapi-types/-/openapi-types-12.1.3.tgz",
+ "integrity": "sha512-N4YtSYJqghVu4iek2ZUvcN/0aqH1kRDuNqzcycDxhOUpg7GdvLa2F3DgS6yBNhInhv2r/6I0Flkn7CqL8+nIcw==",
+ "peer": true
+ },
+ "node_modules/parseurl": {
+ "version": "1.3.3",
+ "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz",
+ "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==",
+ "engines": {
+ "node": ">= 0.8"
+ }
+ },
+ "node_modules/path-is-absolute": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/path-is-absolute/-/path-is-absolute-1.0.1.tgz",
+ "integrity": "sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg==",
+ "engines": {
+ "node": ">=0.10.0"
+ }
+ },
+ "node_modules/path-to-regexp": {
+ "version": "0.1.7",
+ "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz",
+ "integrity": "sha512-5DFkuoqlv1uYQKxy8omFBeJPQcdoE07Kv2sferDCrAq1ohOU+MSDswDIbnx3YAM60qIOnYa53wBhXW0EbMonrQ=="
+ },
+ "node_modules/proxy-addr": {
+ "version": "2.0.7",
+ "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.7.tgz",
+ "integrity": "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg==",
+ "dependencies": {
+ "forwarded": "0.2.0",
+ "ipaddr.js": "1.9.1"
+ },
+ "engines": {
+ "node": ">= 0.10"
+ }
+ },
+ "node_modules/qs": {
+ "version": "6.5.2",
+ "resolved": "https://registry.npmjs.org/qs/-/qs-6.5.2.tgz",
+ "integrity": "sha512-N5ZAX4/LxJmF+7wN74pUD6qAh9/wnvdQcjq9TZjevvXzSUo7bfmw91saqMjzGS2xq91/odN2dW/WOl7qQHNDGA==",
+ "engines": {
+ "node": ">=0.6"
+ }
+ },
+ "node_modules/range-parser": {
+ "version": "1.2.1",
+ "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz",
+ "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==",
+ "engines": {
+ "node": ">= 0.6"
+ }
+ },
+ "node_modules/raw-body": {
+ "version": "3.0.1",
+ "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-3.0.1.tgz",
+ "integrity": "sha512-9G8cA+tuMS75+6G/TzW8OtLzmBDMo8p1JRxN5AZ+LAp8uxGA8V8GZm4GQ4/N5QNQEnLmg6SS7wyuSmbKepiKqA==",
+ "dependencies": {
+ "bytes": "3.1.2",
+ "http-errors": "2.0.0",
+ "iconv-lite": "0.7.0",
+ "unpipe": "1.0.0"
+ },
+ "engines": {
+ "node": ">= 0.10"
+ }
+ },
+ "node_modules/raw-body/node_modules/depd": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/depd/-/depd-2.0.0.tgz",
+ "integrity": "sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw==",
+ "engines": {
+ "node": ">= 0.8"
+ }
+ },
+ "node_modules/raw-body/node_modules/http-errors": {
+ "version": "2.0.0",
+ "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-2.0.0.tgz",
+ "integrity": "sha512-FtwrG/euBzaEjYeRqOgly7G0qviiXoJWnvEH2Z1plBdXgbyjv34pHTSb9zoeHMyDy33+DWy5Wt9Wo+TURtOYSQ==",
+ "dependencies": {
+ "depd": "2.0.0",
+ "inherits": "2.0.4",
+ "setprototypeof": "1.2.0",
+ "statuses": "2.0.1",
+ "toidentifier": "1.0.1"
+ },
+ "engines": {
+ "node": ">= 0.8"
+ }
+ },
+ "node_modules/raw-body/node_modules/iconv-lite": {
+ "version": "0.7.0",
+ "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.7.0.tgz",
+ "integrity": "sha512-cf6L2Ds3h57VVmkZe+Pn+5APsT7FpqJtEhhieDCvrE2MK5Qk9MyffgQyuxQTm6BChfeZNtcOLHp9IcWRVcIcBQ==",
+ "dependencies": {
+ "safer-buffer": ">= 2.1.2 < 3.0.0"
+ },
+ "engines": {
+ "node": ">=0.10.0"
+ },
+ "funding": {
+ "type": "opencollective",
+ "url": "https://opencollective.com/express"
+ }
+ },
+ "node_modules/raw-body/node_modules/inherits": {
+ "version": "2.0.4",
+ "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz",
+ "integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ=="
+ },
+ "node_modules/raw-body/node_modules/setprototypeof": {
+ "version": "1.2.0",
+ "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.2.0.tgz",
+ "integrity": "sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw=="
+ },
+ "node_modules/raw-body/node_modules/statuses": {
+ "version": "2.0.1",
+ "resolved": "https://registry.npmjs.org/statuses/-/statuses-2.0.1.tgz",
+ "integrity": "sha512-RwNA9Z/7PrK06rYLIzFMlaF+l73iwpzsqRIFgbMLbTcLD6cOao82TaWefPXQvB2fOC4AjuYSEndS7N/mTCbkdQ==",
+ "engines": {
+ "node": ">= 0.8"
+ }
+ },
+ "node_modules/safe-buffer": {
+ "version": "5.1.2",
+ "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz",
+ "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g=="
+ },
+ "node_modules/safer-buffer": {
+ "version": "2.1.2",
+ "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz",
+ "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg=="
+ },
+ "node_modules/semver": {
+ "version": "7.7.2",
+ "resolved": "https://registry.npmjs.org/semver/-/semver-7.7.2.tgz",
+ "integrity": "sha512-RF0Fw+rO5AMf9MAyaRXI4AV0Ulj5lMHqVxxdSgiVbixSCXoEmmX/jk0CuJw4+3SqroYO9VoUh+HcuJivvtJemA==",
+ "bin": {
+ "semver": "bin/semver.js"
+ },
+ "engines": {
+ "node": ">=10"
+ }
+ },
+ "node_modules/send": {
+ "version": "0.16.2",
+ "resolved": "https://registry.npmjs.org/send/-/send-0.16.2.tgz",
+ "integrity": "sha512-E64YFPUssFHEFBvpbbjr44NCLtI1AohxQ8ZSiJjQLskAdKuriYEP6VyGEsRDH8ScozGpkaX1BGvhanqCwkcEZw==",
+ "dependencies": {
+ "debug": "2.6.9",
+ "depd": "~1.1.2",
+ "destroy": "~1.0.4",
+ "encodeurl": "~1.0.2",
+ "escape-html": "~1.0.3",
+ "etag": "~1.8.1",
+ "fresh": "0.5.2",
+ "http-errors": "~1.6.2",
+ "mime": "1.4.1",
+ "ms": "2.0.0",
+ "on-finished": "~2.3.0",
+ "range-parser": "~1.2.0",
+ "statuses": "~1.4.0"
+ },
+ "engines": {
+ "node": ">= 0.8.0"
+ }
+ },
+ "node_modules/seq-queue": {
+ "version": "0.0.5",
+ "resolved": "https://registry.npmjs.org/seq-queue/-/seq-queue-0.0.5.tgz",
+ "integrity": "sha512-hr3Wtp/GZIc/6DAGPDcV4/9WoZhjrkXsi5B/07QgX8tsdc6ilr7BFM6PM6rbdAX1kFSDYeZGLipIZZKyQP0O5Q=="
+ },
+ "node_modules/serve-static": {
+ "version": "1.13.2",
+ "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.13.2.tgz",
+ "integrity": "sha512-p/tdJrO4U387R9oMjb1oj7qSMaMfmOyd4j9hOFoxZe2baQszgHcSWjuya/CiT5kgZZKRudHNOA0pYXOl8rQ5nw==",
+ "dependencies": {
+ "encodeurl": "~1.0.2",
+ "escape-html": "~1.0.3",
+ "parseurl": "~1.3.2",
+ "send": "0.16.2"
+ },
+ "engines": {
+ "node": ">= 0.8.0"
+ }
+ },
+ "node_modules/setprototypeof": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.0.tgz",
+ "integrity": "sha512-BvE/TwpZX4FXExxOxZyRGQQv651MSwmWKZGqvmPcRIjDqWub67kTKuIMx43cZZrS/cBBzwBcNDWoFxt2XEFIpQ=="
+ },
+ "node_modules/side-channel": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/side-channel/-/side-channel-1.1.0.tgz",
+ "integrity": "sha512-ZX99e6tRweoUXqR+VBrslhda51Nh5MTQwou5tnUDgbtyM0dBgmhEDtWGP/xbKn6hqfPRHujUNwz5fy/wbbhnpw==",
+ "dependencies": {
+ "es-errors": "^1.3.0",
+ "object-inspect": "^1.13.3",
+ "side-channel-list": "^1.0.0",
+ "side-channel-map": "^1.0.1",
+ "side-channel-weakmap": "^1.0.2"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/side-channel-list": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/side-channel-list/-/side-channel-list-1.0.0.tgz",
+ "integrity": "sha512-FCLHtRD/gnpCiCHEiJLOwdmFP+wzCmDEkc9y7NsYxeF4u7Btsn1ZuwgwJGxImImHicJArLP4R0yX4c2KCrMrTA==",
+ "dependencies": {
+ "es-errors": "^1.3.0",
+ "object-inspect": "^1.13.3"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/side-channel-map": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/side-channel-map/-/side-channel-map-1.0.1.tgz",
+ "integrity": "sha512-VCjCNfgMsby3tTdo02nbjtM/ewra6jPHmpThenkTYh8pG9ucZ/1P8So4u4FGBek/BjpOVsDCMoLA/iuBKIFXRA==",
+ "dependencies": {
+ "call-bound": "^1.0.2",
+ "es-errors": "^1.3.0",
+ "get-intrinsic": "^1.2.5",
+ "object-inspect": "^1.13.3"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/side-channel-weakmap": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/side-channel-weakmap/-/side-channel-weakmap-1.0.2.tgz",
+ "integrity": "sha512-WPS/HvHQTYnHisLo9McqBHOJk2FkHO/tlpvldyrnem4aeQp4hai3gythswg6p01oSoTl58rcpiFAjF2br2Ak2A==",
+ "dependencies": {
+ "call-bound": "^1.0.2",
+ "es-errors": "^1.3.0",
+ "get-intrinsic": "^1.2.5",
+ "object-inspect": "^1.13.3",
+ "side-channel-map": "^1.0.1"
+ },
+ "engines": {
+ "node": ">= 0.4"
+ },
+ "funding": {
+ "url": "https://github.com/sponsors/ljharb"
+ }
+ },
+ "node_modules/sqlstring": {
+ "version": "2.3.3",
+ "resolved": "https://registry.npmjs.org/sqlstring/-/sqlstring-2.3.3.tgz",
+ "integrity": "sha512-qC9iz2FlN7DQl3+wjwn3802RTyjCx7sDvfQEXchwa6CWOx07/WVfh91gBmQ9fahw8snwGEWU3xGzOt4tFyHLxg==",
+ "engines": {
+ "node": ">= 0.6"
+ }
+ },
+ "node_modules/statuses": {
+ "version": "1.4.0",
+ "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.4.0.tgz",
+ "integrity": "sha512-zhSCtt8v2NDrRlPQpCNtw/heZLtfUDqxBM1udqikb/Hbk52LK4nQSwr10u77iopCW5LsyHpuXS0GnEc48mLeew==",
+ "engines": {
+ "node": ">= 0.6"
+ }
+ },
+ "node_modules/swagger-jsdoc": {
+ "version": "6.2.8",
+ "resolved": "https://registry.npmjs.org/swagger-jsdoc/-/swagger-jsdoc-6.2.8.tgz",
+ "integrity": "sha512-VPvil1+JRpmJ55CgAtn8DIcpBs0bL5L3q5bVQvF4tAW/k/9JYSj7dCpaYCAv5rufe0vcCbBRQXGvzpkWjvLklQ==",
+ "dependencies": {
+ "commander": "6.2.0",
+ "doctrine": "3.0.0",
+ "glob": "7.1.6",
+ "lodash.mergewith": "^4.6.2",
+ "swagger-parser": "^10.0.3",
+ "yaml": "2.0.0-1"
+ },
+ "bin": {
+ "swagger-jsdoc": "bin/swagger-jsdoc.js"
+ },
+ "engines": {
+ "node": ">=12.0.0"
+ }
+ },
+ "node_modules/swagger-parser": {
+ "version": "10.0.3",
+ "resolved": "https://registry.npmjs.org/swagger-parser/-/swagger-parser-10.0.3.tgz",
+ "integrity": "sha512-nF7oMeL4KypldrQhac8RyHerJeGPD1p2xDh900GPvc+Nk7nWP6jX2FcC7WmkinMoAmoO774+AFXcWsW8gMWEIg==",
+ "dependencies": {
+ "@apidevtools/swagger-parser": "10.0.3"
+ },
+ "engines": {
+ "node": ">=10"
+ }
+ },
+ "node_modules/swagger-ui-dist": {
+ "version": "5.29.0",
+ "resolved": "https://registry.npmjs.org/swagger-ui-dist/-/swagger-ui-dist-5.29.0.tgz",
+ "integrity": "sha512-gqs7Md3AxP4mbpXAq31o5QW+wGUZsUzVatg70yXpUR245dfIis5jAzufBd+UQM/w2xSfrhvA1eqsrgnl2PbezQ==",
+ "dependencies": {
+ "@scarf/scarf": "=1.4.0"
+ }
+ },
+ "node_modules/swagger-ui-express": {
+ "version": "5.0.1",
+ "resolved": "https://registry.npmjs.org/swagger-ui-express/-/swagger-ui-express-5.0.1.tgz",
+ "integrity": "sha512-SrNU3RiBGTLLmFU8GIJdOdanJTl4TOmT27tt3bWWHppqYmAZ6IDuEuBvMU6nZq0zLEe6b/1rACXCgLZqO6ZfrA==",
+ "dependencies": {
+ "swagger-ui-dist": ">=5.0.0"
+ },
+ "engines": {
+ "node": ">= v0.10.32"
+ },
+ "peerDependencies": {
+ "express": ">=4.0.0 || >=5.0.0-beta"
+ }
+ },
+ "node_modules/toidentifier": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.1.tgz",
+ "integrity": "sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA==",
+ "engines": {
+ "node": ">=0.6"
+ }
+ },
+ "node_modules/type-is": {
+ "version": "1.6.18",
+ "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz",
+ "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==",
+ "dependencies": {
+ "media-typer": "0.3.0",
+ "mime-types": "~2.1.24"
+ },
+ "engines": {
+ "node": ">= 0.6"
+ }
+ },
+ "node_modules/unpipe": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz",
+ "integrity": "sha512-pjy2bYhSsufwWlKwPc+l3cN7+wuJlK6uz0YdJEOlQDbl6jo/YlPi4mb8agUkVC8BF7V8NuzeyPNqRksA3hztKQ==",
+ "engines": {
+ "node": ">= 0.8"
+ }
+ },
+ "node_modules/utils-merge": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz",
+ "integrity": "sha512-pMZTvIkT1d+TFGvDOqodOclx0QWkkgi6Tdoa8gC8ffGAAqz9pzPTZWAybbsHHoED/ztMtkv/VoYTYyShUn81hA==",
+ "engines": {
+ "node": ">= 0.4.0"
+ }
+ },
+ "node_modules/validator": {
+ "version": "13.15.15",
+ "resolved": "https://registry.npmjs.org/validator/-/validator-13.15.15.tgz",
+ "integrity": "sha512-BgWVbCI72aIQy937xbawcs+hrVaN/CZ2UwutgaJ36hGqRrLNM+f5LUT/YPRbo8IV/ASeFzXszezV+y2+rq3l8A==",
+ "engines": {
+ "node": ">= 0.10"
+ }
+ },
+ "node_modules/vary": {
+ "version": "1.1.2",
+ "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz",
+ "integrity": "sha512-BNGbWLfd0eUPabhkXUVm0j8uuvREyTh5ovRa/dyow/BqAbZJyC+5fU+IzQOzmAKzYqYRAISoRhdQr3eIZ/PXqg==",
+ "engines": {
+ "node": ">= 0.8"
+ }
+ },
+ "node_modules/wrappy": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz",
+ "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ=="
+ },
+ "node_modules/yaml": {
+ "version": "2.0.0-1",
+ "resolved": "https://registry.npmjs.org/yaml/-/yaml-2.0.0-1.tgz",
+ "integrity": "sha512-W7h5dEhywMKenDJh2iX/LABkbFnBxasD27oyXWDS/feDsxiw0dD5ncXdYXgkvAsXIY2MpW/ZKkr9IU30DBdMNQ==",
+ "engines": {
+ "node": ">= 6"
+ }
+ },
+ "node_modules/z-schema": {
+ "version": "5.0.5",
+ "resolved": "https://registry.npmjs.org/z-schema/-/z-schema-5.0.5.tgz",
+ "integrity": "sha512-D7eujBWkLa3p2sIpJA0d1pr7es+a7m0vFAnZLlCEKq/Ij2k0MLi9Br2UPxoxdYystm5K1yeBGzub0FlYUEWj2Q==",
+ "dependencies": {
+ "lodash.get": "^4.4.2",
+ "lodash.isequal": "^4.5.0",
+ "validator": "^13.7.0"
+ },
+ "bin": {
+ "z-schema": "bin/z-schema"
+ },
+ "engines": {
+ "node": ">=8.0.0"
+ },
+ "optionalDependencies": {
+ "commander": "^9.4.1"
+ }
+ },
+ "node_modules/z-schema/node_modules/commander": {
+ "version": "9.5.0",
+ "resolved": "https://registry.npmjs.org/commander/-/commander-9.5.0.tgz",
+ "integrity": "sha512-KRs7WVDKg86PWiuAqhDrAQnTXZKraVcCc6vFdL14qrZ/DcWwuRo7VoiYXalXO7S5GKpqYiVEwCbgFDfxNHKJBQ==",
+ "optional": true,
+ "engines": {
+ "node": "^12.20.0 || >=14"
+ }
+ }
+ }
+}
diff --git a/package.json b/package.json
new file mode 100644
index 0000000..e52ea20
--- /dev/null
+++ b/package.json
@@ -0,0 +1,21 @@
+{
+ "name": "middle-office-system",
+ "version": "0.0.0",
+ "private": true,
+ "scripts": {
+ "start": "node ./bin/www"
+ },
+ "dependencies": {
+ "bcryptjs": "^2.4.3",
+ "body-parser": "^2.2.0",
+ "cookie-parser": "~1.4.4",
+ "cors": "^2.8.5",
+ "debug": "~2.6.9",
+ "express": "~4.16.1",
+ "jsonwebtoken": "^9.0.2",
+ "morgan": "~1.9.1",
+ "mysql2": "^3.15.0",
+ "swagger-jsdoc": "^6.2.8",
+ "swagger-ui-express": "^5.0.1"
+ }
+}
diff --git a/public/index.html b/public/index.html
new file mode 100644
index 0000000..ab1ad8a
--- /dev/null
+++ b/public/index.html
@@ -0,0 +1,13 @@
+
+
+
+ Express
+
+
+
+
+ Express
+ Welcome to Express
+
+
+
diff --git a/public/stylesheets/style.css b/public/stylesheets/style.css
new file mode 100644
index 0000000..9453385
--- /dev/null
+++ b/public/stylesheets/style.css
@@ -0,0 +1,8 @@
+body {
+ padding: 50px;
+ font: 14px "Lucida Grande", Helvetica, Arial, sans-serif;
+}
+
+a {
+ color: #00B7FF;
+}
diff --git a/routes/auth.js b/routes/auth.js
new file mode 100644
index 0000000..b469a18
--- /dev/null
+++ b/routes/auth.js
@@ -0,0 +1,346 @@
+const express = require('express');
+const bcrypt = require('bcryptjs');
+const jwt = require('jsonwebtoken');
+const {getDB} = require('../database');
+
+const router = express.Router();
+const JWT_SECRET = process.env.JWT_SECRET || 'your-secret-key';
+router.post('/register', async (req, res) => {
+ try {
+ const db = getDB();
+ await db.query('START TRANSACTION');
+
+ const {
+ username,
+ phone,
+ password,
+ city,
+ district_id: district,
+ province,
+ inviter = null,
+ captchaId,
+ captchaText,
+ smsCode, // 短信验证码
+ role = 'user'
+ } = req.body;
+
+ if (!username || !phone || !password || !city || !district || !province) {
+ return res.status(400).json({success: false, message: '用户名、手机号、密码、城市和区域不能为空'});
+ }
+
+ if (!captchaId || !captchaText) {
+ return res.status(400).json({success: false, message: '图形验证码不能为空'});
+ }
+ const storedCaptcha = global.captchaStore.get(captchaId);
+ console.log(storedCaptcha);
+
+ if (!storedCaptcha) {
+ return res.status(400).json({
+ success: false,
+ message: '验证码不存在或已过期'
+ });
+ }
+
+ // 检查是否过期
+ if (Date.now() > storedCaptcha.expires) {
+ global.captchaStore.delete(captchaId);
+ return res.status(400).json({
+ success: false,
+ message: '验证码已过期'
+ });
+ }
+
+ // 验证验证码(不区分大小写)
+ const isValid = storedCaptcha.text === captchaText.toLowerCase();
+
+ // 删除已验证的验证码
+ global.captchaStore.delete(captchaId);
+
+ if (!isValid) {
+ return res.status(400).json({
+ success: false,
+ message: '验证码错误'
+ });
+ }
+ if (!smsCode) {
+ return res.status(400).json({success: false, message: '短信验证码不能为空'});
+ }
+ // 验证短信验证码
+ const smsAPI = require('./sms');
+ const smsValid = smsAPI.verifySMSCode(phone, smsCode);
+ if (!smsValid) {
+ return res.status(400).json({success: false, message: '短信验证码错误或已过期'});
+ }
+
+ // 验证手机号格式
+ const phoneRegex = /^1[3-9]\d{9}$/;
+ if (!phoneRegex.test(phone)) {
+ return res.status(400).json({success: false, message: '手机号格式不正确'});
+ }
+
+
+ // 检查用户是否已存在
+ const [existingUsers] = await db.execute(
+ 'SELECT id, payment_status FROM users WHERE username = ? OR phone = ?',
+ [username, phone]
+ );
+
+ if (existingUsers.length > 0) {
+ return res.status(400).json({success: false, message: '用户名或手机号已存在'});
+ }
+
+ // 加密密码
+ const hashedPassword = await bcrypt.hash(password, 10);
+
+ // 创建用户(初始状态为未支付)
+ const [result] = await db.execute(
+ 'INSERT INTO users (username, phone, password, role, points, audit_status, city, district_id, payment_status, province, inviter) VALUES (?, ?, ?, ?, ?, ?, ?, ?, "unpaid", ?, ?)',
+ [username, phone, hashedPassword, role, 0, 'pending', city, district, province, inviter]
+ );
+
+ const userId = result.insertId;
+ await db.query('COMMIT');
+
+ // 生成JWT token(用于支付流程)
+ const token = jwt.sign(
+ {userId: userId, username, role},
+ JWT_SECRET,
+ {expiresIn: '24h'}
+ );
+
+ res.status(201).json({
+ success: true,
+ message: '用户信息创建成功,请完成支付以激活账户',
+ token,
+ user: {
+ id: userId,
+ username,
+ phone,
+ role,
+ points: 0,
+ audit_status: 'pending',
+ city,
+ district,
+ paymentStatus: 'unpaid'
+ },
+ needPayment: true
+ });
+ } catch (error) {
+ try {
+ // await getDB().query('ROLLBACK');
+ } catch (rollbackError) {
+ console.error('回滚错误:', rollbackError);
+ }
+ console.error('注册错误详情:', error);
+ console.error('错误堆栈:', error.stack);
+ res.status(500).json({
+ success: false,
+ message: '注册失败',
+ error: process.env.NODE_ENV === 'development' ? error.message : undefined
+ });
+ }
+});
+
+
+router.post('/login', async (req, res) => {
+ console.log(123456)
+ try {
+ const db = getDB();
+ const {username, password, captchaId, captchaText} = req.body;
+
+ if (!username || !password) {
+ return res.status(400).json({success: false, message: '用户名和密码不能为空'});
+ }
+
+ if (!captchaId || !captchaText) {
+ return res.status(400).json({success: false, message: '验证码不能为空'});
+ }
+ // 获取存储的验证码
+ const storedCaptcha = global.captchaStore.get(captchaId);
+ console.log(storedCaptcha);
+
+ if (!storedCaptcha) {
+ return res.status(400).json({
+ success: false,
+ message: '验证码不存在或已过期'
+ });
+ }
+
+ // 检查是否过期
+ if (Date.now() > storedCaptcha.expires) {
+ global.captchaStore.delete(captchaId);
+ return res.status(400).json({
+ success: false,
+ message: '验证码已过期'
+ });
+ }
+
+ // 验证验证码(不区分大小写)
+ const isValid = storedCaptcha.text === captchaText.toLowerCase();
+
+ // 删除已验证的验证码
+ global.captchaStore.delete(captchaId);
+
+ if (!isValid) {
+ return res.status(400).json({
+ success: false,
+ message: '验证码错误'
+ });
+ }
+
+ // 注意:验证码已在前端通过 /captcha/verify 接口验证过,这里不再重复验证
+
+ // 查找用户(包含支付状态)
+ console.log('登录尝试 - 用户名:', username);
+ const [users] = await db.execute(
+ 'SELECT * FROM users WHERE username = ?',
+ [username]
+ );
+
+ console.log('查找到的用户数量:', users.length);
+ if (users.length === 0) {
+ console.log('用户不存在:', username);
+ return res.status(401).json({success: false, message: '用户名或密码错误'});
+ }
+
+ const user = users[0];
+ console.log('找到用户:', user.username, '密码长度:', user.password ? user.password.length : 'null');
+
+ // 验证密码
+ console.log('验证密码 - 输入密码:', password, '数据库密码前10位:', user.password ? user.password.substring(0, 10) : 'null');
+ const isValidPassword = await bcrypt.compare(password, user.password);
+ console.log('密码验证结果:', isValidPassword);
+
+ if (!isValidPassword) {
+ console.log('密码验证失败');
+ return res.status(401).json({success: false, message: '用户名或密码错误'});
+ }
+
+ // 检查支付状态(管理员除外)
+ if (user.role !== 'admin' && user.payment_status === 'unpaid') {
+ const token = jwt.sign(
+ {userId: user.id, username: user.username, role: user.role},
+ JWT_SECRET,
+ {expiresIn: '5m'}
+ );
+ return res.status(200).json({
+ success: false,
+ message: '您的账户尚未激活,请完成支付后再登录',
+ needPayment: true,
+ user: user[0],
+ token
+ });
+ }
+
+ // 检查用户审核状态(管理员除外,只阻止被拒绝的用户)
+ if (user.role !== 'admin' && user.audit_status === 'rejected') {
+ return res.status(403).json({success: false, message: '您的账户审核未通过,请联系管理员'});
+ }
+ // 待审核用户可以正常登录使用系统,但匹配功能会有限制
+
+ // 生成JWT token
+ const token = jwt.sign(
+ {userId: user.id, username: user.username, role: user.role},
+ JWT_SECRET,
+ {expiresIn: '24h'}
+ );
+ const [is_distribution] = await db.execute(`
+ SELECT *
+ FROM distribution
+ WHERE user_id = ?`, [user.id]);
+ user.distribution = is_distribution.length > 0 ? true : false;
+ res.json({
+ success: true,
+ message: '登录成功',
+ token,
+ user
+ });
+ } catch (error) {
+ console.error('登录错误:', error);
+ res.status(500).json({success: false, message: '登录失败'});
+ }
+});
+
+// 验证token中间件
+const authenticateToken = (req, res, next) => {
+ const authHeader = req.headers['authorization'];
+ const token = authHeader && authHeader.split(' ')[1];
+
+ if (!token) {
+ return res.status(401).json({success: false, message: '访问令牌缺失'});
+ }
+
+ jwt.verify(token, JWT_SECRET, (err, user) => {
+ if (err) {
+ return res.status(403).json({success: false, message: '访问令牌无效'});
+ }
+ req.user = user;
+ next();
+ });
+};
+
+// 获取当前用户信息
+router.get('/me', authenticateToken, async (req, res) => {
+ try {
+ const db = getDB();
+ const [users] = await db.execute(
+ 'SELECT id, username, role, avatar, points, created_at FROM users WHERE id = ?',
+ [req.user.userId]
+ );
+
+ if (users.length === 0) {
+ return res.status(404).json({success: false, message: '用户不存在'});
+ }
+
+ res.json({success: true, user: users[0]});
+ } catch (error) {
+ console.error('获取用户信息错误:', error);
+ res.status(500).json({success: false, message: '获取用户信息失败'});
+ }
+});
+
+// 修改密码
+router.put('/change-password', authenticateToken, async (req, res) => {
+ try {
+ const db = getDB();
+ const {currentPassword, newPassword} = req.body;
+
+ if (!currentPassword || !newPassword) {
+ return res.status(400).json({success: false, message: '旧密码和新密码不能为空'});
+ }
+
+ // 获取用户当前密码
+ const [users] = await db.execute(
+ 'SELECT password FROM users WHERE id = ?',
+ [req.user.userId]
+ );
+
+ if (users.length === 0) {
+ return res.status(404).json({success: false, message: '用户不存在'});
+ }
+
+ // 验证旧密码
+ const isValidPassword = await bcrypt.compare(currentPassword, users[0].password);
+
+ if (!isValidPassword) {
+ return res.status(400).json({success: false, message: '旧密码错误'});
+ }
+
+ // 加密新密码
+ const hashedNewPassword = await bcrypt.hash(newPassword, 10);
+
+ // 更新密码
+ await db.execute(
+ 'UPDATE users SET password = ? WHERE id = ?',
+ [hashedNewPassword, req.user.userId]
+ );
+
+ res.json({success: true, message: '密码修改成功'});
+ } catch (error) {
+ console.error('修改密码错误:', error);
+ res.status(500).json({success: false, message: '修改密码失败'});
+ }
+});
+
+module.exports = router;
+module.exports.authenticateToken = authenticateToken;
\ No newline at end of file
diff --git a/routes/captcha.js b/routes/captcha.js
new file mode 100644
index 0000000..4fb4f11
--- /dev/null
+++ b/routes/captcha.js
@@ -0,0 +1,220 @@
+const express = require('express');
+const crypto = require('crypto');
+const router = express.Router();
+
+
+
+// 内存存储验证码(生产环境建议使用Redis)
+
+
+/**
+ * 生成随机验证码字符串
+ * @param {number} length 验证码长度
+ * @returns {string} 验证码字符串
+ */
+function generateCaptchaText(length = 4) {
+ const chars = 'ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789';
+ let result = '';
+ for (let i = 0; i < length; i++) {
+ result += chars.charAt(Math.floor(Math.random() * chars.length));
+ }
+ return result;
+}
+
+/**
+ * 生成SVG验证码图片
+ * @param {string} text 验证码文本
+ * @returns {string} SVG字符串
+ */
+function generateCaptchaSVG(text) {
+ const width = 120;
+ const height = 40;
+ const fontSize = 18;
+
+ // 生成随机颜色
+ const getRandomColor = () => {
+ const colors = ['#FF6B6B', '#4ECDC4', '#45B7D1', '#96CEB4', '#FFEAA7', '#DDA0DD', '#98D8C8'];
+ return colors[Math.floor(Math.random() * colors.length)];
+ };
+
+ // 生成干扰线
+ const generateNoise = () => {
+ let noise = '';
+ for (let i = 0; i < 3; i++) {
+ const x1 = Math.random() * width;
+ const y1 = Math.random() * height;
+ const x2 = Math.random() * width;
+ const y2 = Math.random() * height;
+ noise += ``;
+ }
+ return noise;
+ };
+
+ // 生成干扰点
+ const generateDots = () => {
+ let dots = '';
+ for (let i = 0; i < 20; i++) {
+ const x = Math.random() * width;
+ const y = Math.random() * height;
+ const r = Math.random() * 2 + 1;
+ dots += ``;
+ }
+ return dots;
+ };
+
+ // 生成文字
+ let textElements = '';
+ const charWidth = width / text.length;
+
+ for (let i = 0; i < text.length; i++) {
+ const char = text[i];
+ const x = charWidth * i + charWidth / 2;
+ const y = height / 2 + fontSize / 3;
+ const rotation = (Math.random() - 0.5) * 30; // 随机旋转角度
+ const color = getRandomColor();
+
+ textElements += `
+
+ ${char}
+ `;
+ }
+
+ const svg = `
+ `;
+
+ return svg;
+}
+
+
+router.get('/generate', (req, res) => {
+ try {
+ // 生成验证码文本
+ const captchaText = generateCaptchaText();
+
+ // 生成唯一ID
+ const captchaId = crypto.randomUUID();
+
+ // 存储验证码(5分钟过期)
+ global.captchaStore.set(captchaId, {
+ text: captchaText.toLowerCase(), // 存储小写用于比较
+ expires: Date.now() + 5 * 60 * 1000 // 5分钟过期
+ });
+
+ // 生成SVG图片
+ const svgImage = generateCaptchaSVG(captchaText);
+ res.json({
+ success: true,
+ data: {
+ captchaId,
+ image: `data:image/svg+xml;base64,${Buffer.from(svgImage).toString('base64')}`
+ }
+ });
+ } catch (error) {
+ console.error('生成验证码失败:', error);
+ res.status(500).json({
+ success: false,
+ message: '生成验证码失败'
+ });
+ }
+});
+
+
+router.post('/verify', (req, res) => {
+ try {
+ const { captchaId, captchaText } = req.body;
+
+ if (!captchaId || !captchaText) {
+ return res.status(400).json({
+ success: false,
+ message: '验证码ID和验证码不能为空'
+ });
+ }
+
+ // 获取存储的验证码
+ const storedCaptcha = global.captchaStore.get(captchaId);
+
+ if (!storedCaptcha) {
+ return res.status(400).json({
+ success: false,
+ message: '验证码不存在或已过期'
+ });
+ }
+
+ // 检查是否过期
+ if (Date.now() > storedCaptcha.expires) {
+ global.captchaStore.delete(captchaId);
+ return res.status(400).json({
+ success: false,
+ message: '验证码已过期'
+ });
+ }
+
+ // 验证验证码(不区分大小写)
+ const isValid = storedCaptcha.text === captchaText.toLowerCase();
+
+ // 验证后删除验证码(无论成功失败)
+ global.captchaStore.delete(captchaId);
+
+ if (isValid) {
+ res.json({
+ success: true,
+ message: '验证码验证成功'
+ });
+ } else {
+ res.status(400).json({
+ success: false,
+ message: '验证码错误'
+ });
+ }
+ } catch (error) {
+ console.error('验证验证码失败:', error);
+ res.status(500).json({
+ success: false,
+ message: '验证验证码失败'
+ });
+ }
+});
+
+// 清理过期验证码的定时任务
+setInterval(() => {
+ const now = Date.now();
+ for (const [id, captcha] of global.captchaStore.entries()) {
+ if (now > captcha.expires) {
+ global.captchaStore.delete(id);
+ }
+ }
+}, 60 * 1000); // 每分钟清理一次
+
+// 导出验证函数供其他模块使用
+module.exports = router;
+module.exports.verifyCaptcha = (captchaId, captchaText) => {
+ const captcha = global.captchaStore.get(captchaId);
+ if (!captcha) {
+ return false; // 验证码不存在或已过期
+ }
+
+ if (captcha.text.toLowerCase() !== captchaText.toLowerCase()) {
+ return false; // 验证码错误
+ }
+
+ // 验证成功后删除验证码(一次性使用)
+ global.captchaStore.delete(captchaId);
+ return true;
+};
\ No newline at end of file
diff --git a/routes/common.js b/routes/common.js
new file mode 100644
index 0000000..d71ca27
--- /dev/null
+++ b/routes/common.js
@@ -0,0 +1,77 @@
+const express = require('express')
+const router = express.Router()
+const {getDB} = require('../database')
+
+/**
+ * @swagger
+ * /api/common/provinces:
+ * get:
+ * summary: 获取省份列表
+ * description: 获取省份列表
+ * responses:
+ * '200':
+ * description: 成功获取分类列表
+ */
+router.get('/provinces', async (req, res) => {
+ try {
+ // 按level分组数据
+ const regionsByLevel = {
+ 1: [], // 省份
+ 2: [], // 城市
+ 3: [] // 区县
+ };
+ if (!global.provinces) {
+ // 一次性获取所有区域数据(省、市、区县)
+ const [allRegions] = await getDB().execute(
+ `SELECT code, name as label, level, parent_code
+ FROM china_regions
+ WHERE level <= 3
+ ORDER BY level, code`
+ );
+
+
+ // 创建code到region的映射,便于快速查找
+ const regionMap = {};
+
+ // 分组并建立映射
+ allRegions.forEach(region => {
+ region.children = []; // 初始化children数组
+ regionsByLevel[region.level].push(region);
+ regionMap[region.code] = region;
+ });
+
+ // 构建层级关系:先处理区县到城市的关系
+ regionsByLevel[3].forEach(district => {
+ const parentCity = regionMap[district.parent_code];
+ if (parentCity) {
+ parentCity.children.push(district);
+ }
+ });
+
+ // 再处理城市到省份的关系
+ regionsByLevel[2].forEach(city => {
+ const parentProvince = regionMap[city.parent_code];
+ if (parentProvince) {
+ parentProvince.children.push(city);
+ }
+ });
+ global.provinces = regionsByLevel[1];
+ } else {
+ console.log('1111')
+ regionsByLevel[1] = global.provinces;
+ }
+
+
+ // 返回省份数据(已包含完整的层级结构)
+ res.json({
+ success: true,
+ data: regionsByLevel[1]
+ });
+ } catch (error) {
+ console.error('获取省份列表错误:', error);
+ res.status(500).json({message: '获取省份列表失败'});
+ }
+});
+
+
+module.exports = router
\ No newline at end of file
diff --git a/routes/index.js b/routes/index.js
new file mode 100644
index 0000000..9264798
--- /dev/null
+++ b/routes/index.js
@@ -0,0 +1,6 @@
+var express = require('express');
+var router = express.Router();
+
+
+
+module.exports = router;
diff --git a/routes/users.js b/routes/users.js
new file mode 100644
index 0000000..b5a7010
--- /dev/null
+++ b/routes/users.js
@@ -0,0 +1,9 @@
+var express = require('express');
+var router = express.Router();
+
+/* GET users listing. */
+router.get('/',function(req, res, next) {
+ res.send('respond with a resource');
+});
+
+module.exports = router;