zan-proxy
Version:
244 lines • 9.56 kB
JavaScript
"use strict";
var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
return c > 3 && r && Object.defineProperty(target, key, r), r;
};
var __metadata = (this && this.__metadata) || function (k, v) {
if (typeof Reflect === "object" && typeof Reflect.metadata === "function") return Reflect.metadata(k, v);
};
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
return new (P || (P = Promise))(function (resolve, reject) {
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
function step(result) { result.done ? resolve(result.value) : new P(function (resolve) { resolve(result.value); }).then(fulfilled, rejected); }
step((generator = generator.apply(thisArg, _arguments || [])).next());
});
};
var __importDefault = (this && this.__importDefault) || function (mod) {
return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
const es6_promisify_1 = require("es6-promisify");
const events_1 = __importDefault(require("events"));
const fs_1 = __importDefault(require("fs"));
const lodash_1 = require("lodash");
const path_1 = __importDefault(require("path"));
const rimraf_1 = __importDefault(require("rimraf"));
const typedi_1 = require("typedi");
const appInfo_1 = require("./appInfo");
const fsWriteFile = es6_promisify_1.promisify(fs_1.default.writeFile);
const fsReadFile = es6_promisify_1.promisify(fs_1.default.readFile);
const logCountPerUser = 500;
let HttpTrafficService = class HttpTrafficService extends events_1.default {
constructor(appInfoService) {
super();
// http请求缓存数据 userId - > [{record},{record},{record}]
this.cache = {};
// 用户的请求id 一个用户可以关联多个请求设备,用户的请求分配唯一个一个请求id
this.userRequestPointer = {};
// 记录用户的监视窗数量
this.userMonitorCount = {};
const proxyDataDir = appInfoService.getProxyDataDir();
// 监控数据缓存目录
this.trafficDir = path_1.default.join(proxyDataDir, 'traffic');
this.filterMap = {};
this.stopRecord = {};
// 创建定时任务,推送日志记录
setInterval(() => {
this.sendCachedData();
}, 2000);
try {
rimraf_1.default.sync(this.trafficDir);
fs_1.default.mkdirSync(this.trafficDir);
}
catch (e) {
console.error(e);
}
}
getFilter(userId) {
const filters = this.filterMap[userId] || '';
return filters;
}
setFilter(userId, filter) {
this.filterMap[userId] = filter;
this.emit('filter', userId, filter);
}
getStatus(userId) {
return {
overflow: this.userRequestPointer[userId] > logCountPerUser,
stopRecord: this.stopRecord[userId] || false,
};
}
setStopRecord(userId, stop) {
this.stopRecord[userId] = stop;
// 发送通知
this.emit('state-change', userId, this.getStatus(userId));
}
clear(userId) {
this.userRequestPointer[userId] = 0;
// 发送通知
this.emit('clear', userId);
}
// 将缓存数据发送给用户
sendCachedData() {
lodash_1.forEach(this.cache, (rows, userId) => {
this.emit('traffic', userId, rows);
});
this.cache = {};
}
// 为请求分配id
getRequestId(userId, urlObj) {
// 处于停止记录状态 则不返回id
if (this.stopRecord[userId]) {
return -1;
}
// 获取当前ip
let id = this.userRequestPointer[userId] || 0;
// 超过500个请求则不再记录
if (id > logCountPerUser) {
return -1;
}
const filter = this.getFilter(userId);
const { href } = urlObj;
if (href.includes(filter)) {
id++;
this.userRequestPointer[userId] = id;
if (id > logCountPerUser) {
const state = this.getStatus(userId);
// 向监控窗推送通知
this.emit('state-change', userId, state);
}
return id;
}
return -1;
}
resetRequestId(userId) {
this.userRequestPointer[userId] = 0;
}
// 获取监控窗口的数量,没有监控窗口 则不做记录
hasMonitor(userId) {
const cnt = this.userMonitorCount[userId] || 0;
return cnt > 0;
}
// 用户监控窗数加1
incMonitor(userId) {
let cnt = this.userMonitorCount[userId] || 0;
if (cnt === 0) {
this.resetRequestId(userId);
}
cnt++;
this.userMonitorCount[userId] = cnt;
}
// 用户监控窗数减一
decMonitor(userId) {
let cnt = this.userMonitorCount[userId] || 0;
cnt--;
this.userMonitorCount[userId] = cnt;
}
// 记录请求
requestBegin({ id, userId, clientIp, method, httpVersion, urlObj, headers, }) {
return __awaiter(this, void 0, void 0, function* () {
const queue = this.cache[userId] || [];
// 原始请求信息
queue.push({
id,
originRequest: Object.assign({
clientIp,
headers,
httpVersion,
method,
}, urlObj),
});
this.cache[userId] = queue;
});
}
// 记录请求body
actualRequest({ userId, id, requestData, originBody }) {
return __awaiter(this, void 0, void 0, function* () {
// 将body写文件
const body = requestData.body;
delete requestData.body;
const queue = this.cache[userId] || [];
queue.push({
id,
requestData,
});
this.cache[userId] = queue;
if (body) {
const bodyPath = this.getRequestBodyPath(userId, id);
yield fsWriteFile(bodyPath, body, { encoding: 'utf-8' });
}
if (originBody) {
const bodyPath = this.getOriginRequestBodyPath(userId, id);
yield fsWriteFile(bodyPath, originBody, { encoding: 'utf-8' });
}
});
}
// 记录响应
serverReturn({ userId, id, toClientResponse }) {
return __awaiter(this, void 0, void 0, function* () {
const queue = this.cache[userId] || [];
const { statusCode, headers, receiveRequestTime, dnsResolveBeginTime, remoteRequestBeginTime, remoteResponseStartTime, remoteResponseEndTime, requestEndTime, remoteIp, body, } = toClientResponse;
queue.push({
id,
response: {
dnsResolveBeginTime,
headers,
receiveRequestTime,
remoteIp,
remoteRequestBeginTime,
remoteResponseEndTime,
remoteResponseStartTime,
requestEndTime,
statusCode,
},
});
this.cache[userId] = queue;
if (body) {
const bodyPath = this.getResponseBodyPath(userId, id);
yield fsWriteFile(bodyPath, body, { encoding: 'utf-8' });
}
});
}
/**
* 获取请求的请求内容
* @param userId
* @param requestId
*/
getRequestBody(userId, requestId) {
return __awaiter(this, void 0, void 0, function* () {
const saveRequestPath = this.getRequestBodyPath(userId, requestId);
return yield fsReadFile(saveRequestPath, { encoding: 'utf-8' });
});
}
/**
* 获取请求的请求内容
* @param userId
* @param requestId
*/
getResponseBody(userId, requestId) {
return __awaiter(this, void 0, void 0, function* () {
const saveResponsePath = this.getResponseBodyPath(userId, requestId);
return yield fsReadFile(saveResponsePath, { encoding: 'utf-8' });
});
}
// 获取请求记录path
getRequestBodyPath(userId, requestId) {
return path_1.default.join(this.trafficDir, userId + '_' + requestId + '_req_body');
}
getOriginRequestBodyPath(userId, requestId) {
return path_1.default.join(this.trafficDir, userId + '_' + requestId + '_req_body_origin');
}
// 获取响应记录path
getResponseBodyPath(userId, requestId) {
return path_1.default.join(this.trafficDir, userId + '_' + requestId + '_res_body');
}
};
HttpTrafficService = __decorate([
typedi_1.Service(),
__metadata("design:paramtypes", [appInfo_1.AppInfoService])
], HttpTrafficService);
exports.HttpTrafficService = HttpTrafficService;
//# sourceMappingURL=httpTraffic.js.map