1 | ;
|
2 | var __assign = (this && this.__assign) || Object.assign || function(t) {
|
3 | for (var s, i = 1, n = arguments.length; i < n; i++) {
|
4 | s = arguments[i];
|
5 | for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p))
|
6 | t[p] = s[p];
|
7 | }
|
8 | return t;
|
9 | };
|
10 | var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
|
11 | return new (P || (P = Promise))(function (resolve, reject) {
|
12 | function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
|
13 | function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
|
14 | function step(result) { result.done ? resolve(result.value) : new P(function (resolve) { resolve(result.value); }).then(fulfilled, rejected); }
|
15 | step((generator = generator.apply(thisArg, _arguments || [])).next());
|
16 | });
|
17 | };
|
18 | var __generator = (this && this.__generator) || function (thisArg, body) {
|
19 | var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g;
|
20 | return g = { next: verb(0), "throw": verb(1), "return": verb(2) }, typeof Symbol === "function" && (g[Symbol.iterator] = function() { return this; }), g;
|
21 | function verb(n) { return function (v) { return step([n, v]); }; }
|
22 | function step(op) {
|
23 | if (f) throw new TypeError("Generator is already executing.");
|
24 | while (_) try {
|
25 | if (f = 1, y && (t = op[0] & 2 ? y["return"] : op[0] ? y["throw"] || ((t = y["return"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t;
|
26 | if (y = 0, t) op = [op[0] & 2, t.value];
|
27 | switch (op[0]) {
|
28 | case 0: case 1: t = op; break;
|
29 | case 4: _.label++; return { value: op[1], done: false };
|
30 | case 5: _.label++; y = op[1]; op = [0]; continue;
|
31 | case 7: op = _.ops.pop(); _.trys.pop(); continue;
|
32 | default:
|
33 | if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; }
|
34 | if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; }
|
35 | if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; }
|
36 | if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; }
|
37 | if (t[2]) _.ops.pop();
|
38 | _.trys.pop(); continue;
|
39 | }
|
40 | op = body.call(thisArg, _);
|
41 | } catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; }
|
42 | if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true };
|
43 | }
|
44 | };
|
45 | var _this = this;
|
46 | Object.defineProperty(exports, "__esModule", { value: true });
|
47 | process.env.NODE_TLS_REJECT_UNAUTHORIZED = '0';
|
48 | require("reflect-metadata");
|
49 | var inversify_1 = require("inversify");
|
50 | var typeorm_1 = require("@bxjs/typeorm");
|
51 | var session_1 = require("./session");
|
52 | var $$ = require("./plugins");
|
53 | var path = require('path');
|
54 | var ErrorStackParser = require('error-stack-parser');
|
55 | var cookie = require('cookie');
|
56 | var MobileDetect = require('mobile-detect');
|
57 | var fetch = require('node-fetch');
|
58 | var _ = require('lodash');
|
59 | var moment = require('moment');
|
60 | var extend = require('extend');
|
61 | var querystring = require('querystring');
|
62 | // const parameter = require('parameter')
|
63 | // const parameterCheckInstance = new parameter({
|
64 | // // translate: function () {
|
65 | // // var args = Array.prototype.slice.call(arguments);
|
66 | // // // Assume there have I18n.t method for convert language.
|
67 | // // return I18n.t.apply(I18n, args);
|
68 | // // }
|
69 | // })
|
70 | var circular_json = require("circular-json");
|
71 | var mockjs = require('mockjs');
|
72 | var shortid = require('shortid');
|
73 | var validatorjs = require('validatorjs');
|
74 | var cross_spawn = require('cross-spawn');
|
75 | var ACMClient = require('acm-client');
|
76 | var co = require('co');
|
77 | // FIXME HACK原生方法JSON转换不可逆的BUG(JAVA端传来的富文本字段内容含有\n\t字符串中的字符生成JSON字符串无法正常解析报错)
|
78 | var raw_stringify = JSON.stringify;
|
79 | function new_stringify(value, replacer, space) {
|
80 | var out = raw_stringify(value, replacer, space);
|
81 | if (_.isString(out)) {
|
82 | out = out.replace(/\\n/g, '\\\\n')
|
83 | .replace(/\\t/g, '\\\\t')
|
84 | .replace(/\\u/g, '\\\\u'); //JAVA端返回的unicode字符转义处理
|
85 | }
|
86 | return out;
|
87 | }
|
88 | JSON.stringify = new_stringify;
|
89 | // ts-node本地调试需要加载对应的源代码后缀名称
|
90 | function get_suffix_ts_or_js() {
|
91 | if (global['__env__'] == 'local' && !/^\/code\/node_modules/.test(__dirname)) {
|
92 | return 'ts';
|
93 | }
|
94 | else {
|
95 | return 'js';
|
96 | }
|
97 | }
|
98 | exports.get_suffix_ts_or_js = get_suffix_ts_or_js;
|
99 | // 准确定位错误码位置,间接得到函数调用位置地址信息,结合符号报表的正确解析处理完美得到错误定位信息,准确代码调试。
|
100 | function __get_base_func_caller_source_position(position) {
|
101 | if (position === void 0) { position = 3; }
|
102 | try {
|
103 | throw new Error();
|
104 | }
|
105 | catch (err) {
|
106 | var out = ErrorStackParser.parse(err);
|
107 | var idx = 0;
|
108 | // 找到第二个TS文件的执行位置
|
109 | var find_ts_sufix_file_count = 0;
|
110 | for (; idx < out.length; idx++) {
|
111 | if (/\.ts$/.test(out[idx].fileName)) {
|
112 | find_ts_sufix_file_count += 1;
|
113 | }
|
114 | if (find_ts_sufix_file_count == position) {
|
115 | break;
|
116 | }
|
117 | }
|
118 | if (find_ts_sufix_file_count == position) {
|
119 | return '[' + out[idx]['fileName'] + ':' + out[idx]['lineNumber'] + ']';
|
120 | }
|
121 | else {
|
122 | // TODO 需要定位为什么调用栈无法找到对应的位置出现越界??
|
123 | // console.error(err)
|
124 | console.error(circular_json.stringify(out, null, 4)
|
125 | .replace(/\r/g, '').replace(/\n/g, ''));
|
126 | return '#';
|
127 | }
|
128 | }
|
129 | }
|
130 | // 获取异常调用栈用于辅助错误提示定位
|
131 | function xstack(err, compact) {
|
132 | if (compact === void 0) { compact = true; }
|
133 | try {
|
134 | // TODO 优化裁剪一些无用信息减少日志尺寸更加便于人工分析处理
|
135 | var stack = ErrorStackParser.parse(err);
|
136 | if (compact) {
|
137 | var sources = [];
|
138 | for (var _i = 0, stack_1 = stack; _i < stack_1.length; _i++) {
|
139 | var v = stack_1[_i];
|
140 | sources.push(v['fileName'] + ":" + v['lineNumber']);
|
141 | }
|
142 | return sources;
|
143 | }
|
144 | return stack;
|
145 | }
|
146 | catch (err1) {
|
147 | var source = __get_base_func_caller_source_position();
|
148 | return "invalid error input param (" + source + ")";
|
149 | }
|
150 | }
|
151 | exports.xstack = xstack;
|
152 | // // 错误栈的递归嵌套格式显示数据结构定义(param嵌套找到最后一个msg的JSON解析语法错误就是错误链的原始错误发生位置)
|
153 | // let x = {
|
154 | // "code": "UNKNOWN",
|
155 | // "msg": "未知错误",
|
156 | // "param": {
|
157 | // "msg": "您输入的用户名或密码错误,请重新登录 (ErrorCode: 1005, url: https://login.alibaba-inc.com/authorize/login.do)"
|
158 | // },
|
159 | // "stack": "[\"/Users/chujinghui/Desktop/work/xjs/bxjs/framework/base.ts:110\",\"/Users/chujinghui/Desktop/work/xjs/bxjs/app/entries/web/mobile/meeting-room-visit.ts:161\",\"/Users/chujinghui/Desktop/work/xjs/bxjs/app/entries/web/mobile/meeting-room-visit.js:40\",\"/Users/chujinghui/Desktop/work/xjs/bxjs/app/entries/web/mobile/meeting-room-visit.js:21\",\"/Users/chujinghui/Desktop/work/xjs/bxjs/app/entries/web/mobile/meeting-room-visit.js:13\",\"internal/process/next_tick.js:188\"]",
|
160 | // }
|
161 | // 对于异常内容的格式化参数解析处理成为四元组code/msg/param/stack
|
162 | function xerror(err, __param) {
|
163 | xassert(err instanceof Error);
|
164 | try {
|
165 | // 标准错误的统一转换处理
|
166 | var data_1 = JSON.parse(err.message);
|
167 | if (data_1.code && data_1.msg && ERRORS[data_1.code]) {
|
168 | return data_1;
|
169 | }
|
170 | }
|
171 | catch (err) {
|
172 | // ignore parse error
|
173 | }
|
174 | // 非标准错误的统一格式转换处理
|
175 | var msg = ERRORS[ERR$UNKNOWN]['zh']; // TODO 错误码多语言回传到客户端问题
|
176 | var code = ERR$UNKNOWN;
|
177 | var param = { msg: err.message, param: __param }; // 用户自定义的错误参数信息 msg为非错误码JSON四元组就是嵌套的终止条件。
|
178 | var stack = xstack(err);
|
179 | var data = { msg: msg, code: code, param: param, stack: stack };
|
180 | return data;
|
181 | }
|
182 | exports.xerror = xerror;
|
183 | // 用于获取错误栈的root cause根本原因(第一个被拦截的错误发生位置)
|
184 | function xroot(err) {
|
185 | xassert(err instanceof Error);
|
186 | var _a = xerror(err), msg = _a.msg, param = _a.param, code = _a.code, stack = _a.stack;
|
187 | // 递归遍历找到错误链的root cause
|
188 | for (; param && param.msg;) {
|
189 | try {
|
190 | var json = JSON.parse(param.msg);
|
191 | param = json.param;
|
192 | }
|
193 | catch (err) {
|
194 | msg = param.msg;
|
195 | code = param.code;
|
196 | stack = param.stack;
|
197 | param = param.param;
|
198 | break;
|
199 | }
|
200 | }
|
201 | return { msg: msg, code: code, param: param, stack: stack };
|
202 | }
|
203 | exports.xroot = xroot;
|
204 | // TODO 报错处理(显示问题反馈联系人信息)
|
205 | // 将未处理的错误上抛的异常链记录下来用于精准追踪代码的执行过程(以及准确获取到根节点的错误码)
|
206 | // 对于promise异步回调的统一出错处理写法实例
|
207 | // export function login(username: string, password: string) {
|
208 | // return new Promise((resolve, reject) => {
|
209 | // co(function* () {
|
210 | // let user = yield buc.oauthclient.login(username, password)
|
211 | // resolve(user)
|
212 | // }).catch(async function (err) {
|
213 | // xthrow(err, reject)
|
214 | // })
|
215 | // })
|
216 | // }
|
217 | function xthrow(code, param, reject_param) {
|
218 | if (code === void 0) { code = ERR$UNKNOWN; }
|
219 | if (param === void 0) { param = undefined; }
|
220 | if (reject_param === void 0) { reject_param = undefined; }
|
221 | // promise中进行reject异常处理的抛出错误方法的形参逻辑预处理转换。
|
222 | var reject = _.isFunction(param) ? param : undefined;
|
223 | if (reject)
|
224 | param = reject_param;
|
225 | var data = {};
|
226 | var source = __get_base_func_caller_source_position();
|
227 | if (code instanceof Error) {
|
228 | try {
|
229 | data = JSON.parse(code.message);
|
230 | // 将透传上抛的错误的路径信息和附加参数也记录下来方便提供完整应用堆栈信息辅助调试业务逻辑
|
231 | if (!_.isArray(data.stack)) {
|
232 | data.stack = [];
|
233 | }
|
234 | data.stack.push(source);
|
235 | }
|
236 | catch (err) {
|
237 | // ignore
|
238 | }
|
239 | // 标准错误直接上抛处理
|
240 | if (data.code && data.msg && ERRORS[data.code]) {
|
241 | // 测试严重BUG reject函数类型表达式为假必须要用lodash判定是否为函数
|
242 | if (_.isFunction(reject)) {
|
243 | // promise回调中进行抛错误处理
|
244 | var err = new Error(JSON.stringify(data));
|
245 | reject(err);
|
246 | return;
|
247 | }
|
248 | else {
|
249 | throw new Error(JSON.stringify(data));
|
250 | }
|
251 | }
|
252 | // 将非标准错误转换为标准错误后再上抛处理
|
253 | data = xerror(code, param);
|
254 | data.code = ERR$UNKNOWN;
|
255 | data.msg = ERRORS[ERR$UNKNOWN]['zh']; // FIXME TODO 错误码的多语言处理转换!!
|
256 | data.param = { msg: code.message, param: param, stack: [source] };
|
257 | }
|
258 | else {
|
259 | // 对于常量定义错误的统一格式化处理
|
260 | data = { code: code, msg: global['ERRORS'][code]['zh'], param: param, stack: [source] };
|
261 | }
|
262 | // 对于是否promise场景下的错误上抛进行正确的转换处理
|
263 | if (_.isFunction(reject)) {
|
264 | // promise回调中进行抛错误处理
|
265 | reject(new Error(JSON.stringify(data)));
|
266 | }
|
267 | else {
|
268 | // 非promise回调中异常传递
|
269 | throw new Error(JSON.stringify(data));
|
270 | }
|
271 | }
|
272 | exports.xthrow = xthrow;
|
273 | function xassert(expr, code, param) {
|
274 | if (code === void 0) { code = ERR$ASSERT; }
|
275 | var source = __get_base_func_caller_source_position();
|
276 | var stack = [source];
|
277 | if (!expr)
|
278 | throw new Error(JSON.stringify({ code: code, msg: global['ERRORS'][code]['zh'], param: param, stack: stack }));
|
279 | return expr;
|
280 | }
|
281 | exports.xassert = xassert;
|
282 | // // https://github.com/node-modules/parameter 参数验证规则详见此文档(egg团队开发的组件)
|
283 | // // 注意事项:GET通过URL传递的参数都是字符串类型应该尽量避免GET传递参数,需要多用POST的JSON格式传递参数并且POSTMAN上进行辅助测试正确数据类型映射。
|
284 | // export function xcheck(param: { [propName: string]: any }, rules: { [propName: string]: any }) {
|
285 | // let errors = parameterCheckInstance.validate(rules, param)
|
286 | // if (_.isEmpty(errors)) {
|
287 | // return true
|
288 | // } else {
|
289 | // xthrow(ERR$PARAM, errors)
|
290 | // }
|
291 | // }
|
292 | function xlog() {
|
293 | var args = [];
|
294 | for (var _i = 0; _i < arguments.length; _i++) {
|
295 | args[_i] = arguments[_i];
|
296 | }
|
297 | // 兼容云端以及本地日志调试(解决任意对象的JSON字符串内容的完整输出)
|
298 | var source = __get_base_func_caller_source_position();
|
299 | var output = circular_json.stringify(args.slice(), null, 4);
|
300 | if (global['__env__'] != 'prod' && !/^\/code\/node_modules/.test(__dirname)) {
|
301 | // 打印到控制台一份日志(在阿里云非线上FC环境中)
|
302 | console.log.apply(undefined, [source + output]);
|
303 | // 写日志文件到/tmp下临时处理一下 TODO 需要改为类似log4j的本地日志库仅在非线上环境使用方便开发单机日常机器上调试。
|
304 | var fs = require('fs');
|
305 | var logFilePath = process.env['NODE_LOGFILE'] ? process.env['NODE_LOGFILE'] : '/tmp/bxjs.log';
|
306 | fs.appendFileSync(logFilePath, source + output + "\r\n");
|
307 | }
|
308 | else {
|
309 | // 生产环境下只打印到控制台绑定的SLS日志服务器上,并且需要去除掉换行信息否则打印会不正常。
|
310 | output = output.replace(/\r/g, '').replace(/\n/g, '');
|
311 | console.log.apply(undefined, [source + output]);
|
312 | }
|
313 | }
|
314 | exports.xlog = xlog;
|
315 | // // 将详细错误信息及时发送到钉钉群上实时反馈给维护者
|
316 | // await xwarn({
|
317 | // code,
|
318 | // // TODO 如何认证通过了获取到用户信息也需要发送过去,方便联系对接人员进行立刻问题处理反馈。
|
319 | // message,
|
320 | // stack,
|
321 | // param,
|
322 | // })
|
323 | // 将详细错误信息及时发送到钉钉群上实时反馈给维护者
|
324 | // 钉钉IM群机器人报警通知
|
325 | function xwarn() {
|
326 | var args = [];
|
327 | for (var _i = 0; _i < arguments.length; _i++) {
|
328 | args[_i] = arguments[_i];
|
329 | }
|
330 | return __awaiter(this, void 0, void 0, function () {
|
331 | var source, out, access_token, mobiles;
|
332 | return __generator(this, function (_a) {
|
333 | switch (_a.label) {
|
334 | case 0:
|
335 | source = __get_base_func_caller_source_position();
|
336 | // 对于异常参数警告信息进行错误内容标准解析
|
337 | if (args.length > 0 && args[0] instanceof Error) {
|
338 | args[0] = xerror(args[0]);
|
339 | }
|
340 | out = [source, moment().format('YYYY-MM-DD HH:mm:ss'), __assign({}, args)];
|
341 | access_token = xconfig('framework.warn.dingding.access_token');
|
342 | mobiles = xconfig('framework.warn.dingding.mobiles');
|
343 | if (!access_token || !mobiles) {
|
344 | access_token = '020a09eac5f2fa320ae851442d5e19e23693c64ad2255c85354b4a49a5a48d35';
|
345 | mobiles = ['15381151346'];
|
346 | }
|
347 | return [4 /*yield*/, xpost("https://oapi.dingtalk.com/robot/send?access_token=" + access_token, {
|
348 | msgtype: 'text',
|
349 | text: {
|
350 | content: out
|
351 | },
|
352 | at: {
|
353 | atMobiles: mobiles,
|
354 | isAtAll: false
|
355 | }
|
356 | })
|
357 | // 线上SLS日志上也保存一份
|
358 | // console.warn(out)
|
359 | ];
|
360 | case 1:
|
361 | _a.sent();
|
362 | // 线上SLS日志上也保存一份
|
363 | // console.warn(out)
|
364 | xlog(out);
|
365 | return [2 /*return*/];
|
366 | }
|
367 | });
|
368 | });
|
369 | }
|
370 | // 捕获未监听到的异常记录后直接退出(运行堆栈已经破坏直接记录日志后异常退出即可,由外部监控自动重启)
|
371 | process.on('uncaughtException', function (err) {
|
372 | return __awaiter(this, void 0, void 0, function () {
|
373 | return __generator(this, function (_a) {
|
374 | switch (_a.label) {
|
375 | case 0:
|
376 | xlog(xerror(err));
|
377 | return [4 /*yield*/, xwarn(err)];
|
378 | case 1:
|
379 | _a.sent();
|
380 | process.exit(-1);
|
381 | return [2 /*return*/];
|
382 | }
|
383 | });
|
384 | });
|
385 | });
|
386 | // 记录await/async中出现未捕获的异常错误
|
387 | process.on('unhandledRejection', function (reason, p) { return __awaiter(_this, void 0, void 0, function () {
|
388 | return __generator(this, function (_a) {
|
389 | switch (_a.label) {
|
390 | case 0:
|
391 | xlog('Unhandled Rejection at: Promise', p, 'reason:', reason);
|
392 | // application specific logging, throwing an error, or other logic here
|
393 | return [4 /*yield*/, xwarn(reason, p)];
|
394 | case 1:
|
395 | // application specific logging, throwing an error, or other logic here
|
396 | _a.sent();
|
397 | process.exit(-1);
|
398 | return [2 /*return*/];
|
399 | }
|
400 | });
|
401 | }); });
|
402 | // async/await的非阻塞异步延迟方法,用于调试阻塞程序的执行进行单步调试的效果。
|
403 | var sleep = require('sleep-async')();
|
404 | function xsleep(ms) {
|
405 | if (ms === void 0) { ms = -1; }
|
406 | if (ms <= 0) {
|
407 | ms = 50 * 365 * 24 * 3600 * 1000; // 50年最大数视为永久阻塞方便断点单步调试问题
|
408 | }
|
409 | return new Promise(function (resolve, reject) {
|
410 | try {
|
411 | sleep.sleep(ms, function () {
|
412 | resolve();
|
413 | });
|
414 | }
|
415 | catch (err) {
|
416 | xlog(xerror(err));
|
417 | resolve();
|
418 | // xthrow(err,reject)
|
419 | }
|
420 | });
|
421 | }
|
422 | exports.xsleep = xsleep;
|
423 | function xpost(url, param, headers, timeout) {
|
424 | if (timeout === void 0) { timeout = 3000; }
|
425 | return __awaiter(this, void 0, void 0, function () {
|
426 | var res, json, text, err_1;
|
427 | return __generator(this, function (_a) {
|
428 | switch (_a.label) {
|
429 | case 0:
|
430 | // TODO 线上测试不稳定超时暂时忽略掉通过进程最大运行时间去控制超时失败
|
431 | timeout = 5000; // -1 不行线上会被阻塞住僵死
|
432 | res = null;
|
433 | json = null;
|
434 | text = null;
|
435 | _a.label = 1;
|
436 | case 1:
|
437 | _a.trys.push([1, 4, , 5]);
|
438 | return [4 /*yield*/, fetch(url, {
|
439 | method: 'POST',
|
440 | body: JSON.stringify(param),
|
441 | headers: __assign({ 'Content-Type': 'application/json' }, headers),
|
442 | timeout: timeout <= 0 ? 0 : timeout,
|
443 | })];
|
444 | case 2:
|
445 | res = _a.sent();
|
446 | return [4 /*yield*/, res.text()]; // 解析出完整的返回内容避免HTML以及非法格式信息便于正确报错定位后端接口错误
|
447 | case 3:
|
448 | text = _a.sent(); // 解析出完整的返回内容避免HTML以及非法格式信息便于正确报错定位后端接口错误
|
449 | json = JSON.parse(text);
|
450 | return [2 /*return*/, json];
|
451 | case 4:
|
452 | err_1 = _a.sent();
|
453 | xthrow(err_1, { url: url, param: param, headers: headers, text: text });
|
454 | return [3 /*break*/, 5];
|
455 | case 5: return [2 /*return*/];
|
456 | }
|
457 | });
|
458 | });
|
459 | }
|
460 | exports.xpost = xpost;
|
461 | // 默认超时3000毫秒
|
462 | function xget(url, param, headers, timeout) {
|
463 | if (timeout === void 0) { timeout = 3000; }
|
464 | return __awaiter(this, void 0, void 0, function () {
|
465 | var res, json, text, err_2;
|
466 | return __generator(this, function (_a) {
|
467 | switch (_a.label) {
|
468 | case 0:
|
469 | // TODO 线上测试不稳定超时暂时忽略掉通过进程最大运行时间去控制超时失败
|
470 | timeout = 5000; // -1 不行线上会被阻塞住僵死
|
471 | res = null;
|
472 | json = null;
|
473 | text = null;
|
474 | _a.label = 1;
|
475 | case 1:
|
476 | _a.trys.push([1, 4, , 5]);
|
477 | url = url + (param ? '?' : '') + querystring.stringify(param);
|
478 | return [4 /*yield*/, fetch(url, {
|
479 | method: 'GET',
|
480 | headers: __assign({ 'Content-Type': 'application/json' }, headers),
|
481 | timeout: timeout <= 0 ? 0 : timeout,
|
482 | })];
|
483 | case 2:
|
484 | res = _a.sent();
|
485 | return [4 /*yield*/, res.text()]; // 解析出完整的返回内容避免HTML以及非法格式信息便于正确报错定位后端接口错误
|
486 | case 3:
|
487 | text = _a.sent(); // 解析出完整的返回内容避免HTML以及非法格式信息便于正确报错定位后端接口错误
|
488 | json = JSON.parse(text);
|
489 | return [2 /*return*/, json];
|
490 | case 4:
|
491 | err_2 = _a.sent();
|
492 | xthrow(err_2, { url: url, param: param, headers: headers, text: text });
|
493 | return [3 /*break*/, 5];
|
494 | case 5: return [2 /*return*/];
|
495 | }
|
496 | });
|
497 | });
|
498 | }
|
499 | exports.xget = xget;
|
500 | // 302临时重定向跳转实现
|
501 | function xredirect(url, param) {
|
502 | if (param === void 0) { param = {}; }
|
503 | // TODO 多个程序实例并发处理的时候存在时序问题不能保证全局变量被准确清空。
|
504 | // 检查应用重复设置重定向地址未及时return返回控制器问题
|
505 | xassert(global['__redirect_url__'] === undefined);
|
506 | if (param) {
|
507 | xassert(_.isPlainObject(param));
|
508 | // 删除param中两个框架预定义参数__url__和__api__不允许进行参数传递(禁止业务逻辑使用避免框架后续升级以及与短网址功能冲突)
|
509 | delete param.__api__;
|
510 | delete param.__url__;
|
511 | // 补额外的附加参数
|
512 | if (/\?/.test(url)) {
|
513 | url += '&';
|
514 | }
|
515 | else {
|
516 | url += '?';
|
517 | }
|
518 | url += querystring.stringify(param);
|
519 | }
|
520 | global['__redirect_url__'] = url;
|
521 | }
|
522 | exports.xredirect = xredirect;
|
523 | // 如果只有key参数表示读取属性(缺省值为undefined),如果key为空表示读取所有的请求cookies属性,否则表示响应设置cookies
|
524 | function xcookie(key, value, option) {
|
525 | if (!arguments.length) {
|
526 | // 读取所有的请求cookies属性object
|
527 | return global['__request_cookies__'] ? global['__request_cookies__'] : {};
|
528 | }
|
529 | else if (arguments.length == 1) {
|
530 | return key ? xcookie()[key] : undefined;
|
531 | }
|
532 | else {
|
533 | if (global['__respond_cookies__'] === undefined) {
|
534 | global['__respond_cookies__'] = {};
|
535 | }
|
536 | if (key) {
|
537 | // COOKIES缺省属性设置(有效时间24小时并且统一关联到根页面上获取COOKIES值)
|
538 | option = xassign({ path: '/', maxAge: 24 * 3600 }, option);
|
539 | global['__respond_cookies__'][key] = cookie.serialize(key, value, option);
|
540 | }
|
541 | return;
|
542 | }
|
543 | }
|
544 | exports.xcookie = xcookie;
|
545 | // 判断user-agent请求是否为移动端
|
546 | function xismobile() {
|
547 | var md = new MobileDetect(global['__user_agent__']);
|
548 | return !!md.mobile();
|
549 | }
|
550 | function xassign(target, source) {
|
551 | var args = [];
|
552 | for (var _i = 2; _i < arguments.length; _i++) {
|
553 | args[_i - 2] = arguments[_i];
|
554 | }
|
555 | var param = [true, target, source].concat(args);
|
556 | return extend.apply(null, param);
|
557 | }
|
558 | // 查询app/config目录下的应用配置数据
|
559 | function xconfig(path, defaultValue) {
|
560 | if (defaultValue === void 0) { defaultValue = undefined; }
|
561 | if (global['__config__']) {
|
562 | return _.get(global['__config__'], path, defaultValue);
|
563 | }
|
564 | var fp = require('path');
|
565 | var fs = require('fs');
|
566 | // 自动获取app/config的相对路径目录位置得到根路径的位置
|
567 | var config_path = '';
|
568 | if (__dirname.includes('/node_modules/@bxjs/base/')) {
|
569 | // 在应用目录下
|
570 | config_path = fp.join(__dirname, '../../../../app/config');
|
571 | }
|
572 | else {
|
573 | // 在axjs库开发目录下
|
574 | config_path = fp.join(__dirname, '../app/config');
|
575 | }
|
576 | // 自动识别判断运行环境global['__env__']并且加载对应的base数据和env数据
|
577 | var config_base_path = config_path + '/config.base.' + get_suffix_ts_or_js();
|
578 | var config_env_path = config_path + ("/config." + global['__env__'] + ".") + get_suffix_ts_or_js();
|
579 | if (!fs.existsSync(config_base_path)) {
|
580 | return defaultValue;
|
581 | }
|
582 | var config_base = require(config_base_path).default;
|
583 | var config_env = {};
|
584 | if (fs.existsSync(config_env_path)) {
|
585 | config_env = require(config_env_path).default;
|
586 | }
|
587 | // bugfix Object.assign不支持深度拷贝问题
|
588 | // global['__config__'] = Object.assign({}, config_base, config_env)
|
589 | // global['__config__'] = _.assign({}, config_env, config_base)
|
590 | global['__config__'] = xassign({}, config_base, config_env);
|
591 | return _.get(global['__config__'], path, defaultValue);
|
592 | }
|
593 | function xconnect(callback, config) {
|
594 | if (config === void 0) { config = 'default'; }
|
595 | return __awaiter(this, void 0, void 0, function () {
|
596 | var _this = this;
|
597 | return __generator(this, function (_a) {
|
598 | return [2 /*return*/, new Promise(function (resolve, reject) { return __awaiter(_this, void 0, void 0, function () {
|
599 | var cfg, mng, name, db_1, err_3;
|
600 | var _this = this;
|
601 | return __generator(this, function (_a) {
|
602 | switch (_a.label) {
|
603 | case 0:
|
604 | cfg = {};
|
605 | _a.label = 1;
|
606 | case 1:
|
607 | _a.trys.push([1, 3, , 4]);
|
608 | cfg = xassign({}, xconfig('plugins.database.default', {}));
|
609 | xassert(!_.isEmpty(cfg), ERR$PARAM, { config: config });
|
610 | // 强制补上约定的实体存放路径定义位置(不允许配置是约定规范)
|
611 | if (__dirname.includes('/node_modules/@bxjs/base/')) {
|
612 | // 在应用目录下
|
613 | cfg['entities'] = [
|
614 | path.join(__dirname, '../../../../app/plugins/database/entity/*.' + get_suffix_ts_or_js())
|
615 | ];
|
616 | }
|
617 | else {
|
618 | // 在axjs库开发目录下
|
619 | cfg['entities'] = [
|
620 | path.join(__dirname, '../app/plugins/database/entity/*.' + get_suffix_ts_or_js())
|
621 | ];
|
622 | }
|
623 | mng = typeorm_1.getConnectionManager();
|
624 | name = cfg.name ? cfg.name : 'default';
|
625 | if (!mng.has(name)) {
|
626 | mng.create(cfg);
|
627 | }
|
628 | db_1 = mng.get(name);
|
629 | if (global['__connection__'] === undefined) {
|
630 | global['__connection__'] = {};
|
631 | }
|
632 | if (!db_1.isConnected) { // TODO 需要进行连接池的管理
|
633 | global['__connection__'][name] = db_1.connect();
|
634 | }
|
635 | return [4 /*yield*/, global['__connection__'][name].then(function (connection) { return __awaiter(_this, void 0, void 0, function () {
|
636 | var out;
|
637 | return __generator(this, function (_a) {
|
638 | switch (_a.label) {
|
639 | case 0:
|
640 | xassert(db_1.isConnected);
|
641 | return [4 /*yield*/, callback(connection)
|
642 | // await db.close() // typeorm没有进行连接池的管理不能进行销毁
|
643 | ];
|
644 | case 1:
|
645 | out = _a.sent();
|
646 | // await db.close() // typeorm没有进行连接池的管理不能进行销毁
|
647 | resolve(out);
|
648 | return [2 /*return*/];
|
649 | }
|
650 | });
|
651 | }); }).catch(function (err) { return __awaiter(_this, void 0, void 0, function () {
|
652 | return __generator(this, function (_a) {
|
653 | // await db.close()
|
654 | xthrow(err, reject, { cfg: cfg });
|
655 | return [2 /*return*/];
|
656 | });
|
657 | }); })];
|
658 | case 2:
|
659 | _a.sent();
|
660 | return [3 /*break*/, 4];
|
661 | case 3:
|
662 | err_3 = _a.sent();
|
663 | xthrow(err_3, reject, { cfg: cfg });
|
664 | return [3 /*break*/, 4];
|
665 | case 4: return [2 /*return*/];
|
666 | }
|
667 | });
|
668 | }); })];
|
669 | });
|
670 | });
|
671 | }
|
672 | // 创建XBaseEntity对象并且自动赋值前端请求的赋值数据
|
673 | function xnew(TYPE, param) {
|
674 | var args = [];
|
675 | for (var _i = 2; _i < arguments.length; _i++) {
|
676 | args[_i - 2] = arguments[_i];
|
677 | }
|
678 | // 泛型实现类似这个功能
|
679 | // asset = new AlilangAsset()
|
680 | // getRepository(AlilangAsset).merge(asset, param as any)
|
681 | // AlilangAsset.merge(asset, param as any)
|
682 | // return asset
|
683 | var obj = new TYPE();
|
684 | if (_.isEmpty(param)) {
|
685 | return obj;
|
686 | }
|
687 | var repo = typeorm_1.getRepository(TYPE);
|
688 | repo.merge.apply(repo, [obj, param].concat(args));
|
689 | return obj;
|
690 | }
|
691 | // 查询构造器易用性封装
|
692 | function xquery(connect, TYPE, alias) {
|
693 | return connect.getRepository(TYPE).createQueryBuilder(alias);
|
694 | }
|
695 | // 分页查询获取总数以及原始记录数据
|
696 | function xcount(sql, page, size) {
|
697 | return __awaiter(this, void 0, void 0, function () {
|
698 | var _a, count, rows;
|
699 | return __generator(this, function (_b) {
|
700 | switch (_b.label) {
|
701 | case 0:
|
702 | xassert(page >= 1);
|
703 | return [4 /*yield*/, Promise.all([
|
704 | sql.getCount(),
|
705 | sql.offset((page - 1) * size).limit(size).getRawMany()
|
706 | ])];
|
707 | case 1:
|
708 | _a = _b.sent(), count = _a[0], rows = _a[1];
|
709 | return [2 /*return*/, [rows, count]];
|
710 | }
|
711 | });
|
712 | });
|
713 | }
|
714 | // 路由参数的修饰符配置
|
715 | // TODO 更多接口相关参数的配置扩展,例如:是否支持JSONP
|
716 | function xroute(param) {
|
717 | // 缺省值处理
|
718 | param = xassign({ name: '', desc: '', path: '', auth: true }, param);
|
719 | return function (target, propertyKey, descriptor) {
|
720 | var _this = this;
|
721 | // TODO 注入到类实例定义中进行全局引用动态类的特性添加(trait功能的动态实现)
|
722 | // 动态绑定路由类实例的上下文属性
|
723 | target.prototype.context = function () {
|
724 | return {
|
725 | param: param,
|
726 | // 是否登录的鉴权方法统一框架层面上的处理实现,此处仅仅是通用接口的约束的定义。
|
727 | auth: function () { return __awaiter(_this, void 0, void 0, function () {
|
728 | var auth, _a;
|
729 | return __generator(this, function (_b) {
|
730 | switch (_b.label) {
|
731 | case 0:
|
732 | if (!(param && param.auth)) return [3 /*break*/, 2];
|
733 | auth = xgot(YAuth);
|
734 | _a = xassert;
|
735 | return [4 /*yield*/, auth.getLoginStatus()];
|
736 | case 1:
|
737 | _a.apply(void 0, [_b.sent(), ERR$UNAUTHORIZED]);
|
738 | _b.label = 2;
|
739 | case 2: return [2 /*return*/];
|
740 | }
|
741 | });
|
742 | }); }
|
743 | };
|
744 | };
|
745 | };
|
746 | }
|
747 | // 完全没有必要的多余定义,需要通过MOCK定义进行细节数据类型的显性定义处理逻辑验证。
|
748 | // // 基本数据类型的规范扩展定义,方便API接口的定义以及形参自动验证合法性,并且与数据库数据类型保持一致。
|
749 | // type INT = number // 有符号整数
|
750 | // type UINT = number // 无符号整数
|
751 | // type DECIMAL = number // 精确小数
|
752 | // type FLOAT = number // 单精度浮点数(不精确小数)
|
753 | // type DOUBLE = number// 双精度浮点数(不精确小数)
|
754 | // type BOOL = boolean
|
755 | // type STR = string
|
756 | // type DATE = string // 年月日 '2017-06-25'
|
757 | // type TIME = string // 时分秒 '00:00:00'
|
758 | // type DATETIME = string // 年月日时分秒 '2017-06-25 00:00:00'
|
759 | // 模拟数据模板定义使用教程 http://mockjs.com/0.1/#%E6%95%B0%E6%8D%AE%E5%8D%A0%E4%BD%8D%E7%AC%A6%E5%AE%9A%E4%B9%89%20DPD
|
760 | function xmock(rules) {
|
761 | return mockjs.mock(rules);
|
762 | }
|
763 | function xrandom(name, data) {
|
764 | var _a;
|
765 | mockjs.Random.extend((_a = {},
|
766 | _a[name] = function () {
|
767 | var args = [];
|
768 | for (var _i = 0; _i < arguments.length; _i++) {
|
769 | args[_i] = arguments[_i];
|
770 | }
|
771 | xassert(data.length > 0);
|
772 | if (data.length == 1)
|
773 | return data[0];
|
774 | var max = data.length - 1;
|
775 | var idx = xmock("@int(0," + max + ")");
|
776 | return data[idx];
|
777 | },
|
778 | _a));
|
779 | }
|
780 | // 扩展一些预定义bxjs的基础随机方法或者覆盖一些mockjs中的方法
|
781 | mockjs.Random.extend({
|
782 | // bxjs表定义的主键统一定义(约定系统中为字符串7-14字节长度算法)
|
783 | id: function () {
|
784 | var args = [];
|
785 | for (var _i = 0; _i < arguments.length; _i++) {
|
786 | args[_i] = arguments[_i];
|
787 | }
|
788 | return shortid.generate();
|
789 | },
|
790 | // 中国手机号随机生成算法(约定系统中的手机号为字符串数据类型)
|
791 | mobile: function () {
|
792 | var args = [];
|
793 | for (var _i = 0; _i < arguments.length; _i++) {
|
794 | args[_i] = arguments[_i];
|
795 | }
|
796 | var isps = [
|
797 | 134, 135, 136, 137, 138, 139, 147, 150, 151, 152, 157, 158, 159, 182, 183, 184, 187, 188, 178,
|
798 | 130, 131, 132, 145, 155, 156, 185, 186, 176,
|
799 | 133, 134, 153, 180, 181, 189, 177, 173,
|
800 | 176, 173, 177, 178, 170,
|
801 | 140, 141, 142, 143, 144, 146, 148, 149, 154
|
802 | ];
|
803 | var max = isps.length - 1;
|
804 | var idx = xmock("@int(0," + max + ")");
|
805 | var num = xmock("@int(100000000,199999999)");
|
806 | return (isps[idx] * 100000000 + num % 100000000) + '';
|
807 | },
|
808 | // 转换为缺省中文内容提示
|
809 | paragraph: function () {
|
810 | var args = [];
|
811 | for (var _i = 0; _i < arguments.length; _i++) {
|
812 | args[_i] = arguments[_i];
|
813 | }
|
814 | switch (args.length) {
|
815 | case 0:
|
816 | return xmock('@cparagraph');
|
817 | case 1:
|
818 | return xmock("@cparagraph(" + args[0] + ")");
|
819 | case 2:
|
820 | return xmock("@cparagraph(" + args[0] + "," + args[1] + ")");
|
821 | default:
|
822 | xassert(false);
|
823 | }
|
824 | },
|
825 | sentence: function () {
|
826 | var args = [];
|
827 | for (var _i = 0; _i < arguments.length; _i++) {
|
828 | args[_i] = arguments[_i];
|
829 | }
|
830 | switch (args.length) {
|
831 | case 0:
|
832 | return xmock('@csentence');
|
833 | case 1:
|
834 | return xmock("@csentence(" + args[0] + ")");
|
835 | case 2:
|
836 | return xmock("@csentence(" + args[0] + "," + args[1] + ")");
|
837 | default:
|
838 | xassert(false);
|
839 | }
|
840 | },
|
841 | title: function () {
|
842 | var args = [];
|
843 | for (var _i = 0; _i < arguments.length; _i++) {
|
844 | args[_i] = arguments[_i];
|
845 | }
|
846 | switch (args.length) {
|
847 | case 0:
|
848 | return xmock('@ctitle');
|
849 | case 1:
|
850 | return xmock("@ctitle(" + args[0] + ")");
|
851 | case 2:
|
852 | return xmock("@ctitle(" + args[0] + "," + args[1] + ")");
|
853 | default:
|
854 | xassert(false);
|
855 | }
|
856 | },
|
857 | });
|
858 | // laravel风格JSON对象验证器封装,详细文档见 https://github.com/skaterdav85/validatorjs
|
859 | function xcheck(param, rules, messages) {
|
860 | var obj = new validatorjs(param, rules);
|
861 | if (obj.fails()) {
|
862 | xthrow(ERR$PARAM, obj.errors);
|
863 | }
|
864 | }
|
865 | // 【IoC容器管理】应用层的插件实现类绑定到BXJS统一注册的标准插件的映射关系在全局容器实例中注册
|
866 | function xbind(TYPE) {
|
867 | var o = new TYPE();
|
868 | return xcontainer.bind(o.id).to(require("@app/plugins/" + o.id).default);
|
869 | }
|
870 | // 【IoC容器管理】框架或应用依赖标准规范接口插件的类实例获取方法
|
871 | function xgot(TYPE) {
|
872 | var o = new TYPE();
|
873 | return xcontainer.get(o.id);
|
874 | }
|
875 | // 同步系统命令调用执行
|
876 | function xcmd() {
|
877 | var args = [];
|
878 | for (var _i = 0; _i < arguments.length; _i++) {
|
879 | args[_i] = arguments[_i];
|
880 | }
|
881 | return __awaiter(this, void 0, void 0, function () {
|
882 | var options, cmd, ret, err_4;
|
883 | return __generator(this, function (_a) {
|
884 | switch (_a.label) {
|
885 | case 0:
|
886 | _a.trys.push([0, 1, , 3]);
|
887 | options = {};
|
888 | options.cwd = options.cwd || process.env.__ctxPath || process.cwd();
|
889 | xassert(_.isArray(args) && args.length > 0);
|
890 | cmd = args.shift();
|
891 | ret = cross_spawn.sync(cmd, args, xassign({ stdio: 'inherit' }, options));
|
892 | xassert(ret.status === 0, ERR$UNKNOWN, ret);
|
893 | return [2 /*return*/, ret];
|
894 | case 1:
|
895 | err_4 = _a.sent();
|
896 | return [4 /*yield*/, xwarn(err_4)];
|
897 | case 2:
|
898 | _a.sent();
|
899 | xthrow(err_4);
|
900 | return [3 /*break*/, 3];
|
901 | case 3: return [2 /*return*/];
|
902 | }
|
903 | });
|
904 | });
|
905 | }
|
906 | // 对于数组嵌套回调函数的nodejs异步处理方法的统一封装
|
907 | function xmap(values, callack) {
|
908 | return __awaiter(this, void 0, void 0, function () {
|
909 | return __generator(this, function (_a) {
|
910 | xassert(_.isArray(values) && _.isFunction(callack));
|
911 | return [2 /*return*/, Promise.all(values.map(callack))];
|
912 | });
|
913 | });
|
914 | }
|
915 | // Refer to document: https://help.aliyun.com/document_detail/62670.html
|
916 | // 获取ACM配置信息接口的统一封装
|
917 | function xacm(group, id) {
|
918 | return __awaiter(this, void 0, void 0, function () {
|
919 | var cfg, acm;
|
920 | var _this = this;
|
921 | return __generator(this, function (_a) {
|
922 | cfg = xconfig("plugins.acm");
|
923 | xassert(group && cfg && cfg[group], ERR$CONFIG, { cfg: cfg });
|
924 | acm = new ACMClient(cfg[group]);
|
925 | return [2 /*return*/, new Promise(function (resolve, reject) { return __awaiter(_this, void 0, void 0, function () {
|
926 | return __generator(this, function (_a) {
|
927 | try {
|
928 | co(function () {
|
929 | var content, err_5;
|
930 | return __generator(this, function (_a) {
|
931 | switch (_a.label) {
|
932 | case 0:
|
933 | _a.trys.push([0, 2, , 3]);
|
934 | group = group + ':' + global['__env__']; // 补上环境后缀支持各种开发环境的个性化配置
|
935 | return [4 /*yield*/, acm.getConfig(id, group)];
|
936 | case 1:
|
937 | content = _a.sent();
|
938 | xassert(content, ERR$CONFIG, { id: id, group: group });
|
939 | resolve(content);
|
940 | return [3 /*break*/, 3];
|
941 | case 2:
|
942 | err_5 = _a.sent();
|
943 | xthrow(err_5, reject, { id: id, group: group, cfg: cfg[group] });
|
944 | return [3 /*break*/, 3];
|
945 | case 3: return [2 /*return*/];
|
946 | }
|
947 | });
|
948 | });
|
949 | }
|
950 | catch (err) {
|
951 | xthrow(err, reject, { id: id, group: group, cfg: cfg[group] });
|
952 | }
|
953 | return [2 /*return*/];
|
954 | });
|
955 | }); })];
|
956 | });
|
957 | });
|
958 | }
|
959 | if (!global['__env__']) {
|
960 | global['__env__'] = 'local'; // local,daily,pre,gray,prod 在统一入口处自动识别配置(目前暂不支持gray配置尚未开发无法自动识别)
|
961 | }
|
962 | global['__config__'] = undefined;
|
963 | global['__session__'] = {};
|
964 | global['__cache__'] = {};
|
965 | global['__user__'] = {};
|
966 | global['__user_agent__'] = undefined;
|
967 | global['__client_ip__'] = undefined;
|
968 | global['__redirect_url__'] = undefined;
|
969 | global['__request_cookies__'] = {};
|
970 | global['__respond_cookies__'] = {};
|
971 | global['xconnect'] = xconnect;
|
972 | global['xnew'] = xnew;
|
973 | global['xquery'] = xquery;
|
974 | global['xcount'] = xcount;
|
975 | global['xassign'] = xassign;
|
976 | global['xconfig'] = xconfig;
|
977 | global['xthrow'] = xthrow;
|
978 | global['xassert'] = xassert;
|
979 | global['xerror'] = xerror;
|
980 | global['xroot'] = xroot;
|
981 | global['xstack'] = xstack;
|
982 | global['xwarn'] = xwarn;
|
983 | global['xlog'] = xlog;
|
984 | global['xpost'] = xpost;
|
985 | global['xget'] = xget;
|
986 | global['xsleep'] = xsleep;
|
987 | global['xredirect'] = xredirect;
|
988 | global['xcookie'] = xcookie;
|
989 | global['xismobile'] = xismobile;
|
990 | global['xsession'] = session_1.xsession;
|
991 | global['xuser'] = session_1.xuser;
|
992 | global['xcache'] = session_1.xcache;
|
993 | global['xroute'] = xroute;
|
994 | global['xmock'] = xmock;
|
995 | global['xrandom'] = xrandom;
|
996 | global['xcheck'] = xcheck;
|
997 | global['xcontainer'] = new inversify_1.Container(); // 全局单实例容器初始化
|
998 | global['xbind'] = xbind;
|
999 | global['xgot'] = xgot;
|
1000 | global['YAuth'] = $$.YAuth; // 全局声明认证插件规范抽象类
|
1001 | global['xcmd'] = xcmd;
|
1002 | global['xmap'] = xmap;
|
1003 | global['xacm'] = xacm;
|
1004 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"base.js","sourceRoot":"","sources":["base.ts"],"names":[],"mappings":"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","sourcesContent":["process.env.NODE_TLS_REJECT_UNAUTHORIZED = '0'\nimport 'reflect-metadata'\nimport {Container} from 'inversify'\nimport {getConnectionManager, Connection, BaseEntity, getRepository, SelectQueryBuilder} from '@bxjs/typeorm'\nimport {xsession, xuser, xcache} from './session'\nimport * as $$ from './plugins'\n\nconst path = require('path')\nconst ErrorStackParser = require('error-stack-parser')\nconst cookie = require('cookie')\nconst MobileDetect = require('mobile-detect')\nconst fetch = require('node-fetch')\nconst _ = require('lodash')\nconst moment = require('moment')\nconst extend = require('extend')\nconst querystring = require('querystring')\n// const parameter = require('parameter')\n// const parameterCheckInstance = new parameter({\n//     // translate: function () {\n//     //     var args = Array.prototype.slice.call(arguments);\n//     //     // Assume there have I18n.t method for convert language.\n//     //     return I18n.t.apply(I18n, args);\n//     // }\n// })\nconst circular_json = require(\"circular-json\")\nconst mockjs = require('mockjs')\nconst shortid = require('shortid')\nconst validatorjs = require('validatorjs')\nconst cross_spawn = require('cross-spawn')\nconst ACMClient = require('acm-client')\nconst co = require('co')\n\n// FIXME HACK原生方法JSON转换不可逆的BUG（JAVA端传来的富文本字段内容含有\\n\\t字符串中的字符生成JSON字符串无法正常解析报错）\nconst raw_stringify = JSON.stringify\n\nfunction new_stringify(value: any, replacer?: (key: string, value: any) => any,\n                       space?: string | number): string {\n    let out = raw_stringify(value, replacer, space)\n    if (_.isString(out)) {\n        out = out.replace(/\\\\n/g, '\\\\\\\\n')\n            .replace(/\\\\t/g, '\\\\\\\\t')\n            .replace(/\\\\u/g, '\\\\\\\\u') //JAVA端返回的unicode字符转义处理\n    }\n    return out\n}\n\nJSON.stringify = new_stringify as any\n\n// ts-node本地调试需要加载对应的源代码后缀名称\nexport function get_suffix_ts_or_js() {\n    if (global['__env__'] == 'local' && !/^\\/code\\/node_modules/.test(__dirname)) {\n        return 'ts'\n    } else {\n        return 'js'\n    }\n}\n\n// 准确定位错误码位置，间接得到函数调用位置地址信息，结合符号报表的正确解析处理完美得到错误定位信息，准确代码调试。\nfunction __get_base_func_caller_source_position(position: number = 3) {\n    try {\n        throw new Error()\n    } catch (err) {\n        let out = ErrorStackParser.parse(err)\n        let idx = 0\n        // 找到第二个TS文件的执行位置\n        let find_ts_sufix_file_count = 0\n        for (; idx < out.length; idx++) {\n            if (/\\.ts$/.test(out[idx].fileName)) {\n                find_ts_sufix_file_count += 1\n            }\n            if (find_ts_sufix_file_count == position) {\n                break\n            }\n        }\n        if (find_ts_sufix_file_count == position) {\n            return '[' + out[idx]['fileName'] + ':' + out[idx]['lineNumber'] + ']'\n        } else {\n            // TODO 需要定位为什么调用栈无法找到对应的位置出现越界？？\n            // console.error(err)\n            console.error(circular_json.stringify(out, null, 4)\n                .replace(/\\r/g, '').replace(/\\n/g, ''))\n            return '#'\n        }\n\n    }\n}\n\n// 获取异常调用栈用于辅助错误提示定位\nexport function xstack(err, compact = true) {\n    try {\n        // TODO 优化裁剪一些无用信息减少日志尺寸更加便于人工分析处理\n        let stack = ErrorStackParser.parse(err)\n        if (compact) {\n            let sources: string[] = []\n            for (let v of stack) {\n                sources.push(`${v['fileName']}:${v['lineNumber']}`)\n            }\n            return sources\n        }\n        return stack\n    } catch (err1) {\n        let source = __get_base_func_caller_source_position()\n        return `invalid error input param (${source})`\n    }\n}\n\n// // 错误栈的递归嵌套格式显示数据结构定义（param嵌套找到最后一个msg的JSON解析语法错误就是错误链的原始错误发生位置）\n// let x = {\n//     \"code\": \"UNKNOWN\",\n//     \"msg\": \"未知错误\",\n//     \"param\": {\n//         \"msg\": \"您输入的用户名或密码错误，请重新登录 (ErrorCode: 1005, url: https://login.alibaba-inc.com/authorize/login.do)\"\n//     },\n//     \"stack\": \"[\\\"/Users/chujinghui/Desktop/work/xjs/bxjs/framework/base.ts:110\\\",\\\"/Users/chujinghui/Desktop/work/xjs/bxjs/app/entries/web/mobile/meeting-room-visit.ts:161\\\",\\\"/Users/chujinghui/Desktop/work/xjs/bxjs/app/entries/web/mobile/meeting-room-visit.js:40\\\",\\\"/Users/chujinghui/Desktop/work/xjs/bxjs/app/entries/web/mobile/meeting-room-visit.js:21\\\",\\\"/Users/chujinghui/Desktop/work/xjs/bxjs/app/entries/web/mobile/meeting-room-visit.js:13\\\",\\\"internal/process/next_tick.js:188\\\"]\",\n// }\n\n// 对于异常内容的格式化参数解析处理成为四元组code/msg/param/stack\nexport function xerror(err, __param?: any) {\n    xassert(err instanceof Error)\n    try {\n        // 标准错误的统一转换处理\n        let data: any = JSON.parse(err.message)\n        if (data.code && data.msg && ERRORS[data.code]) {\n            return data\n        }\n    } catch (err) {\n        // ignore parse error\n    }\n    // 非标准错误的统一格式转换处理\n    let msg = ERRORS[ERR$UNKNOWN]['zh'] // TODO 错误码多语言回传到客户端问题\n    let code = ERR$UNKNOWN\n    let param: any = {msg: err.message, param: __param} // 用户自定义的错误参数信息 msg为非错误码JSON四元组就是嵌套的终止条件。\n    let stack = xstack(err)\n    let data = {msg, code, param, stack}\n    return data\n}\n\n// 用于获取错误栈的root cause根本原因（第一个被拦截的错误发生位置）\nexport function xroot(err: Error) {\n    xassert(err instanceof Error)\n    let {msg, param, code, stack} = xerror(err)\n\n    // 递归遍历找到错误链的root cause\n    for (; param && param.msg;) {\n        try {\n            let json: any = JSON.parse(param.msg)\n            param = json.param\n        } catch (err) {\n            msg = param.msg\n            code = param.code\n            stack = param.stack\n            param = param.param\n            break\n        }\n    }\n    return {msg, code, param, stack}\n}\n\n// TODO 报错处理（显示问题反馈联系人信息）\n// 将未处理的错误上抛的异常链记录下来用于精准追踪代码的执行过程（以及准确获取到根节点的错误码）\n// 对于promise异步回调的统一出错处理写法实例\n// export function login(username: string, password: string) {\n//     return new Promise((resolve, reject) => {\n//         co(function* () {\n//             let user = yield buc.oauthclient.login(username, password)\n//             resolve(user)\n//         }).catch(async function (err) {\n//             xthrow(err, reject)\n//         })\n//     })\n// }\nexport function xthrow(code: string | Error = ERR$UNKNOWN, param: any = undefined, reject_param: any = undefined) {\n    // promise中进行reject异常处理的抛出错误方法的形参逻辑预处理转换。\n    let reject: any = _.isFunction(param) ? param : undefined\n    if (reject) param = reject_param\n    let data: any = {}\n    let source = __get_base_func_caller_source_position()\n\n    if (code instanceof Error) {\n        try {\n            data = JSON.parse(code.message)\n            // 将透传上抛的错误的路径信息和附加参数也记录下来方便提供完整应用堆栈信息辅助调试业务逻辑\n            if (!_.isArray(data.stack)) {\n                data.stack = []\n            }\n            data.stack.push(source)\n        } catch (err) {\n            // ignore\n        }\n        // 标准错误直接上抛处理\n        if (data.code && data.msg && ERRORS[data.code]) {\n            // 测试严重BUG reject函数类型表达式为假必须要用lodash判定是否为函数\n            if (_.isFunction(reject)) {\n                // promise回调中进行抛错误处理\n                let err = new Error(JSON.stringify(data))\n                reject(err)\n                return\n            } else {\n                throw new Error(JSON.stringify(data))\n            }\n        }\n        // 将非标准错误转换为标准错误后再上抛处理\n        data = xerror(code, param)\n        data.code = ERR$UNKNOWN\n        data.msg = ERRORS[ERR$UNKNOWN]['zh'] // FIXME TODO 错误码的多语言处理转换！！\n        data.param = {msg: code.message, param, stack: [source]}\n    } else {\n        // 对于常量定义错误的统一格式化处理\n        data = {code, msg: global['ERRORS'][code as string]['zh'], param, stack: [source]}\n    }\n\n    // 对于是否promise场景下的错误上抛进行正确的转换处理\n    if (_.isFunction(reject)) {\n        // promise回调中进行抛错误处理\n        reject(new Error(JSON.stringify(data)))\n    } else {\n        // 非promise回调中异常传递\n        throw new Error(JSON.stringify(data))\n    }\n}\n\nexport function xassert(expr: any, code: string = ERR$ASSERT, param?: any) {\n    let source = __get_base_func_caller_source_position()\n    let stack = [source]\n    if (!expr) throw new Error(JSON.stringify({code, msg: global['ERRORS'][code]['zh'], param, stack}))\n    return expr\n}\n\n// // https://github.com/node-modules/parameter 参数验证规则详见此文档（egg团队开发的组件）\n// // 注意事项：GET通过URL传递的参数都是字符串类型应该尽量避免GET传递参数，需要多用POST的JSON格式传递参数并且POSTMAN上进行辅助测试正确数据类型映射。\n// export function xcheck(param: { [propName: string]: any }, rules: { [propName: string]: any }) {\n//     let errors = parameterCheckInstance.validate(rules, param)\n//     if (_.isEmpty(errors)) {\n//         return true\n//     } else {\n//         xthrow(ERR$PARAM, errors)\n//     }\n// }\n\nexport function xlog(...args) {\n    // 兼容云端以及本地日志调试（解决任意对象的JSON字符串内容的完整输出）\n    let source = __get_base_func_caller_source_position()\n    let output = circular_json.stringify([...args], null, 4)\n    if (global['__env__'] != 'prod' && !/^\\/code\\/node_modules/.test(__dirname)) {\n        // 打印到控制台一份日志(在阿里云非线上FC环境中)\n        console.log.apply(undefined, [source + output])\n        // 写日志文件到/tmp下临时处理一下 TODO 需要改为类似log4j的本地日志库仅在非线上环境使用方便开发单机日常机器上调试。\n        const fs = require('fs')\n        const logFilePath = process.env['NODE_LOGFILE'] ? process.env['NODE_LOGFILE'] : '/tmp/bxjs.log'\n        fs.appendFileSync(logFilePath, source + output + \"\\r\\n\")\n    } else {\n        // 生产环境下只打印到控制台绑定的SLS日志服务器上，并且需要去除掉换行信息否则打印会不正常。\n        output = output.replace(/\\r/g, '').replace(/\\n/g, '')\n        console.log.apply(undefined, [source + output])\n    }\n}\n\n// // 将详细错误信息及时发送到钉钉群上实时反馈给维护者\n// await xwarn({\n//     code,\n//     // TODO 如何认证通过了获取到用户信息也需要发送过去，方便联系对接人员进行立刻问题处理反馈。\n//     message,\n//     stack,\n//     param,\n// })\n// 将详细错误信息及时发送到钉钉群上实时反馈给维护者\n// 钉钉IM群机器人报警通知\nasync function xwarn(...args) {\n    // 得到xwarn方法被调用的位置\n    let source = __get_base_func_caller_source_position()\n\n    // 对于异常参数警告信息进行错误内容标准解析\n    if (args.length > 0 && args[0] instanceof Error) {\n        args[0] = xerror(args[0])\n    }\n\n    let out = [source, moment().format('YYYY-MM-DD HH:mm:ss'), {...args}]\n\n    // 从配置信息中读取报警通知人手机列表和对应的群机器人的webhook的access_token信息\n    let access_token = xconfig('framework.warn.dingding.access_token')\n    let mobiles = xconfig('framework.warn.dingding.mobiles')\n    if (!access_token || !mobiles) {\n        access_token = '020a09eac5f2fa320ae851442d5e19e23693c64ad2255c85354b4a49a5a48d35'\n        mobiles = ['15381151346']\n    }\n\n    await xpost(`https://oapi.dingtalk.com/robot/send?access_token=${access_token}`, {\n        msgtype: 'text',\n        text: {\n            content: out\n        },\n        at: {\n            atMobiles: mobiles,\n            isAtAll: false\n        }\n    })\n\n    // 线上SLS日志上也保存一份\n    // console.warn(out)\n    xlog(out)\n}\n\n// 捕获未监听到的异常记录后直接退出（运行堆栈已经破坏直接记录日志后异常退出即可，由外部监控自动重启）\nprocess.on('uncaughtException', async function (err) {\n    xlog(xerror(err))\n    await xwarn(err)\n    process.exit(-1)\n})\n\n// 记录await/async中出现未捕获的异常错误\nprocess.on('unhandledRejection', async (reason, p) => {\n    xlog('Unhandled Rejection at: Promise', p, 'reason:', reason);\n    // application specific logging, throwing an error, or other logic here\n    await xwarn(reason, p)\n    process.exit(-1)\n})\n\n// async/await的非阻塞异步延迟方法，用于调试阻塞程序的执行进行单步调试的效果。\nconst sleep = require('sleep-async')()\n\nexport function xsleep(ms: number = -1) {\n    if (ms <= 0) {\n        ms = 50 * 365 * 24 * 3600 * 1000 // 50年最大数视为永久阻塞方便断点单步调试问题\n    }\n    return new Promise((resolve, reject) => {\n        try {\n            sleep.sleep(ms, () => {\n                resolve()\n            })\n        } catch (err) {\n            xlog(xerror(err))\n            resolve()\n            // xthrow(err,reject)\n        }\n    })\n}\n\nexport async function xpost(url: string, param?: { [propName: string]: any },\n                            headers?: { [propName: string]: any }, timeout: number = 3000) {\n    // TODO 线上测试不稳定超时暂时忽略掉通过进程最大运行时间去控制超时失败\n    timeout = 5000 // -1 不行线上会被阻塞住僵死\n    let res: any = null\n    let json: any = null\n    let text: any = null\n    try {\n        res = await fetch(url, {\n            method: 'POST',\n            body: JSON.stringify(param),\n            headers: {'Content-Type': 'application/json', ...headers},\n            timeout: timeout <= 0 ? 0 : timeout, // 默认3秒超时接口返回避免僵死\n        })\n        text = await res.text() // 解析出完整的返回内容避免HTML以及非法格式信息便于正确报错定位后端接口错误\n        json = JSON.parse(text)\n        return json\n    } catch (err) {\n        xthrow(err, {url, param, headers, text})\n    }\n}\n\n// 默认超时3000毫秒\nexport async function xget(url: string, param?: { [propName: string]: any },\n                           headers?: { [propName: string]: any }, timeout: number = 3000) {\n    // TODO 线上测试不稳定超时暂时忽略掉通过进程最大运行时间去控制超时失败\n    timeout = 5000 // -1 不行线上会被阻塞住僵死\n    let res: any = null\n    let json: any = null\n    let text: any = null\n    try {\n        url = url + (param ? '?' : '') + querystring.stringify(param)\n        res = await fetch(url, {\n            method: 'GET',\n            headers: {'Content-Type': 'application/json', ...headers},\n            timeout: timeout <= 0 ? 0 : timeout, // 默认3秒超时接口返回避免僵死\n        })\n        text = await res.text() // 解析出完整的返回内容避免HTML以及非法格式信息便于正确报错定位后端接口错误\n        json = JSON.parse(text)\n        return json\n    } catch (err) {\n        xthrow(err, {url, param, headers, text})\n    }\n}\n\n// 302临时重定向跳转实现\nexport function xredirect(url: string, param: any = {}) {\n    // TODO 多个程序实例并发处理的时候存在时序问题不能保证全局变量被准确清空。\n    // 检查应用重复设置重定向地址未及时return返回控制器问题\n    xassert(global['__redirect_url__'] === undefined)\n    if (param) {\n        xassert(_.isPlainObject(param))\n        // 删除param中两个框架预定义参数__url__和__api__不允许进行参数传递（禁止业务逻辑使用避免框架后续升级以及与短网址功能冲突）\n        delete param.__api__\n        delete param.__url__\n        // 补额外的附加参数\n        if (/\\?/.test(url)) {\n            url += '&'\n        } else {\n            url += '?'\n        }\n        url += querystring.stringify(param)\n    }\n    global['__redirect_url__'] = url\n}\n\n// 如果只有key参数表示读取属性（缺省值为undefined），如果key为空表示读取所有的请求cookies属性，否则表示响应设置cookies\nexport function xcookie(key?: string, value?: string, option?: {}): any {\n    if (!arguments.length) {\n        // 读取所有的请求cookies属性object\n        return global['__request_cookies__'] ? global['__request_cookies__'] : {}\n    } else if (arguments.length == 1) {\n        return key ? xcookie()[key] : undefined\n    } else {\n        if (global['__respond_cookies__'] === undefined) {\n            global['__respond_cookies__'] = {}\n        }\n        if (key) {\n            // COOKIES缺省属性设置（有效时间24小时并且统一关联到根页面上获取COOKIES值）\n            option = xassign({path: '/', maxAge: 24 * 3600}, option)\n            global['__respond_cookies__'][key] = cookie.serialize(key, value, option)\n        }\n        return\n    }\n}\n\n// 判断user-agent请求是否为移动端\nfunction xismobile(): boolean {\n    const md = new MobileDetect(global['__user_agent__'])\n    return !!md.mobile()\n}\n\nfunction xassign(target, source, ...args) {\n    const param = [true, target, source, ...args]\n    return extend.apply(null, param)\n}\n\n// 查询app/config目录下的应用配置数据\nfunction xconfig(path: string, defaultValue: any = undefined) {\n    if (global['__config__']) {\n        return _.get(global['__config__'], path, defaultValue)\n    }\n\n    const fp = require('path')\n    const fs = require('fs')\n    // 自动获取app/config的相对路径目录位置得到根路径的位置\n    let config_path = ''\n    if (__dirname.includes('/node_modules/@bxjs/base/')) {\n        // 在应用目录下\n        config_path = fp.join(__dirname, '../../../../app/config')\n    } else {\n        // 在axjs库开发目录下\n        config_path = fp.join(__dirname, '../app/config')\n    }\n\n    // 自动识别判断运行环境global['__env__']并且加载对应的base数据和env数据\n    const config_base_path = config_path + '/config.base.' + get_suffix_ts_or_js()\n    const config_env_path = config_path + `/config.${global['__env__']}.` + get_suffix_ts_or_js()\n    if (!fs.existsSync(config_base_path)) {\n        return defaultValue\n    }\n    let config_base = require(config_base_path).default\n    let config_env = {}\n    if (fs.existsSync(config_env_path)) {\n        config_env = require(config_env_path).default\n    }\n    // bugfix Object.assign不支持深度拷贝问题\n    // global['__config__'] = Object.assign({}, config_base, config_env)\n    // global['__config__'] = _.assign({}, config_env, config_base)\n    global['__config__'] = xassign({}, config_base, config_env)\n    return _.get(global['__config__'], path, defaultValue)\n}\n\nasync function xconnect(callback: (connect: Connection) => Promise<any>, config = 'default') {\n    return new Promise(async (resolve, reject) => {\n        let cfg = {} as any\n        try {\n            cfg = xassign({}, xconfig('plugins.database.default', {}))\n            xassert(!_.isEmpty(cfg), ERR$PARAM, {config})\n            // 强制补上约定的实体存放路径定义位置（不允许配置是约定规范）\n            if (__dirname.includes('/node_modules/@bxjs/base/')) {\n                // 在应用目录下\n                cfg['entities'] = [\n                    path.join(__dirname, '../../../../app/plugins/database/entity/*.' + get_suffix_ts_or_js())\n                ]\n            } else {\n                // 在axjs库开发目录下\n                cfg['entities'] = [\n                    path.join(__dirname, '../app/plugins/database/entity/*.' + get_suffix_ts_or_js())\n                ]\n            }\n            // 获取连接池中的链接\n            const mng = getConnectionManager()\n            const name = cfg.name ? cfg.name : 'default'\n            if (!mng.has(name)) {\n                mng.create(cfg)\n            }\n            const db = mng.get(name)\n            if (global['__connection__'] === undefined) {\n                global['__connection__'] = {}\n            }\n            if (!db.isConnected) { // TODO 需要进行连接池的管理\n                global['__connection__'][name] = db.connect()\n            }\n            await global['__connection__'][name].then(async connection => {\n                xassert(db.isConnected)\n                const out = await callback(connection)\n                // await db.close() // typeorm没有进行连接池的管理不能进行销毁\n                resolve(out)\n            }).catch(async err => {\n                // await db.close()\n                xthrow(err, reject, {cfg})\n            })\n        } catch (err) {\n            xthrow(err, reject, {cfg})\n        }\n    })\n}\n\n// 创建XBaseEntity对象并且自动赋值前端请求的赋值数据\nfunction xnew<T extends BaseEntity>(TYPE: new () => T, param?: any, ...args): T {\n    // 泛型实现类似这个功能\n    // asset = new AlilangAsset()\n    // getRepository(AlilangAsset).merge(asset, param as any)\n    // AlilangAsset.merge(asset, param as any)\n    // return asset\n    let obj = new TYPE()\n    if (_.isEmpty(param)) {\n        return obj\n    }\n    let repo = getRepository<T>(TYPE)\n    repo.merge.apply(repo, [obj, param, ...args])\n    return obj\n}\n\n// 查询构造器易用性封装\nfunction xquery<T>(connect: Connection, TYPE: new () => T, alias?: string): SelectQueryBuilder<T> {\n    return connect.getRepository(TYPE).createQueryBuilder(alias)\n}\n\n// 分页查询获取总数以及原始记录数据\nasync function xcount<T>(sql: SelectQueryBuilder<T>, page: number, size: number): Promise<[any[] | null, number]> {\n    xassert(page >= 1)\n    const [count, rows] = await Promise.all([\n        sql.getCount(),\n        sql.offset((page - 1) * size).limit(size).getRawMany()\n    ])\n    return [rows, count]\n}\n\n\n// 路由参数的修饰符配置\n// TODO 更多接口相关参数的配置扩展，例如：是否支持JSONP\nfunction xroute(param: { name?: string, desc?: string, path?: string, auth?: boolean }) {\n    // 缺省值处理\n    param = xassign({name: '', desc: '', path: '', auth: true}, param)\n    return function (target: Function, propertyKey: string, descriptor: PropertyDescriptor) {\n        // TODO 注入到类实例定义中进行全局引用动态类的特性添加（trait功能的动态实现）\n        // 动态绑定路由类实例的上下文属性\n        target.prototype.context = () => {\n            return {\n                param: param, // 保存当前控制器用户定义参数信息\n                // 是否登录的鉴权方法统一框架层面上的处理实现，此处仅仅是通用接口的约束的定义。\n                auth: async () => {\n                    // 调用登录功能的前端接口实现，取到对应的实现方法。\n                    if (param && param.auth) {\n                        // 需要鉴权进行会话有效性进行合法性校验处理！！\n                        // 未认证错误抛出处理，前端单页应用接口报错逻辑处理正确错误提示跳转。\n                        const auth = xgot(YAuth)\n                        xassert(await auth.getLoginStatus(), ERR$UNAUTHORIZED)\n                    }\n                }\n            }\n        }\n    }\n}\n\n// 完全没有必要的多余定义，需要通过MOCK定义进行细节数据类型的显性定义处理逻辑验证。\n// // 基本数据类型的规范扩展定义，方便API接口的定义以及形参自动验证合法性，并且与数据库数据类型保持一致。\n// type INT = number   // 有符号整数\n// type UINT = number  // 无符号整数\n// type DECIMAL = number // 精确小数\n// type FLOAT = number // 单精度浮点数（不精确小数）\n// type DOUBLE = number// 双精度浮点数（不精确小数）\n// type BOOL = boolean\n// type STR = string\n// type DATE = string // 年月日 '2017-06-25'\n// type TIME = string // 时分秒 '00:00:00'\n// type DATETIME = string // 年月日时分秒 '2017-06-25 00:00:00'\n\n// 模拟数据模板定义使用教程 http://mockjs.com/0.1/#%E6%95%B0%E6%8D%AE%E5%8D%A0%E4%BD%8D%E7%AC%A6%E5%AE%9A%E4%B9%89%20DPD\nfunction xmock<T>(rules: T): any {\n    return mockjs.mock(rules)\n}\n\nfunction xrandom(name: string, data: any[]) {\n    mockjs.Random.extend({\n        [name]: (...args) => {\n            xassert(data.length > 0)\n            if (data.length == 1) return data[0]\n            let max = data.length - 1\n            let idx = xmock(`@int(0,${max})`)\n            return data[idx]\n        }\n    })\n}\n\n// 扩展一些预定义bxjs的基础随机方法或者覆盖一些mockjs中的方法\nmockjs.Random.extend({\n    // bxjs表定义的主键统一定义（约定系统中为字符串7-14字节长度算法）\n    id: (...args) => {\n        return shortid.generate()\n    },\n    // 中国手机号随机生成算法(约定系统中的手机号为字符串数据类型)\n    mobile: (...args) => {\n        const isps = [\n            134, 135, 136, 137, 138, 139, 147, 150, 151, 152, 157, 158, 159, 182, 183, 184, 187, 188, 178,\n            130, 131, 132, 145, 155, 156, 185, 186, 176,\n            133, 134, 153, 180, 181, 189, 177, 173,\n            176, 173, 177, 178, 170,\n            140, 141, 142, 143, 144, 146, 148, 149, 154]\n        let max = isps.length - 1\n        let idx = xmock(`@int(0,${max})`)\n        let num = xmock(`@int(100000000,199999999)`)\n        return (isps[idx] * 100000000 + num % 100000000) + ''\n    },\n    // 转换为缺省中文内容提示\n    paragraph: (...args) => {\n        switch (args.length) {\n            case 0:\n                return xmock('@cparagraph')\n            case 1:\n                return xmock(`@cparagraph(${args[0]})`)\n            case 2:\n                return xmock(`@cparagraph(${args[0]},${args[1]})`)\n            default:\n                xassert(false)\n        }\n\n    },\n    sentence: (...args) => {\n        switch (args.length) {\n            case 0:\n                return xmock('@csentence')\n            case 1:\n                return xmock(`@csentence(${args[0]})`)\n            case 2:\n                return xmock(`@csentence(${args[0]},${args[1]})`)\n            default:\n                xassert(false)\n        }\n\n    },\n    title: (...args) => {\n        switch (args.length) {\n            case 0:\n                return xmock('@ctitle')\n            case 1:\n                return xmock(`@ctitle(${args[0]})`)\n            case 2:\n                return xmock(`@ctitle(${args[0]},${args[1]})`)\n            default:\n                xassert(false)\n        }\n\n    },\n})\n\n// laravel风格JSON对象验证器封装，详细文档见 https://github.com/skaterdav85/validatorjs\nfunction xcheck<T>(param: T, rules: T, messages?: Object) {\n    let obj = new validatorjs(param, rules)\n    if (obj.fails()) {\n        xthrow(ERR$PARAM, obj.errors)\n    }\n}\n\n// 【IoC容器管理】应用层的插件实现类绑定到BXJS统一注册的标准插件的映射关系在全局容器实例中注册\nfunction xbind<T>(TYPE: new () => T) {\n    const o: any = new TYPE()\n    return xcontainer.bind<T>(o.id).to(require(`@app/plugins/${o.id}`).default)\n}\n\n// 【IoC容器管理】框架或应用依赖标准规范接口插件的类实例获取方法\nfunction xgot<T>(TYPE: new () => T) {\n    const o: any = new TYPE()\n    return xcontainer.get<T>(o.id)\n}\n\n// 同步系统命令调用执行\nasync function xcmd(...args: string[]): Promise<any> {\n    try {\n        const options: any = {}\n        options.cwd = options.cwd || process.env.__ctxPath || process.cwd();\n        xassert(_.isArray(args) && args.length > 0)\n        const cmd = args.shift()\n        const ret = cross_spawn.sync(cmd, args, xassign({stdio: 'inherit'}, options))\n        xassert(ret.status === 0, ERR$UNKNOWN, ret)\n        return ret\n    } catch (err) {\n        await xwarn(err)\n        xthrow(err)\n    }\n}\n\n// 对于数组嵌套回调函数的nodejs异步处理方法的统一封装\nasync function xmap<T>(values: T[], callack: (v: T) => Promise<any>): Promise<any[]> {\n    xassert(_.isArray(values) && _.isFunction(callack))\n    return Promise.all(values.map(callack))\n}\n\n// Refer to document:  https://help.aliyun.com/document_detail/62670.html\n// 获取ACM配置信息接口的统一封装\nasync function xacm(group: string, id: string) {\n    const cfg = xconfig(`plugins.acm`)\n    xassert(group && cfg && cfg[group], ERR$CONFIG, {cfg})\n    const acm = new ACMClient(cfg[group])\n    return new Promise(async (resolve, reject) => {\n        try {\n            co(function* () {\n                try {\n                    group = group + ':' + global['__env__'] // 补上环境后缀支持各种开发环境的个性化配置\n                    const content = yield acm.getConfig(id, group)\n                    xassert(content, ERR$CONFIG, {id, group})\n                    resolve(content)\n                } catch (err) {\n                    xthrow(err, reject, {id, group, cfg: cfg[group]})\n                }\n            });\n        } catch (err) {\n            xthrow(err, reject, {id, group, cfg: cfg[group]})\n        }\n    })\n}\n\nif (!global['__env__']) {\n    global['__env__'] = 'local' // local,daily,pre,gray,prod 在统一入口处自动识别配置(目前暂不支持gray配置尚未开发无法自动识别)\n}\nglobal['__config__'] = undefined\nglobal['__session__'] = {}\nglobal['__cache__'] = {}\nglobal['__user__'] = {}\nglobal['__user_agent__'] = undefined\nglobal['__client_ip__'] = undefined\nglobal['__redirect_url__'] = undefined\nglobal['__request_cookies__'] = {}\nglobal['__respond_cookies__'] = {}\n\nglobal['xconnect'] = xconnect\nglobal['xnew'] = xnew\nglobal['xquery'] = xquery\nglobal['xcount'] = xcount\nglobal['xassign'] = xassign\nglobal['xconfig'] = xconfig\nglobal['xthrow'] = xthrow\nglobal['xassert'] = xassert\nglobal['xerror'] = xerror\nglobal['xroot'] = xroot\nglobal['xstack'] = xstack\nglobal['xwarn'] = xwarn\nglobal['xlog'] = xlog\nglobal['xpost'] = xpost\nglobal['xget'] = xget\nglobal['xsleep'] = xsleep\nglobal['xredirect'] = xredirect\nglobal['xcookie'] = xcookie\nglobal['xismobile'] = xismobile\nglobal['xsession'] = xsession\nglobal['xuser'] = xuser\nglobal['xcache'] = xcache\nglobal['xroute'] = xroute\nglobal['xmock'] = xmock\nglobal['xrandom'] = xrandom\nglobal['xcheck'] = xcheck\nglobal['xcontainer'] = new Container() // 全局单实例容器初始化\nglobal['xbind'] = xbind\nglobal['xgot'] = xgot\nglobal['YAuth'] = $$.YAuth // 全局声明认证插件规范抽象类\nglobal['xcmd'] = xcmd\nglobal['xmap'] = xmap\nglobal['xacm'] = xacm\n"]} |
\ | No newline at end of file |