midway-fatcms 0.0.1-beta.26 → 0.0.1-beta.29
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/config/config.default.js +6 -4
- package/dist/controller/home.controller.js +2 -1
- package/dist/controller/manage/CrudStandardDesignApi.js +8 -8
- package/dist/controller/render/AppRenderController.js +2 -1
- package/dist/middleware/global.middleware.js +22 -9
- package/dist/models/AsyncTaskModel.d.ts +2 -0
- package/dist/models/AsyncTaskModel.js +3 -1
- package/dist/models/RedisKeys.d.ts +2 -0
- package/dist/models/RedisKeys.js +2 -0
- package/dist/models/bizmodels.d.ts +1 -0
- package/dist/service/UserSessionService.d.ts +21 -0
- package/dist/service/UserSessionService.js +71 -1
- package/dist/service/WorkbenchService.d.ts +32 -0
- package/dist/service/WorkbenchService.js +54 -11
- package/dist/service/anyapi/AnyApiSandboxService.js +12 -12
- package/dist/service/asyncTask/AsyncTaskRunnerService.d.ts +14 -1
- package/dist/service/asyncTask/AsyncTaskRunnerService.js +70 -25
- package/dist/service/asyncTask/handler/ExportExcelAsyncTaskHandler.js +1 -1
- package/dist/service/asyncTask/handler/ExportExcelByInnerHttpHandler.js +11 -2
- package/dist/service/base/RedisCacheService.d.ts +7 -0
- package/dist/service/base/RedisCacheService.js +7 -0
- package/dist/service/crudstd/CrudStdService.d.ts +0 -6
- package/dist/service/crudstd/CrudStdService.js +13 -33
- package/package.json +1 -5
- package/src/config/config.default.ts +0 -207
- package/src/config/seed/aeskey.txt +0 -1
- package/src/config/utils.ts +0 -22
- package/src/configuration.ts +0 -109
- package/src/controller/base/BaseApiController.ts +0 -170
- package/src/controller/gateway/AnyApiGatewayController.ts +0 -33
- package/src/controller/gateway/AsyncTaskController.ts +0 -157
- package/src/controller/gateway/CrudMtdGatewayController.ts +0 -111
- package/src/controller/gateway/CrudStdGatewayController.ts +0 -101
- package/src/controller/gateway/DocGatewayController.ts +0 -173
- package/src/controller/gateway/FileController.ts +0 -109
- package/src/controller/gateway/ProxyApiGatewayController.ts +0 -47
- package/src/controller/gateway/PublicApiController.ts +0 -142
- package/src/controller/gateway/StaticController.ts +0 -298
- package/src/controller/helpers.controller.ts +0 -161
- package/src/controller/home.controller.ts +0 -64
- package/src/controller/manage/AnyApiMangeApi.ts +0 -66
- package/src/controller/manage/AppLogMangeApi.ts +0 -53
- package/src/controller/manage/AppMangeApi.ts +0 -53
- package/src/controller/manage/AppPageMangeApi.ts +0 -52
- package/src/controller/manage/AppSchemaHistoryApi.ts +0 -49
- package/src/controller/manage/CrudMethodsMangeApi.ts +0 -49
- package/src/controller/manage/CrudStandardDesignApi.ts +0 -346
- package/src/controller/manage/DataDictManageApi.ts +0 -78
- package/src/controller/manage/DeployManageApi.ts +0 -175
- package/src/controller/manage/DocLibManageApi.ts +0 -69
- package/src/controller/manage/DocManageApi.ts +0 -99
- package/src/controller/manage/FileManageApi.ts +0 -45
- package/src/controller/manage/LowCodeTplManageApi.ts +0 -52
- package/src/controller/manage/MenuManageApi.ts +0 -58
- package/src/controller/manage/ProxyApiMangeApi.ts +0 -52
- package/src/controller/manage/SuperAdminManageApi.ts +0 -139
- package/src/controller/manage/SysConfigMangeApi.ts +0 -95
- package/src/controller/manage/SystemInfoManageApi.ts +0 -53
- package/src/controller/manage/UserAccountManageApi.ts +0 -94
- package/src/controller/manage/WorkbenchMangeApi.ts +0 -72
- package/src/controller/myinfo/AuthController.ts +0 -108
- package/src/controller/myinfo/MyInfoController.ts +0 -32
- package/src/controller/render/AppRenderController.ts +0 -79
- package/src/controller/test.controller.ts +0 -37
- package/src/filter/default.filter.ts +0 -13
- package/src/filter/notfound.filter.ts +0 -10
- package/src/index.ts +0 -106
- package/src/interface.ts +0 -31
- package/src/libs/crud-pro/CrudPro.ts +0 -165
- package/src/libs/crud-pro/defaultConfigs.ts +0 -15
- package/src/libs/crud-pro/exceptions.ts +0 -124
- package/src/libs/crud-pro/interfaces.ts +0 -190
- package/src/libs/crud-pro/models/ExecuteContext.ts +0 -120
- package/src/libs/crud-pro/models/ExecuteContextFunc.ts +0 -96
- package/src/libs/crud-pro/models/FuncContext.ts +0 -21
- package/src/libs/crud-pro/models/RequestCfgModel.ts +0 -141
- package/src/libs/crud-pro/models/RequestModel.ts +0 -141
- package/src/libs/crud-pro/models/ResModel.ts +0 -19
- package/src/libs/crud-pro/models/ServiceHub.ts +0 -32
- package/src/libs/crud-pro/models/SqlCfgModel.ts +0 -52
- package/src/libs/crud-pro/models/SqlSegArg.ts +0 -13
- package/src/libs/crud-pro/models/Transaction.ts +0 -73
- package/src/libs/crud-pro/models/TransactionMySQL.ts +0 -79
- package/src/libs/crud-pro/models/TransactionPostgres.ts +0 -91
- package/src/libs/crud-pro/models/TransactionSqlServer.ts +0 -102
- package/src/libs/crud-pro/models/keys.ts +0 -159
- package/src/libs/crud-pro/services/CrudProCachedCfgService.ts +0 -83
- package/src/libs/crud-pro/services/CrudProExecuteFuncService.ts +0 -128
- package/src/libs/crud-pro/services/CrudProExecuteSqlService.ts +0 -262
- package/src/libs/crud-pro/services/CrudProFieldUpdateService.ts +0 -60
- package/src/libs/crud-pro/services/CrudProFieldValidateService.ts +0 -180
- package/src/libs/crud-pro/services/CrudProGenSqlCondition.ts +0 -354
- package/src/libs/crud-pro/services/CrudProGenSqlService.ts +0 -185
- package/src/libs/crud-pro/services/CrudProOriginToExecuteSql.ts +0 -393
- package/src/libs/crud-pro/services/CrudProServiceBase.ts +0 -94
- package/src/libs/crud-pro/services/CrudProTableMetaService.ts +0 -86
- package/src/libs/crud-pro/services/CurdProServiceHub.ts +0 -92
- package/src/libs/crud-pro/sql.txt +0 -120
- package/src/libs/crud-pro/utils/CompareUtils.ts +0 -23
- package/src/libs/crud-pro/utils/DatabaseName.ts +0 -60
- package/src/libs/crud-pro/utils/DateTimeUtils.ts +0 -20
- package/src/libs/crud-pro/utils/MemoryRefreshCache.ts +0 -64
- package/src/libs/crud-pro/utils/MessageParseUtils.ts +0 -33
- package/src/libs/crud-pro/utils/MixinUtils.ts +0 -285
- package/src/libs/crud-pro/utils/ModelUtils.ts +0 -55
- package/src/libs/crud-pro/utils/MultiKeyMap.ts +0 -72
- package/src/libs/crud-pro/utils/SqlFuncUtils.ts +0 -29
- package/src/libs/crud-pro/utils/TypeUtils.ts +0 -188
- package/src/libs/crud-pro/utils/ValidateUtils.ts +0 -165
- package/src/libs/crud-pro/utils/pool/MySQLUtils.ts +0 -20
- package/src/libs/crud-pro/utils/pool/PostgresUtils.ts +0 -22
- package/src/libs/crud-pro/utils/pool/SqlServerUtils.ts +0 -22
- package/src/libs/crud-pro/utils/sqlConvert/convertColumnName.ts +0 -26
- package/src/libs/crud-pro/utils/sqlConvert/convertMix.ts +0 -26
- package/src/libs/crud-pro/utils/sqlConvert/convertMsSql.ts +0 -11
- package/src/libs/crud-pro/utils/sqlConvert/convertPgSql.ts +0 -11
- package/src/libs/crud-pro/utils/sqlConvert/convertPgType.ts +0 -129
- package/src/libs/global-config/global-config.ts +0 -78
- package/src/libs/utils/common-dto.ts +0 -52
- package/src/libs/utils/crypto-utils.ts +0 -50
- package/src/libs/utils/errorToString.ts +0 -61
- package/src/libs/utils/fatcms-request.ts +0 -103
- package/src/libs/utils/functions.ts +0 -73
- package/src/libs/utils/ordernum-utils.ts +0 -14
- package/src/libs/utils/parseConfig.ts +0 -54
- package/src/libs/utils/parseCreateSql.ts +0 -91
- package/src/libs/utils/render-utils.ts +0 -184
- package/src/middleware/forbidden.middleware.ts +0 -52
- package/src/middleware/global.middleware.ts +0 -280
- package/src/middleware/permission.middleware.ts +0 -80
- package/src/middleware/tx.middleware.ts +0 -30
- package/src/models/AsyncTaskModel.ts +0 -82
- package/src/models/RedisKeys.ts +0 -13
- package/src/models/SystemEntities.ts +0 -115
- package/src/models/SystemPerm.ts +0 -105
- package/src/models/SystemTables.ts +0 -27
- package/src/models/bizmodels.ts +0 -120
- package/src/models/contextLogger.ts +0 -132
- package/src/models/devops.ts +0 -17
- package/src/models/userSession.ts +0 -216
- package/src/schedule/anonymousContext.ts +0 -73
- package/src/schedule/index.ts +0 -12
- package/src/schedule/runSchedule.ts +0 -82
- package/src/schedule/scheduleNames.ts +0 -21
- package/src/service/AuthService.ts +0 -272
- package/src/service/EnumInfoService.ts +0 -130
- package/src/service/FileCenterService.ts +0 -395
- package/src/service/SysConfigService.ts +0 -37
- package/src/service/UserAccountService.ts +0 -107
- package/src/service/UserSessionService.ts +0 -78
- package/src/service/VisitStatService.ts +0 -166
- package/src/service/WorkbenchService.ts +0 -165
- package/src/service/anyapi/AnyApiSandboxService.ts +0 -121
- package/src/service/anyapi/AnyApiService.ts +0 -186
- package/src/service/asyncTask/AsyncTaskRunnerService.ts +0 -266
- package/src/service/asyncTask/AsyncTaskService.ts +0 -21
- package/src/service/asyncTask/handler/ExcelInfoModel.ts +0 -11
- package/src/service/asyncTask/handler/ExportExcelAsyncTaskHandler.ts +0 -245
- package/src/service/asyncTask/handler/ExportExcelByInnerHttpHandler.ts +0 -147
- package/src/service/asyncTask/handler/ExportExcelByStdCrudHandler.ts +0 -138
- package/src/service/base/ApiBaseService.ts +0 -42
- package/src/service/base/ApiRateLimiter.ts +0 -59
- package/src/service/base/BaseService.ts +0 -100
- package/src/service/base/RedisCacheService.ts +0 -38
- package/src/service/crudstd/CrudStdActionService.ts +0 -27
- package/src/service/crudstd/CrudStdConstant.ts +0 -62
- package/src/service/crudstd/CrudStdRelationService.ts +0 -78
- package/src/service/crudstd/CrudStdService.ts +0 -283
- package/src/service/curd/CrudProQuick.ts +0 -131
- package/src/service/curd/CurdMixByAccountService.ts +0 -90
- package/src/service/curd/CurdMixByDictService.ts +0 -114
- package/src/service/curd/CurdMixByLinkToCustomService.ts +0 -219
- package/src/service/curd/CurdMixBySysConfigService.ts +0 -78
- package/src/service/curd/CurdMixByWorkbenchService.ts +0 -71
- package/src/service/curd/CurdMixService.ts +0 -97
- package/src/service/curd/CurdMixUtils.ts +0 -311
- package/src/service/curd/CurdProService.ts +0 -229
- package/src/service/curd/fixCfgModel.ts +0 -139
- package/src/service/proxyapi/ProxyApiLoadService.ts +0 -174
- package/src/service/proxyapi/ProxyApiService.ts +0 -262
- package/src/service/proxyapi/ProxyApiUtils.ts +0 -32
- package/src/service/proxyapi/RouteHandler.ts +0 -8
- package/src/service/proxyapi/RouteTrie.ts +0 -74
- package/src/service/proxyapi/WeightedRandom.ts +0 -37
- package/src/service/proxyapi/WeightedRoundRobin.ts +0 -44
- package/src/views/404_app.html +0 -31
- package/src/views/404_workbench.html +0 -34
- package/src/views/static/favicon.ico +0 -0
|
@@ -1,186 +0,0 @@
|
|
|
1
|
-
import { Inject, Provide } from '@midwayjs/core';
|
|
2
|
-
import { Context } from '@midwayjs/koa';
|
|
3
|
-
import { KeysOfSimpleSQL } from '@/libs/crud-pro/models/keys';
|
|
4
|
-
import { SystemTables } from '@/models/SystemTables';
|
|
5
|
-
import { LRUCache } from 'lru-cache';
|
|
6
|
-
import { BizException } from '@/models/devops';
|
|
7
|
-
import { CurdMixService } from '../curd/CurdMixService';
|
|
8
|
-
import { ISysAnyApiEntity } from '@/models/SystemEntities';
|
|
9
|
-
import { AnyApiSandboxService, IRunInSandboxParams } from './AnyApiSandboxService';
|
|
10
|
-
import { parseJsonObject } from '@/libs/utils/functions';
|
|
11
|
-
import * as _ from 'lodash';
|
|
12
|
-
import { MixinUtils } from '@/libs/crud-pro/utils/MixinUtils';
|
|
13
|
-
import { WorkbenchService } from '../WorkbenchService';
|
|
14
|
-
import { validateByCfgString } from '@/libs/crud-pro/utils/ValidateUtils';
|
|
15
|
-
import { API_BASE_TYPE, ApiBaseService } from '../base/ApiBaseService';
|
|
16
|
-
import { GLOBAL_STATIC_CONFIG } from '@/libs/global-config/global-config';
|
|
17
|
-
|
|
18
|
-
const lruCache = new LRUCache<string, any>({
|
|
19
|
-
max: 500,
|
|
20
|
-
ttl: 1000 * 60,
|
|
21
|
-
ttlAutopurge: true,
|
|
22
|
-
});
|
|
23
|
-
|
|
24
|
-
@Provide()
|
|
25
|
-
export class AnyApiService extends ApiBaseService {
|
|
26
|
-
@Inject()
|
|
27
|
-
protected ctx: Context;
|
|
28
|
-
|
|
29
|
-
@Inject()
|
|
30
|
-
private curdMixService: CurdMixService;
|
|
31
|
-
|
|
32
|
-
@Inject()
|
|
33
|
-
private anyApiSandboxService: AnyApiSandboxService;
|
|
34
|
-
|
|
35
|
-
@Inject()
|
|
36
|
-
protected workbenchService: WorkbenchService;
|
|
37
|
-
|
|
38
|
-
async executeAnyApiMethod(methodCode: string, headers: any, body: any, query: any) {
|
|
39
|
-
const anyApi = await this.getAnyApiMethod(methodCode);
|
|
40
|
-
if (!anyApi || anyApi.status !== 1) {
|
|
41
|
-
throw new BizException('接口不存在或已下线:' + methodCode);
|
|
42
|
-
}
|
|
43
|
-
|
|
44
|
-
const isSupport = await this.workbenchService.isSupportCurrentWorkbench(anyApi.workbench_code_array);
|
|
45
|
-
if (!isSupport) {
|
|
46
|
-
throw new BizException('此接口不支持在当前站点打开:' + methodCode);
|
|
47
|
-
}
|
|
48
|
-
|
|
49
|
-
await super.beforeCheckApiAccessibility(API_BASE_TYPE.PROXY_API, anyApi);
|
|
50
|
-
|
|
51
|
-
if (!anyApi.func_code) {
|
|
52
|
-
throw new BizException('没有配置代码:' + methodCode);
|
|
53
|
-
}
|
|
54
|
-
|
|
55
|
-
if (!this.checkFuncCode(anyApi.func_code)) {
|
|
56
|
-
throw new BizException('接口代码校验失败:不满足形式: class HttpController { async handleRequest(params){} }; ');
|
|
57
|
-
}
|
|
58
|
-
|
|
59
|
-
const allParams: IRunInSandboxParams = { headers, body, query };
|
|
60
|
-
allParams.params = this.toAnyApiParams(anyApi, allParams); // 组合参数
|
|
61
|
-
|
|
62
|
-
// 执行代码
|
|
63
|
-
const funcRes = await this.executeFunctionCode(anyApi, allParams);
|
|
64
|
-
return funcRes;
|
|
65
|
-
}
|
|
66
|
-
|
|
67
|
-
private async getAnyApiMethod(methodCode: string): Promise<ISysAnyApiEntity> {
|
|
68
|
-
// 开发环境,不使用缓存
|
|
69
|
-
if (this.isEnableDebug()) {
|
|
70
|
-
return this._getAnyApiMethod(methodCode);
|
|
71
|
-
}
|
|
72
|
-
|
|
73
|
-
// 线上环境可以使用缓存
|
|
74
|
-
let anyApi = lruCache.get(methodCode);
|
|
75
|
-
if (!anyApi) {
|
|
76
|
-
anyApi = await this._getAnyApiMethod(methodCode);
|
|
77
|
-
lruCache.set(methodCode, anyApi);
|
|
78
|
-
}
|
|
79
|
-
return anyApi;
|
|
80
|
-
}
|
|
81
|
-
|
|
82
|
-
private async _getAnyApiMethod(methodCode: string): Promise<ISysAnyApiEntity> {
|
|
83
|
-
const { SystemDbName, SystemDbType } = GLOBAL_STATIC_CONFIG.getConfig();
|
|
84
|
-
|
|
85
|
-
const res = await this.curdMixService.executeCrudByCfg(
|
|
86
|
-
{ condition: { method: methodCode } },
|
|
87
|
-
{
|
|
88
|
-
sqlTable: SystemTables.sys_anyapi,
|
|
89
|
-
method: `get_sys_anyapi_${methodCode}`,
|
|
90
|
-
sqlSimpleName: KeysOfSimpleSQL.SIMPLE_QUERY_ONE,
|
|
91
|
-
sqlDatabase: SystemDbName,
|
|
92
|
-
sqlDbType: SystemDbType,
|
|
93
|
-
updateCfg: {},
|
|
94
|
-
}
|
|
95
|
-
);
|
|
96
|
-
const obj: ISysAnyApiEntity = res.getOneObj();
|
|
97
|
-
if (!obj) {
|
|
98
|
-
return null;
|
|
99
|
-
}
|
|
100
|
-
obj.parsedOthers = parseJsonObject(obj.others);
|
|
101
|
-
return obj;
|
|
102
|
-
}
|
|
103
|
-
|
|
104
|
-
/**
|
|
105
|
-
* 构造参数
|
|
106
|
-
* @param anyApi
|
|
107
|
-
* @param allParams
|
|
108
|
-
* @private
|
|
109
|
-
*/
|
|
110
|
-
private toAnyApiParams(anyApi: ISysAnyApiEntity, allParams: IRunInSandboxParams): any {
|
|
111
|
-
if (!anyApi.req_params) {
|
|
112
|
-
return {};
|
|
113
|
-
}
|
|
114
|
-
|
|
115
|
-
const configs = anyApi.parsedOthers?.req_params || [];
|
|
116
|
-
const params = {};
|
|
117
|
-
for (let i = 0; i < configs.length; i++) {
|
|
118
|
-
const { name, fromType, fromKey, validate_required, validate_cfg_list, valueType, defaultValue } = configs[i];
|
|
119
|
-
const value = _.get(allParams, [fromType, fromKey], defaultValue);
|
|
120
|
-
if (validate_required && MixinUtils.isEmpty(value)) {
|
|
121
|
-
throw new BizException(`参数校验失败:${name} 不能为空`);
|
|
122
|
-
}
|
|
123
|
-
|
|
124
|
-
// 参数校验
|
|
125
|
-
if (Array.isArray(validate_cfg_list)) {
|
|
126
|
-
for (let j = 0; j < validate_cfg_list.length; j++) {
|
|
127
|
-
const validateCfg = validate_cfg_list[j];
|
|
128
|
-
validateByCfgString(validateCfg, value, name);
|
|
129
|
-
}
|
|
130
|
-
}
|
|
131
|
-
|
|
132
|
-
// 格式转换
|
|
133
|
-
params[name] = MixinUtils.parseValueByType(value, valueType);
|
|
134
|
-
}
|
|
135
|
-
return params;
|
|
136
|
-
}
|
|
137
|
-
|
|
138
|
-
/**
|
|
139
|
-
* 运行代码
|
|
140
|
-
* @param anyApiEntity
|
|
141
|
-
* @param allParams
|
|
142
|
-
* @private
|
|
143
|
-
*/
|
|
144
|
-
private async executeFunctionCode(anyApiEntity: ISysAnyApiEntity, allParams: IRunInSandboxParams): Promise<any> {
|
|
145
|
-
return this.anyApiSandboxService.runInSandbox(anyApiEntity, allParams);
|
|
146
|
-
}
|
|
147
|
-
|
|
148
|
-
private checkFuncCode(func_code: string): boolean {
|
|
149
|
-
if (!func_code) {
|
|
150
|
-
return false;
|
|
151
|
-
}
|
|
152
|
-
|
|
153
|
-
const lineHasKey = (lineTrim: string, target: string): boolean => {
|
|
154
|
-
const words = lineTrim.split(/[\s{}();]/);
|
|
155
|
-
for (let i = 0; i < words.length; i++) {
|
|
156
|
-
const word = words[i];
|
|
157
|
-
if (word && word === target) {
|
|
158
|
-
return true;
|
|
159
|
-
}
|
|
160
|
-
}
|
|
161
|
-
return false;
|
|
162
|
-
};
|
|
163
|
-
|
|
164
|
-
let hasClass = false;
|
|
165
|
-
let hasMethod = false;
|
|
166
|
-
const lines = func_code.split('\n');
|
|
167
|
-
for (let i = 0; i < lines.length; i++) {
|
|
168
|
-
const line = lines[i];
|
|
169
|
-
if (line) {
|
|
170
|
-
const lineTrim = line.trim();
|
|
171
|
-
if (lineTrim) {
|
|
172
|
-
if (lineHasKey(lineTrim, 'class') && lineHasKey(lineTrim, 'HttpController')) {
|
|
173
|
-
hasClass = true;
|
|
174
|
-
}
|
|
175
|
-
if (lineHasKey(lineTrim, 'async') && lineHasKey(lineTrim, 'handleRequest')) {
|
|
176
|
-
hasMethod = true;
|
|
177
|
-
}
|
|
178
|
-
if (lineTrim.startsWith('handleRequest')) {
|
|
179
|
-
hasMethod = true;
|
|
180
|
-
}
|
|
181
|
-
}
|
|
182
|
-
}
|
|
183
|
-
}
|
|
184
|
-
return hasMethod && hasClass;
|
|
185
|
-
}
|
|
186
|
-
}
|
|
@@ -1,266 +0,0 @@
|
|
|
1
|
-
import { Inject, Provide } from '@midwayjs/core';
|
|
2
|
-
import { Context, IMidwayKoaContext } from '@midwayjs/koa';
|
|
3
|
-
import { BaseService } from '@/service/base/BaseService';
|
|
4
|
-
import { IScheduleService } from '@/interface';
|
|
5
|
-
import { CurdProService } from '@/service/curd/CurdProService';
|
|
6
|
-
import { ISysAsyncTaskHandler, SysAsyncTaskEntity, SysAsyncTaskPartial, SysAsyncTaskStatus } from '@/models/AsyncTaskModel';
|
|
7
|
-
import { GLOBAL_STATIC_CONFIG } from '@/libs/global-config/global-config';
|
|
8
|
-
import { SystemTables } from '@/models/SystemTables';
|
|
9
|
-
import { KeysOfSimpleSQL } from '@/libs/crud-pro/models/keys';
|
|
10
|
-
import { errorToString } from '@/libs/utils/errorToString';
|
|
11
|
-
import { ANONYMOUS_CONTEXT } from '@/schedule';
|
|
12
|
-
import { RedisKeys } from '@/models/RedisKeys';
|
|
13
|
-
import { getCurrentFullMoment } from '@/libs/utils/functions';
|
|
14
|
-
import { ExportExcelAsyncTaskHandler } from './handler/ExportExcelAsyncTaskHandler';
|
|
15
|
-
|
|
16
|
-
const END_STATUS_LIST = [SysAsyncTaskStatus.PART_SUCCEEDED, SysAsyncTaskStatus.SUCCEEDED, SysAsyncTaskStatus.FAILED, SysAsyncTaskStatus.CANCELLED, SysAsyncTaskStatus.PAUSED];
|
|
17
|
-
|
|
18
|
-
class AsyncTaskRunner {
|
|
19
|
-
isBusy = false;
|
|
20
|
-
taskHandlerMap: Map<string, ISysAsyncTaskHandler> = new Map();
|
|
21
|
-
|
|
22
|
-
async executeTaskList(taskList: SysAsyncTaskEntity[]) {
|
|
23
|
-
if (!taskList || taskList.length === 0) {
|
|
24
|
-
return;
|
|
25
|
-
}
|
|
26
|
-
|
|
27
|
-
this.isBusy = true;
|
|
28
|
-
|
|
29
|
-
for (let i = 0; i < taskList.length; i++) {
|
|
30
|
-
const taskElement = taskList[i];
|
|
31
|
-
taskElement.task_status = SysAsyncTaskStatus.RUNNING;
|
|
32
|
-
|
|
33
|
-
try {
|
|
34
|
-
// 1. 更新当前这个任务的状态
|
|
35
|
-
await this.updateTaskStatus(taskElement, {
|
|
36
|
-
task_status: taskElement.task_status,
|
|
37
|
-
started_at: getCurrentFullMoment(),
|
|
38
|
-
updated_at: getCurrentFullMoment(),
|
|
39
|
-
});
|
|
40
|
-
|
|
41
|
-
// 任务状态的设置在自定义的执行器里面。
|
|
42
|
-
await this.executeTask(taskElement);
|
|
43
|
-
|
|
44
|
-
// 当前不是终止状态,并且没有异常就设置成成功。
|
|
45
|
-
if (!END_STATUS_LIST.includes(taskElement.task_status)) {
|
|
46
|
-
taskElement.task_status = SysAsyncTaskStatus.SUCCEEDED;
|
|
47
|
-
}
|
|
48
|
-
} catch (error) {
|
|
49
|
-
taskElement.task_status = SysAsyncTaskStatus.FAILED;
|
|
50
|
-
taskElement.error_message = errorToString(error);
|
|
51
|
-
}
|
|
52
|
-
|
|
53
|
-
try {
|
|
54
|
-
// 2. 更新当前这个任务的状态
|
|
55
|
-
await this.updateTaskStatus(taskElement, {
|
|
56
|
-
task_status: taskElement.task_status,
|
|
57
|
-
error_message: taskElement.error_message,
|
|
58
|
-
updated_at: getCurrentFullMoment(),
|
|
59
|
-
completed_at: getCurrentFullMoment(),
|
|
60
|
-
});
|
|
61
|
-
} catch (error) {
|
|
62
|
-
ANONYMOUS_CONTEXT.getApp().getCoreLogger().error('[AsyncTaskRunner] executeTaskList error', error);
|
|
63
|
-
}
|
|
64
|
-
}
|
|
65
|
-
|
|
66
|
-
this.isBusy = false;
|
|
67
|
-
}
|
|
68
|
-
|
|
69
|
-
private async executeTask(taskElement: SysAsyncTaskEntity) {
|
|
70
|
-
const taskType = taskElement.task_type;
|
|
71
|
-
const taskHandler = this.taskHandlerMap.get(taskType);
|
|
72
|
-
if (!taskHandler) {
|
|
73
|
-
throw new Error('TaskHandler not found , taskType = ' + taskType);
|
|
74
|
-
}
|
|
75
|
-
const updateTaskStatus = (updatePartials: SysAsyncTaskPartial) => {
|
|
76
|
-
return this.updateTaskStatus(taskElement, updatePartials);
|
|
77
|
-
};
|
|
78
|
-
await taskHandler.execute({ task: taskElement, updateTaskStatus });
|
|
79
|
-
}
|
|
80
|
-
|
|
81
|
-
/**
|
|
82
|
-
* 更新任务状态或任务进度
|
|
83
|
-
* @param taskElement
|
|
84
|
-
* @param updatePartials
|
|
85
|
-
* @private
|
|
86
|
-
*/
|
|
87
|
-
private async updateTaskStatus(taskElement: SysAsyncTaskEntity, updatePartials: SysAsyncTaskPartial) {
|
|
88
|
-
if (!updatePartials) {
|
|
89
|
-
throw new Error('updatePartials not found');
|
|
90
|
-
}
|
|
91
|
-
return await ANONYMOUS_CONTEXT.runServiceAtAnonymousContext(async (ctx: IMidwayKoaContext) => {
|
|
92
|
-
const curdProService: CurdProService = await ctx.requestContext.getAsync('curdProService');
|
|
93
|
-
const { SystemDbName, SystemDbType } = GLOBAL_STATIC_CONFIG.getConfig();
|
|
94
|
-
|
|
95
|
-
const res = await curdProService.executeCrudByCfg(
|
|
96
|
-
{
|
|
97
|
-
condition: {
|
|
98
|
-
id: taskElement.id,
|
|
99
|
-
},
|
|
100
|
-
data: updatePartials,
|
|
101
|
-
},
|
|
102
|
-
{
|
|
103
|
-
sqlTable: SystemTables.sys_async_tasks,
|
|
104
|
-
sqlSimpleName: KeysOfSimpleSQL.SIMPLE_UPDATE,
|
|
105
|
-
sqlDatabase: SystemDbName,
|
|
106
|
-
sqlDbType: SystemDbType,
|
|
107
|
-
}
|
|
108
|
-
);
|
|
109
|
-
return res.getResModel().affected;
|
|
110
|
-
});
|
|
111
|
-
}
|
|
112
|
-
|
|
113
|
-
/**
|
|
114
|
-
* 获取当前进程可以处理的任务类型。
|
|
115
|
-
*/
|
|
116
|
-
public getHandlerTaskTypeList(): string[] {
|
|
117
|
-
const keys = this.taskHandlerMap.keys();
|
|
118
|
-
return [...keys];
|
|
119
|
-
}
|
|
120
|
-
}
|
|
121
|
-
|
|
122
|
-
/**
|
|
123
|
-
* 业务可以扩展
|
|
124
|
-
*/
|
|
125
|
-
export const ASYNC_TASK_RUNNER = new AsyncTaskRunner();
|
|
126
|
-
|
|
127
|
-
//内置的异步任务处理器:EXCEL导出
|
|
128
|
-
ASYNC_TASK_RUNNER.taskHandlerMap.set('EXCEL_EXPORT', new ExportExcelAsyncTaskHandler());
|
|
129
|
-
|
|
130
|
-
/**
|
|
131
|
-
* Redis锁
|
|
132
|
-
*/
|
|
133
|
-
const ASYNC_TASK_LOCK = RedisKeys.ASYNC_TASK_LOCK;
|
|
134
|
-
|
|
135
|
-
/**
|
|
136
|
-
* 上次检查异步任务的时间点
|
|
137
|
-
*/
|
|
138
|
-
const ASYNC_TASK_RUNTIME_OBJ = {
|
|
139
|
-
LAST_CHECK_ASYNC_TASK_UPDATE_TIME: 0,
|
|
140
|
-
};
|
|
141
|
-
|
|
142
|
-
@Provide()
|
|
143
|
-
export class AsyncTaskRunnerService extends BaseService implements IScheduleService {
|
|
144
|
-
@Inject()
|
|
145
|
-
protected ctx: Context;
|
|
146
|
-
|
|
147
|
-
@Inject()
|
|
148
|
-
private curdProService: CurdProService;
|
|
149
|
-
|
|
150
|
-
async fetchPendingTasks(): Promise<void> {
|
|
151
|
-
const { SystemDbName, SystemDbType } = GLOBAL_STATIC_CONFIG.getConfig();
|
|
152
|
-
|
|
153
|
-
// 只获取本进程能够处理的任务类型。
|
|
154
|
-
const taskTypeList = ASYNC_TASK_RUNNER.getHandlerTaskTypeList();
|
|
155
|
-
if (taskTypeList.length === 0) {
|
|
156
|
-
return Promise.resolve();
|
|
157
|
-
}
|
|
158
|
-
|
|
159
|
-
// 查询等待处理的任务。
|
|
160
|
-
const queryRes = await this.curdProService.executeCrudByCfg(
|
|
161
|
-
{
|
|
162
|
-
condition: {
|
|
163
|
-
task_status: SysAsyncTaskStatus.PENDING,
|
|
164
|
-
task_type: {
|
|
165
|
-
$in: taskTypeList,
|
|
166
|
-
},
|
|
167
|
-
},
|
|
168
|
-
orderBy: 'id+',
|
|
169
|
-
limit: 10,
|
|
170
|
-
},
|
|
171
|
-
{
|
|
172
|
-
sqlTable: SystemTables.sys_async_tasks,
|
|
173
|
-
sqlSimpleName: KeysOfSimpleSQL.SIMPLE_QUERY_PAGE,
|
|
174
|
-
sqlDatabase: SystemDbName,
|
|
175
|
-
sqlDbType: SystemDbType,
|
|
176
|
-
}
|
|
177
|
-
);
|
|
178
|
-
|
|
179
|
-
const taskList = queryRes.getResRows();
|
|
180
|
-
|
|
181
|
-
if (taskList.length === 0) {
|
|
182
|
-
return Promise.resolve();
|
|
183
|
-
}
|
|
184
|
-
const taskIds = taskList.map(elem => elem.id).filter(Boolean);
|
|
185
|
-
|
|
186
|
-
// 将状态更新为处理中。防止其它进程重复处理。
|
|
187
|
-
await this.curdProService.executeCrudByCfg(
|
|
188
|
-
{
|
|
189
|
-
condition: {
|
|
190
|
-
id: {
|
|
191
|
-
$in: taskIds,
|
|
192
|
-
},
|
|
193
|
-
},
|
|
194
|
-
data: {
|
|
195
|
-
task_status: SysAsyncTaskStatus.RUNNING,
|
|
196
|
-
},
|
|
197
|
-
},
|
|
198
|
-
{
|
|
199
|
-
sqlTable: SystemTables.sys_async_tasks,
|
|
200
|
-
sqlSimpleName: KeysOfSimpleSQL.SIMPLE_UPDATE,
|
|
201
|
-
sqlDatabase: SystemDbName,
|
|
202
|
-
sqlDbType: SystemDbType,
|
|
203
|
-
}
|
|
204
|
-
);
|
|
205
|
-
|
|
206
|
-
// 开始执行。
|
|
207
|
-
ASYNC_TASK_RUNNER.executeTaskList(taskList).then(() => {
|
|
208
|
-
console.log('[AsyncTaskRunnerService] ASYNC_TASK_RUNNER finished taskIds ==> ' + JSON.stringify(taskIds));
|
|
209
|
-
});
|
|
210
|
-
}
|
|
211
|
-
|
|
212
|
-
async runBySchedule() {
|
|
213
|
-
// 1. 当前很忙
|
|
214
|
-
if (ASYNC_TASK_RUNNER.isBusy) {
|
|
215
|
-
return Promise.resolve();
|
|
216
|
-
}
|
|
217
|
-
|
|
218
|
-
// 2. 发现了新任务
|
|
219
|
-
const isExistNewTask = await this.isExistNewTask();
|
|
220
|
-
if (!isExistNewTask) {
|
|
221
|
-
return Promise.resolve();
|
|
222
|
-
}
|
|
223
|
-
|
|
224
|
-
// 这里的过期时间1分钟即可。fetchPendingTasks函数不可能超过一分钟。
|
|
225
|
-
// 因为这里只是从数据库中获取一批任务,放到自己的任务队列里。还没触发执行。
|
|
226
|
-
const lock = await this.redisService.set(ASYNC_TASK_LOCK, 1, 'EX', 60, 'NX');
|
|
227
|
-
if (lock !== 'OK') {
|
|
228
|
-
return Promise.resolve();
|
|
229
|
-
}
|
|
230
|
-
|
|
231
|
-
try {
|
|
232
|
-
await this.fetchPendingTasks();
|
|
233
|
-
} catch (e) {
|
|
234
|
-
console.error('[AsyncTaskRunnerService] fetchPendingTasks error', errorToString(e));
|
|
235
|
-
}
|
|
236
|
-
|
|
237
|
-
await this.redisService.del(ASYNC_TASK_LOCK);
|
|
238
|
-
|
|
239
|
-
return Promise.resolve();
|
|
240
|
-
}
|
|
241
|
-
|
|
242
|
-
/**
|
|
243
|
-
* 是否存在新任务
|
|
244
|
-
* @private
|
|
245
|
-
*/
|
|
246
|
-
private async isExistNewTask(): Promise<boolean> {
|
|
247
|
-
// 刚启动,没有检查过。
|
|
248
|
-
if (!ASYNC_TASK_RUNTIME_OBJ.LAST_CHECK_ASYNC_TASK_UPDATE_TIME) {
|
|
249
|
-
ASYNC_TASK_RUNTIME_OBJ.LAST_CHECK_ASYNC_TASK_UPDATE_TIME = Date.now();
|
|
250
|
-
return true;
|
|
251
|
-
}
|
|
252
|
-
|
|
253
|
-
//Redis没有任何内容
|
|
254
|
-
const updateTime = await this.redisService.get(RedisKeys.ASYNC_TASK_UPDATE_TIME);
|
|
255
|
-
if (!updateTime) {
|
|
256
|
-
ASYNC_TASK_RUNTIME_OBJ.LAST_CHECK_ASYNC_TASK_UPDATE_TIME = Date.now();
|
|
257
|
-
return false;
|
|
258
|
-
}
|
|
259
|
-
|
|
260
|
-
const updateTimeNumber = Number.parseInt(updateTime, 10);
|
|
261
|
-
|
|
262
|
-
const isExistNewTask: boolean = updateTimeNumber > ASYNC_TASK_RUNTIME_OBJ.LAST_CHECK_ASYNC_TASK_UPDATE_TIME;
|
|
263
|
-
ASYNC_TASK_RUNTIME_OBJ.LAST_CHECK_ASYNC_TASK_UPDATE_TIME = updateTimeNumber;
|
|
264
|
-
return isExistNewTask;
|
|
265
|
-
}
|
|
266
|
-
}
|
|
@@ -1,21 +0,0 @@
|
|
|
1
|
-
import { Inject, Provide } from '@midwayjs/core';
|
|
2
|
-
import { Context } from '@midwayjs/koa';
|
|
3
|
-
import { BaseService } from '@/service/base/BaseService';
|
|
4
|
-
import { runScheduleTaskOnce } from '@/schedule';
|
|
5
|
-
import { INNER_SCHEDULE_NAMES } from '@/schedule/scheduleNames';
|
|
6
|
-
import { RedisKeys } from '@/models/RedisKeys';
|
|
7
|
-
|
|
8
|
-
@Provide()
|
|
9
|
-
export class AsyncTaskService extends BaseService {
|
|
10
|
-
@Inject()
|
|
11
|
-
protected ctx: Context;
|
|
12
|
-
|
|
13
|
-
async startTask() {
|
|
14
|
-
await this.redisService.set(RedisKeys.ASYNC_TASK_UPDATE_TIME, `${Date.now()}`);
|
|
15
|
-
runScheduleTaskOnce(INNER_SCHEDULE_NAMES.asyncTaskRunnerService).then(schedule => {
|
|
16
|
-
console.log(schedule);
|
|
17
|
-
});
|
|
18
|
-
}
|
|
19
|
-
|
|
20
|
-
async cancelTask(id: number) {}
|
|
21
|
-
}
|
|
@@ -1,11 +0,0 @@
|
|
|
1
|
-
export interface IExcelHeaderInfo {
|
|
2
|
-
title: string;
|
|
3
|
-
dataIndex: string;
|
|
4
|
-
}
|
|
5
|
-
|
|
6
|
-
export interface IExcelAsyncTaskHandler {
|
|
7
|
-
getExcelHeader(): Promise<IExcelHeaderInfo[]>;
|
|
8
|
-
getPageSize(): Promise<number>;
|
|
9
|
-
getTotalCount(): Promise<number>;
|
|
10
|
-
getExcelDataList(pageNo: number, pageSize: number): Promise<any[]>;
|
|
11
|
-
}
|