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数据查询目标服务,实现服务器状态监控,发现异常发送到微信群提醒的更多相关文章

  1. python 全栈开发,Day75(Django与Ajax,文件上传,ajax发送json数据,基于Ajax的文件上传,SweetAlert插件)

    昨日内容回顾 基于对象的跨表查询 正向查询:关联属性在A表中,所以A对象找关联B表数据,正向查询 反向查询:关联属性在A表中,所以B对象找A对象,反向查询 一对多: 按字段:xx book ----- ...

  2. Django与Ajax,文件上传,ajax发送json数据,基于Ajax的文件上传,SweetAlert插件

    一.Django与Ajax AJAX准备知识:JSON 什么是 JSON ? JSON 指的是 JavaScript 对象表示法(JavaScript Object Notation) JSON 是轻 ...

  3. ajax 发送json数据时为什么需要设置contentType: "application/json”

    1. ajax发送json数据时设置contentType: "application/json”和不设置时到底有什么区别? contentType: "application/j ...

  4. ajax发送json数据时为什么需要设置contentType: "application/json”

    1. ajax发送json数据时设置contentType: "application/json”和不设置时到底有什么区别?contentType: "application/js ...

  5. Django中数据传输编码格式、ajax发送json数据、ajax发送文件、django序列化组件、ajax结合sweetalert做二次弹窗、批量增加数据

    前后端传输数据的编码格式(contentType) 提交post请求的两种方式: form表单 ajax请求 前后端传输数据的编码格式 urlencoded formdata(form表单里的) ja ...

  6. SpringMVC客户端发送json数据时报400错误

    当测试客户端发送json数据给服务器时,找不到响应路径? 原来是参数类型不符,即使是json也要考虑参数的个数和类型 解决:将age请求参数由"udf"改为"3" ...

  7. iOS开发网络篇—发送json数据给服务器以及多值参数

    iOS开发网络篇—发送json数据给服务器以及多值参数 一.发送JSON数据给服务器 发送JSON数据给服务器的步骤: (1)一定要使用POST请求 (2)设置请求头 (3)设置JSON数据为请求体 ...

  8. 【转】iOS开发网络篇—发送json数据给服务器以及多值参数

    原文: http://www.cnblogs.com/wendingding/p/3950132.html 一.发送JSON数据给服务器 发送JSON数据给服务器的步骤: (1)一定要使用POST请求 ...

  9. perl post发送json数据

    sub  wx_init {                #$login_url ="https://wx.qq.com/cgi-bin/mmwebwx-bin/webwxinit?r=- ...

  10. JSON的简单使用_向前台发送JSON数据

    转自:http://www.cnblogs.com/digdeep/p/5574366.html 1.前台页面 <%@ page language="java" conten ...

随机推荐

  1. C 语言用户输入详解:scanf、fgets、内存地址解析及实用指南

    C 语言中的用户输入 您已经学习了 printf() 函数用于在 C 语言中输出值. 要获取用户输入,可以使用 scanf() 函数: // 声明一个整数变量,用于存储我们从用户那里获得的数字 int ...

  2. 【C#上位机】西门子1200PLC实用定位控制程序案例

    1. 引言 新阁教育这篇文章是一篇综合性非常强的文章,从PLC输入输出及步进电机接线开始,到PLC运动控制程序编写,再到后续的ModbusTCP通信协议及上位机编程实现最终控制,涉及知识面比较广,能够 ...

  3. wchar_t 字符拼接

    wcscat(L"C:\\abc", L"\\GPR.log");

  4. DevEco Device Tool 3.0 Release带来5大能力升级,让智能设备开发更高效

    原文:https://mp.weixin.qq.com/s/QYlHUU05BDlzVxbfZbiKOg,点击链接查看更多技术内容. DevEco Device Tool是面向智能设备开发者提供的一站 ...

  5. Python 爬虫进阶五之多线程的用法

    Python 爬虫进阶五之多线程的用法 作者 崔庆才   发表于 2016-11-03   分类于 Python   阅读次数: 60553   本文字数: 7.5k   阅读时长 ≈ 7 分钟 前言 ...

  6. mysql 必知必会整理—数据插入和更新还有删除[十]

    前言 简单整理数据插入更新还有删除. 正文 插入: INSERT INTO customers( customers.cust_address, customers.cust_city, custom ...

  7. lattice crosslink开发板mipi核心板csi测试dsi屏lif md6000 fpga 常见问题解答

    1. 概述 CrossLink开发板,是用Lattice的芯片CrossLink 家族系列的,LIF-MD6000-6JM80I.该芯片用于桥接视频接口功能,自带2路MIPI硬核的功能,4 LANE  ...

  8. 最简编译CockroachDB 21.2

    编译CockroachDB比较麻烦,尤其是从git下载代码编译还需要关联项目的下载,本文整理从官网下载代码的编译过程,非常简单,几乎没有异常,供大家参考. 编译CockroachDB 21.2 1.安 ...

  9. 力扣8(java)-字符串转整数(atoi)(中等)

    题目: 请你来实现一个 myAtoi(string s) 函数,使其能将字符串转换成一个 32 位有符号整数(类似 C/C++ 中的 atoi 函数). 函数 myAtoi(string s) 的算法 ...

  10. 力扣24(java&python)-两两交换链表中的节点(中等)

    题目: 给你一个链表,两两交换其中相邻的节点,并返回交换后链表的头节点.你必须在不修改节点内部的值的情况下完成本题(即,只能进行节点交换) 示例 1: 输入:head = [1,2,3,4] 输出:[ ...