perf: 优化站点ip检查

pull/409/head
xiaojunnuo 2025-05-28 13:57:31 +08:00
parent 3a147141b1
commit a463711b03
11 changed files with 219 additions and 139 deletions

View File

@ -55,4 +55,15 @@ export const siteInfoApi = {
method: "post", method: "post",
}); });
}, },
async IpCheckChange(id: number, ipCheck: boolean) {
return await request({
url: apiPrefix + "/ipCheckChange",
method: "post",
data: {
id,
ipCheck,
},
});
},
}; };

View File

@ -1,9 +1,9 @@
// @ts-ignore // @ts-ignore
import { useI18n } from "vue-i18n"; import { useI18n } from "vue-i18n";
import { AddReq, CreateCrudOptionsProps, CreateCrudOptionsRet, DelReq, dict, EditReq, UserPageQuery, UserPageRes } from "@fast-crud/fast-crud"; import { AddReq, ColumnCompositionProps, compute, CreateCrudOptionsProps, CreateCrudOptionsRet, DelReq, dict, EditReq, UserPageQuery, UserPageRes } from "@fast-crud/fast-crud";
import { siteInfoApi } from "./api"; import { siteInfoApi } from "./api";
import dayjs from "dayjs"; import dayjs from "dayjs";
import { notification } from "ant-design-vue"; import { Modal, notification } from "ant-design-vue";
import { useSettingStore } from "/@/store/settings"; import { useSettingStore } from "/@/store/settings";
import { mySuiteApi } from "/@/views/certd/suite/mine/api"; import { mySuiteApi } from "/@/views/certd/suite/mine/api";
import { mitter } from "/@/utils/util.mitt"; import { mitter } from "/@/utils/util.mitt";
@ -119,10 +119,13 @@ export default function ({ crudExpose, context }: CreateCrudOptionsProps): Creat
}); });
}, },
}, },
ipMonitor: { ipCheck: {
order: 0, order: 10,
type: "link", type: "link",
text: null, text: null,
show: compute(({ row }) => {
return row.ipCheck === true;
}),
tooltip: { tooltip: {
title: "IP管理", title: "IP管理",
}, },
@ -327,26 +330,46 @@ export default function ({ crudExpose, context }: CreateCrudOptionsProps): Creat
}, },
}, },
ipCheck: { ipCheck: {
title: "检查IP", title: "开启IP检查",
search: {
show: false,
},
type: "dict-switch", type: "dict-switch",
dict: dict({ dict: dict({
data: [ data: [
{ label: "启用", value: false, color: "green" }, { label: "启用", value: false, color: "green" },
{ label: "禁用", value: true, color: "red" }, { label: "禁用", value: true, color: "gray" },
], ],
}), }),
form: { form: {
value: false, value: false,
rules: [{ required: true, message: "请选择" }],
}, },
column: { column: {
width: 100,
sorter: true,
align: "center", align: "center",
width: 100,
conditionalRender: false,
component: {
name: "a-switch",
vModel: "checked",
on: {
change({ row, $event }) {
Modal.confirm({
title: "提示",
content: `确定${$event ? "开启" : "关闭"}IP检查`,
onOk: async () => {
await api.IpCheckChange(row.id, $event);
await crudExpose.doRefresh();
if ($event) {
openSiteIpMonitorDialog({ siteId: row.id });
}
},
onCancel: async () => {
await crudExpose.doRefresh();
},
});
},
},
},
}, },
}, } as ColumnCompositionProps,
ipCount: { ipCount: {
title: "IP数量", title: "IP数量",
search: { search: {

View File

@ -12,12 +12,15 @@ export default function ({ crudExpose, context }: CreateCrudOptionsProps): Creat
const { crudBinding } = crudExpose; const { crudBinding } = crudExpose;
const pageRequest = async (query: UserPageQuery): Promise<UserPageRes> => { const pageRequest = async (query: UserPageQuery): Promise<UserPageRes> => {
if (!query.query) {
query.query = {};
}
query.query.siteId = context.props.siteId;
return await api.GetList(query); return await api.GetList(query);
}; };
const editRequest = async (req: EditReq) => { const editRequest = async (req: EditReq) => {
const { form, row } = req; const { form, row } = req;
form.id = row.id; form.id = row.id;
form.siteId = context.props.siteId;
const res = await api.UpdateObj(form); const res = await api.UpdateObj(form);
return res; return res;
}; };
@ -73,6 +76,7 @@ export default function ({ crudExpose, context }: CreateCrudOptionsProps): Creat
}, },
load: { load: {
text: "同步IP", text: "同步IP",
type: "primary",
async click() { async click() {
Modal.confirm({ Modal.confirm({
title: "同步IP", title: "同步IP",
@ -87,6 +91,23 @@ export default function ({ crudExpose, context }: CreateCrudOptionsProps): Creat
}); });
}, },
}, },
checkAll: {
text: "检查全部",
type: "primary",
click: () => {
Modal.confirm({
title: "确认",
content: "确认触发检查全部IP站点的证书吗?",
onOk: async () => {
await siteIpApi.CheckAll(context.props.siteId);
notification.success({
message: "检查任务已提交",
description: "请稍后刷新页面查看结果",
});
},
});
},
},
}, },
}, },
rowHandle: { rowHandle: {
@ -105,7 +126,7 @@ export default function ({ crudExpose, context }: CreateCrudOptionsProps): Creat
await api.DoCheck(row.id); await api.DoCheck(row.id);
await crudExpose.doRefresh(); await crudExpose.doRefresh();
notification.success({ notification.success({
message: "检查完成", message: "检查任务已提交",
}); });
}, },
}, },
@ -152,7 +173,7 @@ export default function ({ crudExpose, context }: CreateCrudOptionsProps): Creat
column: { column: {
width: 200, width: 200,
sorter: true, sorter: true,
show: true, show: false,
cellRender({ value }) { cellRender({ value }) {
return ( return (
<a-tooltip title={value} placement="left"> <a-tooltip title={value} placement="left">
@ -173,6 +194,7 @@ export default function ({ crudExpose, context }: CreateCrudOptionsProps): Creat
}, },
column: { column: {
width: 200, width: 200,
show: false,
sorter: true, sorter: true,
cellRender({ value }) { cellRender({ value }) {
return <a-tooltip title={value}>{value}</a-tooltip>; return <a-tooltip title={value}>{value}</a-tooltip>;
@ -224,6 +246,29 @@ export default function ({ crudExpose, context }: CreateCrudOptionsProps): Creat
}, },
}, },
}, },
checkStatus: {
title: "检查状态",
search: {
show: false,
},
type: "dict-select",
dict: checkStatusDict,
form: {
show: false,
},
column: {
width: 100,
align: "center",
sorter: true,
cellRender({ value, row, key }) {
return (
<a-tooltip title={row.error}>
<fs-values-format v-model={value} dict={checkStatusDict}></fs-values-format>
</a-tooltip>
);
},
},
},
lastCheckTime: { lastCheckTime: {
title: "上次检查时间", title: "上次检查时间",
search: { search: {
@ -280,29 +325,6 @@ export default function ({ crudExpose, context }: CreateCrudOptionsProps): Creat
align: "center", align: "center",
}, },
}, },
checkStatus: {
title: "检查状态",
search: {
show: false,
},
type: "dict-select",
dict: checkStatusDict,
form: {
show: false,
},
column: {
width: 100,
align: "center",
sorter: true,
cellRender({ value, row, key }) {
return (
<a-tooltip title={row.error}>
<fs-values-format v-model={value} dict={checkStatusDict}></fs-values-format>
</a-tooltip>
);
},
},
},
remark: { remark: {
title: "备注", title: "备注",
search: { search: {

View File

@ -5,7 +5,7 @@
</template> </template>
<script lang="ts" setup> <script lang="ts" setup>
import { onActivated, onMounted } from "vue"; import { onActivated, onMounted, ref, Ref } from "vue";
import { useFs } from "@fast-crud/fast-crud"; import { useFs } from "@fast-crud/fast-crud";
import createCrudOptions from "./crud"; import createCrudOptions from "./crud";
import { siteIpApi } from "./api"; import { siteIpApi } from "./api";
@ -22,19 +22,11 @@ const { crudBinding, crudRef, crudExpose } = useFs({
props, props,
}, },
}); });
function checkAll() {
Modal.confirm({ const siteInfoRef: Ref<any> = ref({});
title: "确认", onMounted(async () => {
content: "确认触发检查全部站点证书吗?", siteInfoRef.value = await siteIpApi.GetObj(props.siteId);
onOk: async () => { });
await siteIpApi.CheckAll();
notification.success({
message: "检查任务已提交",
description: "请稍后刷新页面查看结果",
});
},
});
}
// //
onMounted(() => { onMounted(() => {

View File

@ -15,13 +15,13 @@ export function notificationProvide(api: any) {
export function getCommonColumnDefine(crudExpose: any, typeRef: any, api: any) { export function getCommonColumnDefine(crudExpose: any, typeRef: any, api: any) {
const notificationTypeDictRef = dict({ const notificationTypeDictRef = dict({
url: "/pi/notification/getTypeDict" url: "/pi/notification/getTypeDict",
}); });
const defaultPluginConfig = { const defaultPluginConfig = {
component: { component: {
name: "a-input", name: "a-input",
vModel: "value" vModel: "value",
} },
}; };
function buildDefineFields(define: any, form: any, mode: string) { function buildDefineFields(define: any, form: any, mode: string) {
@ -38,7 +38,7 @@ export function getCommonColumnDefine(crudExpose: any, typeRef: any, api: any) {
const key = "body." + mapKey; const key = "body." + mapKey;
const field = { const field = {
...value, ...value,
key key,
}; };
const column = merge({ title: key }, defaultPluginConfig, field); const column = merge({ title: key }, defaultPluginConfig, field);
//eval //eval
@ -69,29 +69,29 @@ export function getCommonColumnDefine(crudExpose: any, typeRef: any, api: any) {
key: "id", key: "id",
type: "number", type: "number",
column: { column: {
width: 100 width: 100,
}, },
form: { form: {
show: false show: false,
} },
}, },
type: { type: {
title: "通知类型", title: "通知类型",
type: "dict-select", type: "dict-select",
dict: notificationTypeDictRef, dict: notificationTypeDictRef,
search: { search: {
show: false show: false,
}, },
column: { column: {
width: 200, width: 200,
component: { component: {
color: "auto" color: "auto",
} },
}, },
editForm: { editForm: {
component: { component: {
disabled: false disabled: false,
} },
}, },
form: { form: {
component: { component: {
@ -108,7 +108,7 @@ export function getCommonColumnDefine(crudExpose: any, typeRef: any, api: any) {
{item.needPlus && <fs-icon icon={"mingcute:vip-1-line"} className={"color-plus"}></fs-icon>} {item.needPlus && <fs-icon icon={"mingcute:vip-1-line"} className={"color-plus"}></fs-icon>}
</span> </span>
); );
} },
}, },
rules: [{ required: true, message: "请选择通知类型" }], rules: [{ required: true, message: "请选择通知类型" }],
valueChange: { valueChange: {
@ -133,7 +133,7 @@ export function getCommonColumnDefine(crudExpose: any, typeRef: any, api: any) {
form.name = define.title; form.name = define.title;
} }
buildDefineFields(define, form, mode); buildDefineFields(define, form, mode);
} },
}, },
helper: computed(() => { helper: computed(() => {
const define = currentDefine.value; const define = currentDefine.value;
@ -141,22 +141,22 @@ export function getCommonColumnDefine(crudExpose: any, typeRef: any, api: any) {
return ""; return "";
} }
return define.desc; return define.desc;
}) }),
} },
} as ColumnCompositionProps, } as ColumnCompositionProps,
name: { name: {
title: "通知名称", title: "通知名称",
search: { search: {
show: true show: true,
}, },
type: ["text"], type: ["text"],
form: { form: {
rules: [{ required: true, message: "请填写名称" }], rules: [{ required: true, message: "请填写名称" }],
helper: "随便填,当多个相同类型的通知时,便于区分" helper: "随便填,当多个相同类型的通知时,便于区分",
}, },
column: { column: {
width: 200 width: 200,
} },
}, },
isDefault: { isDefault: {
title: "是否默认", title: "是否默认",
@ -164,13 +164,13 @@ export function getCommonColumnDefine(crudExpose: any, typeRef: any, api: any) {
dict: dict({ dict: dict({
data: [ data: [
{ label: "是", value: true, color: "success" }, { label: "是", value: true, color: "success" },
{ label: "否", value: false, color: "default" } { label: "否", value: false, color: "default" },
] ],
}), }),
form: { form: {
value: false, value: false,
rules: [{ required: true, message: "请选择是否默认" }], rules: [{ required: true, message: "请选择是否默认" }],
order: 999 order: 999,
}, },
column: { column: {
align: "center", align: "center",
@ -192,12 +192,12 @@ export function getCommonColumnDefine(crudExpose: any, typeRef: any, api: any) {
}, },
onCancel: async () => { onCancel: async () => {
await crudExpose.doRefresh(); await crudExpose.doRefresh();
} },
}); });
} },
} },
} },
} },
} as ColumnCompositionProps, } as ColumnCompositionProps,
test: { test: {
title: "测试", title: "测试",
@ -207,16 +207,16 @@ export function getCommonColumnDefine(crudExpose: any, typeRef: any, api: any) {
}), }),
component: { component: {
name: "api-test", name: "api-test",
action: "TestRequest" action: "TestRequest",
}, },
order: 990, order: 990,
col: { col: {
span: 24 span: 24,
} },
}, },
column: { column: {
show: false show: false,
} },
}, },
setting: { setting: {
column: { show: false }, column: { show: false },
@ -235,8 +235,8 @@ export function getCommonColumnDefine(crudExpose: any, typeRef: any, api: any) {
valueResolve({ form }) { valueResolve({ form }) {
const setting = form.body; const setting = form.body;
form.setting = JSON.stringify(setting); form.setting = JSON.stringify(setting);
} },
} },
} as ColumnCompositionProps } as ColumnCompositionProps,
}; };
} }

View File

@ -32,23 +32,23 @@ export default function ({ crudExpose, context }: CreateCrudOptionsProps): Creat
pageRequest, pageRequest,
addRequest, addRequest,
editRequest, editRequest,
delRequest delRequest,
}, },
form: { form: {
labelCol: { labelCol: {
//固定label宽度 //固定label宽度
span: null, span: null,
style: { style: {
width: "145px" width: "145px",
} },
} },
}, },
rowHandle: { rowHandle: {
width: 200 width: 200,
}, },
columns: { columns: {
...commonColumnsDefine ...commonColumnsDefine,
} },
} },
}; };
} }

View File

@ -4,6 +4,7 @@ import { AuthService } from "../../../modules/sys/authority/service/auth-service
import { SiteInfoService } from "../../../modules/monitor/service/site-info-service.js"; import { SiteInfoService } from "../../../modules/monitor/service/site-info-service.js";
import { UserSiteMonitorSetting } from "../../../modules/mine/service/models.js"; import { UserSiteMonitorSetting } from "../../../modules/mine/service/models.js";
import { merge } from "lodash-es"; import { merge } from "lodash-es";
import {SiteIpService} from "../../../modules/monitor/service/site-ip-service.js";
/** /**
*/ */
@ -14,6 +15,8 @@ export class SiteInfoController extends CrudController<SiteInfoService> {
service: SiteInfoService; service: SiteInfoService;
@Inject() @Inject()
authService: AuthService; authService: AuthService;
@Inject()
siteIpService: SiteIpService;
getService(): SiteInfoService { getService(): SiteInfoService {
return this.service; return this.service;
@ -96,7 +99,16 @@ export class SiteInfoController extends CrudController<SiteInfoService> {
await this.service.checkAllByUsers(userId); await this.service.checkAllByUsers(userId);
return this.ok(); return this.ok();
} }
@Post('/ipCheckChange', { summary: Constants.per.authOnly })
async ipCheckChange(@Body(ALL) bean: any) {
const userId = this.getUserId();
await this.service.checkUserId(bean.id, userId)
await this.service.ipCheckChange({
id: bean.id,
ipCheck: bean.ipCheck
});
return this.ok();
}
@Post("/setting/get", { summary: Constants.per.authOnly }) @Post("/setting/get", { summary: Constants.per.authOnly })

View File

@ -44,7 +44,9 @@ export class SiteInfoController extends CrudController<SiteIpService> {
bean.userId = this.getUserId(); bean.userId = this.getUserId();
bean.from = "manual" bean.from = "manual"
const res = await this.service.add(bean); const res = await this.service.add(bean);
this.service.check(res.id); const siteEntity = await this.siteInfoService.info(bean.siteId);
const {domain, httpsPort} = siteEntity;
this.service.check(res.id,domain, httpsPort);
return this.ok(res); return this.ok(res);
} }
@ -53,7 +55,9 @@ export class SiteInfoController extends CrudController<SiteIpService> {
await this.service.checkUserId(bean.id, this.getUserId()); await this.service.checkUserId(bean.id, this.getUserId());
delete bean.userId; delete bean.userId;
await this.service.update(bean); await this.service.update(bean);
this.service.check(bean.id); const siteEntity = await this.siteInfoService.info(bean.siteId);
const {domain, httpsPort} = siteEntity;
this.service.check(bean.id,domain, httpsPort);
return this.ok(); return this.ok();
} }
@Post('/info', { summary: Constants.per.authOnly }) @Post('/info', { summary: Constants.per.authOnly })
@ -71,7 +75,11 @@ export class SiteInfoController extends CrudController<SiteIpService> {
@Post('/check', { summary: Constants.per.authOnly }) @Post('/check', { summary: Constants.per.authOnly })
async check(@Body('id') id: number) { async check(@Body('id') id: number) {
await this.service.checkUserId(id, this.getUserId()); await this.service.checkUserId(id, this.getUserId());
this.service.check(id); const entity = await this.service.info(id);
const siteEntity = await this.siteInfoService.info(entity.siteId);
const domain = siteEntity.domain;
const port = siteEntity.httpsPort;
this.service.check(id,domain,port);
return this.ok(); return this.ok();
} }
@ -79,7 +87,8 @@ export class SiteInfoController extends CrudController<SiteIpService> {
async checkAll(@Body('siteId') siteId: number) { async checkAll(@Body('siteId') siteId: number) {
const userId = this.getUserId(); const userId = this.getUserId();
await this.siteInfoService.checkUserId(siteId, userId); await this.siteInfoService.checkUserId(siteId, userId);
await this.service.checkAll(siteId); const siteEntity = await this.siteInfoService.info(siteId);
await this.service.checkAll(siteEntity);
return this.ok(); return this.ok();
} }
@ -94,4 +103,5 @@ export class SiteInfoController extends CrudController<SiteIpService> {
return this.ok(); return this.ok();
} }
} }

View File

@ -12,6 +12,7 @@ import { isComm, isPlus } from '@certd/plus-core';
import { UserSuiteService } from '@certd/commercial-core'; import { UserSuiteService } from '@certd/commercial-core';
import { UserSettingsService } from "../../mine/service/user-settings-service.js"; import { UserSettingsService } from "../../mine/service/user-settings-service.js";
import { UserSiteMonitorSetting } from "../../mine/service/models.js"; import { UserSiteMonitorSetting } from "../../mine/service/models.js";
import {SiteIpService} from "./site-ip-service.js";
@Provide() @Provide()
@Scope(ScopeEnum.Request, { allowDowngrade: true }) @Scope(ScopeEnum.Request, { allowDowngrade: true })
@ -31,6 +32,8 @@ export class SiteInfoService extends BaseService<SiteInfoEntity> {
@Inject() @Inject()
userSettingsService: UserSettingsService; userSettingsService: UserSettingsService;
@Inject()
siteIpService: SiteIpService;
//@ts-ignore //@ts-ignore
getRepository() { getRepository() {
@ -128,6 +131,13 @@ export class SiteInfoService extends BaseService<SiteInfoEntity> {
}; };
await this.update(updateData); await this.update(updateData);
//检查ip
if( site.ipCheck){
await this.siteIpService.checkAll(site)
}
if (!notify) { if (!notify) {
return; return;
} }
@ -156,7 +166,7 @@ export class SiteInfoService extends BaseService<SiteInfoEntity> {
} }
/** /**
* *
* @param id * @param id
* @param notify * @param notify
* @param retryTimes * @param retryTimes
@ -250,4 +260,16 @@ export class SiteInfoService extends BaseService<SiteInfoEntity> {
async saveSetting(userId: number, bean: UserSiteMonitorSetting) { async saveSetting(userId: number, bean: UserSiteMonitorSetting) {
await this.userSettingsService.saveSetting(userId, bean); await this.userSettingsService.saveSetting(userId, bean);
} }
async ipCheckChange(req: {id: any; ipCheck: any}) {
await this.update({
id: req.id,
ipCheck: req.ipCheck,
});
if(req.ipCheck){
const site = await this.info(req.id);
await this.siteIpService.sync(site)
}
}
} }

View File

@ -1,18 +1,17 @@
import { Inject, Provide, Scope, ScopeEnum } from "@midwayjs/core"; import {Inject, Provide, Scope, ScopeEnum} from "@midwayjs/core";
import { BaseService, SysSettingsService } from "@certd/lib-server"; import {BaseService, SysSettingsService} from "@certd/lib-server";
import { InjectEntityModel } from "@midwayjs/typeorm"; import {InjectEntityModel} from "@midwayjs/typeorm";
import { Repository } from "typeorm"; import {Repository} from "typeorm";
import { SiteInfoEntity } from "../entity/site-info.js"; import {SiteInfoEntity} from "../entity/site-info.js";
import { NotificationService } from "../../pipeline/service/notification-service.js"; import {NotificationService} from "../../pipeline/service/notification-service.js";
import { UserSuiteService } from "@certd/commercial-core"; import {UserSuiteService} from "@certd/commercial-core";
import { UserSettingsService } from "../../mine/service/user-settings-service.js"; import {UserSettingsService} from "../../mine/service/user-settings-service.js";
import { SiteIpEntity } from "../entity/site-ip.js"; import {SiteIpEntity} from "../entity/site-ip.js";
import dns from "dns"; import dns from "dns";
import { logger, safePromise } from "@certd/basic"; import {logger, safePromise} from "@certd/basic";
import dayjs from "dayjs"; import dayjs from "dayjs";
import { siteTester } from "./site-tester.js"; import {siteTester} from "./site-tester.js";
import { PeerCertificate } from "tls"; import {PeerCertificate} from "tls";
import { SiteInfoService } from "./site-info-service.js";
@Provide() @Provide()
@Scope(ScopeEnum.Request, { allowDowngrade: true }) @Scope(ScopeEnum.Request, { allowDowngrade: true })
@ -31,9 +30,6 @@ export class SiteIpService extends BaseService<SiteIpEntity> {
@Inject() @Inject()
userSettingsService: UserSettingsService; userSettingsService: UserSettingsService;
@Inject()
siteInfoService: SiteInfoService;
//@ts-ignore //@ts-ignore
getRepository() { getRepository() {
@ -83,11 +79,11 @@ export class SiteIpService extends BaseService<SiteIpEntity> {
}); });
} }
await this.checkAll(entity.id); await this.checkAll(entity);
} }
async check(ipId: number, domain?: string, port?: number) { async check(ipId: number, domain: string, port: number) {
if(!ipId){ if(!ipId){
return return
} }
@ -95,11 +91,6 @@ export class SiteIpService extends BaseService<SiteIpEntity> {
if (!entity) { if (!entity) {
return; return;
} }
if (domain == null || port == null){
const siteEntity = await this.siteInfoService.info(entity.siteId);
domain = siteEntity.domain;
port = siteEntity.httpsPort;
}
try { try {
await this.update({ await this.update({
id: entity.id, id: entity.id,
@ -151,8 +142,8 @@ export class SiteIpService extends BaseService<SiteIpEntity> {
} }
} }
async checkAll(siteId: number) { async checkAll(siteInfo: SiteInfoEntity) {
const siteInfo = await this.siteInfoService.info(siteId); const siteId = siteInfo.id;
const ips = await this.repository.find({ const ips = await this.repository.find({
where: { where: {
siteId: siteId siteId: siteId
@ -162,13 +153,14 @@ export class SiteIpService extends BaseService<SiteIpEntity> {
const port = siteInfo.httpsPort; const port = siteInfo.httpsPort;
const promiseList = []; const promiseList = [];
for (const ip of ips) { for (const ip of ips) {
promiseList.push(async () => { const func = async () => {
try { try {
await this.check(ip.id, domain, port); await this.check(ip.id, domain, port);
} catch (e) { } catch (e) {
logger.error("check site ip error", e); logger.error("check site ip error", e);
} }
}); }
promiseList.push(func());
} }
Promise.all(promiseList); Promise.all(promiseList);
} }

View File

@ -52,16 +52,12 @@ export class SiteTester {
if (req.ipAddress) { if (req.ipAddress) {
//使用固定的ip //使用固定的ip
const ipAddress = req.ipAddress; const ipAddress = req.ipAddress;
options.lookup = (hostname: string, options: any, callback: any) => { options.headers={
//判断ip是v4 还是v6 host: options.host,
console.log("options", options); //sni
console.log("ipaddress", ipAddress); servername: options.host
if (ipAddress.indexOf(":") > -1) { }
callback(null, ipAddress, 6); options.host = ipAddress;
} else {
callback(null, ipAddress, 4);
}
};
} }
options.agent = new https.Agent({ keepAlive: false }); options.agent = new https.Agent({ keepAlive: false });