NODEJS通过发送json数据查询目标服务,实现服务器状态监控,发现异常发送到微信群提醒
root@aea87fa6e6a2:/home/node# cat login2.js
const request = require('request-promise');
const moment = require('moment');
const cron = require('node-cron');
process.env.TZ = 'Asia/Shanghai'; // 设置时区为上海时区
const rp = require('request-promise');
// 三台服务器地址、用户名和密码数组
const nTime=30;
const servers = [
{ ip: '192.168.0.4', username: '0463511', password: 'i@123456' },
{ ip: '192.168.0.95', username: '0721702', password: '123456' },
{ ip: '192.168.0.196', username: '1820318', password: '123456' }
];
// 登录链接
const loginUrl = 'http://{{server}}/api/auth/login'; // {{server}} 是占位符,将 在每次迭代中替换为服务器地址
// 数据采集链接
const dataUrl = 'http://{{server}}/api/icsp/scheduleLog/page?page=1&limit=10';
// 定时器间隔(以毫秒为单位)
const interval = 60000;
// 循环检测主函数
async function loopCheck() {
for (const server of servers) {
try {
serverIP=server.ip;
// 构建当前服务器的登录链接
const currentLoginUrl = loginUrl.replace('{{server}}', server.ip);
// 构建当前服务器的数据采集链接
const currentDataUrl = dataUrl.replace('{{server}}', server.ip);
// 发送登录请求
const response = await request.post({
url: currentLoginUrl,
json: true,
body: {
userName: server.username,
password: server.password
}
});
// 登录成功则输出信息
if (response.statusCode === 200) {
console.log(`[${moment().format('YYYY-MM-DD HH:mm:ss')}] 登录服 务器 ${server.ip} 成功`);
// 获取登录凭证(token)
const token = response.data.token;
// 使用获取的 token 发起数据采集请求
await dataRequest(currentDataUrl, token,server.ip);
} else {
console.error(`[${moment().format('YYYY-MM-DD HH:mm:ss')}] 登录 服务器 ${server.ip} 失败:`, response.message);
}
} catch (error) {
console.error(`[${moment().format('YYYY-MM-DD HH:mm:ss')}] 登录服务 器 ${server.ip} 失败:`, error.message);
}
await sleep(3000);//服务器间的检测间隔
}
// 检测完所有服务器后,等待一段时间后再次执行循环检测
// setTimeout(loopCheck, interval); //这里是循环检测入口
}
async function loopCheck_info() {
for (const server of servers) {
try {
serverIP=server.ip;
// 构建当前服务器的登录链接
const currentLoginUrl = loginUrl.replace('{{server}}', server.ip);
// 构建当前服务器的数据采集链接
const currentDataUrl = dataUrl.replace('{{server}}', server.ip);
// 发送登录请求
const response = await request.post({
url: currentLoginUrl,
json: true,
body: {
userName: server.username,
password: server.password
}
});
// 登录成功则输出信息
if (response.statusCode === 200) {
console.log(`[${moment().format('YYYY-MM-DD HH:mm:ss')}] 登录服 务器 ${server.ip} 成功`);
// 获取登录凭证(token)
const token = response.data.token;
// 使用获取的 token 发起数据采集请求
await dataRequest_info(currentDataUrl, token,server.ip);
} else {
console.error(`[${moment().format('YYYY-MM-DD HH:mm:ss')}] 登录 服务器 ${server.ip} 失败:`, response.message);
}
} catch (error) {
console.error(`[${moment().format('YYYY-MM-DD HH:mm:ss')}] 登录服务 器 ${server.ip} 失败:`, error.message);
}
await sleep(3000);//服务器间的检测间隔
}
// 检测完所有服务器后,等待一段时间后再次执行循环检测
// setTimeout(loopCheck, interval); //这里是循环检测入口
}
// 数据请求
async function dataRequest(dataUrl, token,serverIP) {
// 构建请求头,包含 token
const headers = {
'Authorization': token,
'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/118.0.0.0 Safari/537.36'
};
// 发送数据采集请求
const response = await request({
url: dataUrl,
method: 'GET',
headers: headers
});
// 解析数据
dataParse(response);
}
// 数据请求2
async function dataRequest_info(dataUrl, token,serverIP) {
// 构建请求头,包含 token
const headers = {
'Authorization': token,
'User-Agent': 'Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/118.0.0.0 Safari/537.36'
};
// 发送数据采集请求
const response = await request({
url: dataUrl,
method: 'GET',
headers: headers
});
// 解析数据
dataParse_info(response);
}
// 解析数据
async function dataParse(response) {
// 检查响应体是否为空
if (!response) {
console.error('Received empty response body.');
return;
}
// 解析 JSON 数据
const jsonData = JSON.parse(response);
// 获取数据数组
const rows = jsonData.data.rows;
// 标志,表示是否找到了符合条件的条目
let found = false;
// 遍历数据数组,查找状态为"数据同步任务运行中"的条目
for (const rowData of rows) {
// 如果状态为"数据同步任务运行中",则打印对应的时间
if (rowData.error === "数据同步任务运行中...") {
found = true;//检测运行变量
//updateTime是最后更新时间,createTime是创建时间。这里用更新时间
const createTime = moment(rowData.updateTime, 'YYYY-MM-DD HH:mm:ss').toDate();
const currentTime = new Date();
const minutesDifference = (currentTime.getTime() - createTime.getTime()) / (1000 * 60);
// 如果时间差超过1小时,则打印超时信息
if (minutesDifference > nTime) {
console.log('脚本监测到系统超时');
console.log('超时时间:', minutesDifference.toFixed(0), '分钟');
await sendLogsToAPI(`${moment().format('YYYY-MM-DD HH:mm:ss')} \nAEO ${serverIP} 异常 \n状态超时: ${minutesDifference.toFixed(0)}分钟` );
}
console.log('时间:', rowData.updateTime);
await sleep(1000);
break; // 找到符合条件的条目后直接跳出循环
}
}
// 如果未找到符合条件的条目,则输出提示信息
if (!found) {
console.log('未找到符合条件的条目');
await sendLogsToAPI(`${moment().format('YYYY-MM-DD HH:mm:ss')} \nAEO ${serverIP} 异常 \n进程没有运行` );
}
}
// 解析数据
async function dataParse_info(response) {
// 检查响应体是否为空
if (!response) {
console.error('Received empty response body.');
return;
}
// 解析 JSON 数据
const jsonData = JSON.parse(response);
// 获取数据数组
const rows = jsonData.data.rows;
// 标志,表示是否找到了符合条件的条目
let found = false;
// 遍历数据数组,查找状态为"数据同步任务运行中"的条目
for (const rowData of rows) {
// 如果状态为"数据同步任务运行中",则打印对应的时间
if (rowData.error === "数据同步任务运行中...") {
found = true;//检测运行变量
//updateTime是最后更新时间,createTime是创建时间。这里用更新时间
const createTime = moment(rowData.updateTime, 'YYYY-MM-DD HH:mm:ss').toDate();
const currentTime = new Date();
const minutesDifference = (currentTime.getTime() - createTime.getTime()) / (1000 * 60);
// 如果时间差超过1小时,则打印超时信息
if (minutesDifference > nTime) {
console.log('脚本监测到系统超时');
console.log('超时时间:', minutesDifference.toFixed(0), '分钟');
await sendLogsToAPI(`${moment().format('YYYY-MM-DD HH:mm:ss')} \nAEO ${serverIP} 异常 \n状态超时: ${minutesDifference.toFixed(0)}分钟`);
}else{
console.log('正常运行中');
await sendLogsToAPI(`${moment().format('YYYY-MM-DD HH:mm:ss')} \nAEO ${serverIP} 正常 \n最后更新: ${minutesDifference.toFixed(0)}分钟前`);
}
console.log('时间:', rowData.updateTime);
await sleep(1000);
break; // 找到符合条件的条目后直接跳出循环
}
}
// 如果未找到符合条件的条目,则输出提示信息
if (!found) {
console.log('未找到符合条件的条目');
await sendLogsToAPI(`${moment().format('YYYY-MM-DD HH:mm:ss')} \nAEO ${serverIP} 异常 \n进程没有运行` );
}
}
const sendLogsToAPI = (logs) => {
const resData = {
"msgtype": "text",
"text": {
"content": logs,
}
};
const options = {
method: "POST",
uri: "https://qyapi.weixin.qq.com/cgi-bin/webhook/send?key=17e3b586-20b3-eca2ffa84130",
// uri: "https://qyapi.weixin.qq.com/cgi-bin/webhook/send?key=8ff711ab-f80b-4874-7adf0831a8bc",
headers: {
"content-type": "application/json",
},
body:JSON.stringify(resData),
};
try {
const response = rp(options);
console.log('提示成功!');
} catch (error) {
console.error('请求出错:', error);
}
};
// 休眠函数
function sleep(ms) {
return new Promise(resolve => setTimeout(resolve, ms));
}
// 开始循环检测
//loopCheck();
// 在每分钟的第 30 秒执行目标函数
cron.schedule('35 8 * * *', () => {
console.log('目标函数在8:35执行!');
loopCheck_info();
// 在这里调用你想要定时执行的函数
});
cron.schedule('*/30 9-20 * * *', () => {
console.log('目标函数在每分钟的第 50分钟执行!');
loopCheck();
// 在这里调用你想要定时执行的函数
});
NODEJS通过发送json数据查询目标服务,实现服务器状态监控,发现异常发送到微信群提醒的更多相关文章
- python 全栈开发,Day75(Django与Ajax,文件上传,ajax发送json数据,基于Ajax的文件上传,SweetAlert插件)
昨日内容回顾 基于对象的跨表查询 正向查询:关联属性在A表中,所以A对象找关联B表数据,正向查询 反向查询:关联属性在A表中,所以B对象找A对象,反向查询 一对多: 按字段:xx book ----- ...
- Django与Ajax,文件上传,ajax发送json数据,基于Ajax的文件上传,SweetAlert插件
一.Django与Ajax AJAX准备知识:JSON 什么是 JSON ? JSON 指的是 JavaScript 对象表示法(JavaScript Object Notation) JSON 是轻 ...
- ajax 发送json数据时为什么需要设置contentType: "application/json”
1. ajax发送json数据时设置contentType: "application/json”和不设置时到底有什么区别? contentType: "application/j ...
- ajax发送json数据时为什么需要设置contentType: "application/json”
1. ajax发送json数据时设置contentType: "application/json”和不设置时到底有什么区别?contentType: "application/js ...
- Django中数据传输编码格式、ajax发送json数据、ajax发送文件、django序列化组件、ajax结合sweetalert做二次弹窗、批量增加数据
前后端传输数据的编码格式(contentType) 提交post请求的两种方式: form表单 ajax请求 前后端传输数据的编码格式 urlencoded formdata(form表单里的) ja ...
- SpringMVC客户端发送json数据时报400错误
当测试客户端发送json数据给服务器时,找不到响应路径? 原来是参数类型不符,即使是json也要考虑参数的个数和类型 解决:将age请求参数由"udf"改为"3" ...
- iOS开发网络篇—发送json数据给服务器以及多值参数
iOS开发网络篇—发送json数据给服务器以及多值参数 一.发送JSON数据给服务器 发送JSON数据给服务器的步骤: (1)一定要使用POST请求 (2)设置请求头 (3)设置JSON数据为请求体 ...
- 【转】iOS开发网络篇—发送json数据给服务器以及多值参数
原文: http://www.cnblogs.com/wendingding/p/3950132.html 一.发送JSON数据给服务器 发送JSON数据给服务器的步骤: (1)一定要使用POST请求 ...
- perl post发送json数据
sub wx_init { #$login_url ="https://wx.qq.com/cgi-bin/mmwebwx-bin/webwxinit?r=- ...
- JSON的简单使用_向前台发送JSON数据
转自:http://www.cnblogs.com/digdeep/p/5574366.html 1.前台页面 <%@ page language="java" conten ...
随机推荐
- clang的lto特性的资料
clang对lto的支持,如下文章介绍的清晰.易懂. ThinLTO llvm+clang 添加 LTO(Link Time Optimization) 支持 编译优化之 - 链接时优化(LTO)入门 ...
- 自动编号工具类:NumAutoUtils详解
在软件开发中,经常需要生成唯一的编号,例如订单号.发票号.实验编号等.为了简化这一过程,本文将介绍一个Java工具类NumAutoUtils,它可以帮助我们生成带有前缀和日期的自动编号. 概述 Num ...
- C语言 03 VSCode开发
安装好 C 语言的开发环境后,就需要创建项目进行开发了. 使用 IDE(集成开发环境)进行开发了. C 语言的开发工具很多,现在主流的有 Clion.Visual Studio.VSCode. 这里以 ...
- DevEco Hvigor高效编译,构建过程新秘籍
作者:Lewei,华为终端BG编译构建技术专家 DevEco Hvigor是使用TypeScript语言开发的全新轻量化的任务调度工具,针对HarmonyOS应用提供了一系列编译构建任务,支持将H ...
- 【中秋国庆不断更】XML在HarmonyOS中的生成,解析与转换(下)
一.XML解析 对于以XML作为载体传递的数据,实际使用中需要对相关的节点进行解析,一般包括解析XML标签和标签值.解析XML属性和属性值.解析XML事件类型和元素深度三类场景. XML模块提供Xml ...
- 【直播预告】HarmonyOS极客松赋能直播第三期:一次开发多端部署与ArkTS卡片开发
- 4天带你上手HarmonyOS ArkUI开发——《HarmonyOS ArkUI入门训练营之健康生活实战》
<HarmonyOS ArkUI入门训练营之健康饮食应用>是面向入门开发者打造的实战课程系列.特邀华为终端BG高级开发工程师作为本次训练营讲师,以健康饮食为例,开展技术教学及实战案例分享 ...
- 实时数仓构建:Flink+OLAP查询的一些实践与思考
今天是一篇架构分享内容. 1.概述 以Flink为主的计算引擎配合OLAP查询分析引擎组合进而构建实时数仓,其技术方案的选择是我们在技术选型过程中最常见的问题之一.也是很多公司和业务支持过程中会实实在 ...
- mysql8在Win10下安装教程
一.准备工作 下载mysql8安装包,下载URL地址:https://mirrors.tuna.tsinghua.edu.cn/mysql/downloads/MySQL-8.0/ 二.管理员权限执行 ...
- Pygame安装以及解决问题:Try to run this command from the system terminal. Make sure that you use the correct version of 'pip......
在这里记录一下我的安装过程: 1.首先找到自己python程序安装目录下的Scripts文件夹(里面有pip这里面): 2.使用快捷键win + R 打开终端,先进入到安装python的盘符,然后进入 ...