zan-proxy
Version:
271 lines (237 loc) • 6.71 kB
text/typescript
import { promisify } from 'es6-promisify';
import EventEmitter from 'events';
import fs from 'fs';
import { forEach } from 'lodash';
import path from 'path';
import rimraf from 'rimraf';
import { Service } from 'typedi';
import { AppInfoService } from './appInfo';
const fsWriteFile = promisify(fs.writeFile);
const fsReadFile = promisify(fs.readFile);
const logCountPerUser = 500;
export class HttpTrafficService extends EventEmitter {
private cache: object;
private userRequestPointer: object;
private userMonitorCount: object;
private trafficDir: string;
private filterMap: object;
private stopRecord: object;
constructor(appInfoService: AppInfoService) {
super();
// http请求缓存数据 userId - > [{record},{record},{record}]
this.cache = {};
// 用户的请求id 一个用户可以关联多个请求设备,用户的请求分配唯一个一个请求id
this.userRequestPointer = {};
// 记录用户的监视窗数量
this.userMonitorCount = {};
const proxyDataDir = appInfoService.getProxyDataDir();
// 监控数据缓存目录
this.trafficDir = path.join(proxyDataDir, 'traffic');
this.filterMap = {};
this.stopRecord = {};
// 创建定时任务,推送日志记录
setInterval(() => {
this.sendCachedData();
}, 2000);
try {
rimraf.sync(this.trafficDir);
fs.mkdirSync(this.trafficDir);
} catch (e) {
console.error(e);
}
}
public getFilter(userId) {
const filters = this.filterMap[userId] || '';
return filters;
}
public setFilter(userId, filter) {
this.filterMap[userId] = filter;
this.emit('filter', userId, filter);
}
public getStatus(userId) {
return {
overflow: this.userRequestPointer[userId] > logCountPerUser,
stopRecord: this.stopRecord[userId] || false,
};
}
public setStopRecord(userId, stop) {
this.stopRecord[userId] = stop;
// 发送通知
this.emit('state-change', userId, this.getStatus(userId));
}
public clear(userId) {
this.userRequestPointer[userId] = 0;
// 发送通知
this.emit('clear', userId);
}
// 将缓存数据发送给用户
public sendCachedData() {
forEach(this.cache, (rows, userId) => {
this.emit('traffic', userId, rows);
});
this.cache = {};
}
// 为请求分配id
public 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;
}
public resetRequestId(userId) {
this.userRequestPointer[userId] = 0;
}
// 获取监控窗口的数量,没有监控窗口 则不做记录
public hasMonitor(userId) {
const cnt = this.userMonitorCount[userId] || 0;
return cnt > 0;
}
// 用户监控窗数加1
public incMonitor(userId) {
let cnt = this.userMonitorCount[userId] || 0;
if (cnt === 0) {
this.resetRequestId(userId);
}
cnt++;
this.userMonitorCount[userId] = cnt;
}
// 用户监控窗数减一
public decMonitor(userId) {
let cnt = this.userMonitorCount[userId] || 0;
cnt--;
this.userMonitorCount[userId] = cnt;
}
// 记录请求
public async requestBegin({
id,
userId,
clientIp,
method,
httpVersion,
urlObj,
headers,
}) {
const queue = this.cache[userId] || [];
// 原始请求信息
queue.push({
id,
originRequest: Object.assign(
{
clientIp,
headers,
httpVersion,
method,
},
urlObj,
),
});
this.cache[userId] = queue;
}
// 记录请求body
public async actualRequest({ userId, id, requestData, originBody }) {
// 将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);
await fsWriteFile(bodyPath, body, { encoding: 'utf-8' });
}
if (originBody) {
const bodyPath = this.getOriginRequestBodyPath(userId, id);
await fsWriteFile(bodyPath, originBody, { encoding: 'utf-8' });
}
}
// 记录响应
public async serverReturn({ userId, id, toClientResponse }) {
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);
await fsWriteFile(bodyPath, body, { encoding: 'utf-8' });
}
}
/**
* 获取请求的请求内容
* @param userId
* @param requestId
*/
public async getRequestBody(userId, requestId) {
const saveRequestPath = this.getRequestBodyPath(userId, requestId);
return await fsReadFile(saveRequestPath, { encoding: 'utf-8' });
}
/**
* 获取请求的请求内容
* @param userId
* @param requestId
*/
public async getResponseBody(userId, requestId) {
const saveResponsePath = this.getResponseBodyPath(userId, requestId);
return await fsReadFile(saveResponsePath, { encoding: 'utf-8' });
}
// 获取请求记录path
private getRequestBodyPath(userId, requestId) {
return path.join(this.trafficDir, userId + '_' + requestId + '_req_body');
}
private getOriginRequestBodyPath(userId, requestId) {
return path.join(
this.trafficDir,
userId + '_' + requestId + '_req_body_origin',
);
}
// 获取响应记录path
private getResponseBodyPath(userId, requestId) {
return path.join(this.trafficDir, userId + '_' + requestId + '_res_body');
}
}