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. OpenHarmony中的HDF单链表及其迭代器

    概念 为了性能考虑,嵌入式系统一般使用C语言进行开发,由于C语言标准库没有封装链表,所以嵌入式系统一般自己设计和实现链表这种数据结构.单链表是链表中的一种,本文描述OpenAtom OpenHarmo ...

  2. Docker学习路线9:运行容器

    要启动一个新的容器,我们使用 docker run 命令,后跟镜像名称.基本语法如下: docker run [选项] 镜像 [COMMAND] [ARG...] 例如,要运行官方的 Nginx 镜像 ...

  3. 如何翻译 Markdown 文件?-2-几种商业及开源解决方案介绍

    背景 近期在搭建英文博客-<e-whisper.com>, 需要对现有的所有中文 Markdown 翻译为英文. 需求如下: 将 Markdown 文件从中文 (zh-CN) 翻译为英文 ...

  4. Native Drawing开发指导,实现HarmonyOS基本图形和字体的绘制

      场景介绍 Native Drawing模块提供了一系列的接口用于基本图形和字体的绘制.常见的应用场景举例: ● 2D图形绘制. ● 文本绘制. 接口说明 接口名 描述 OH_Drawing_Bit ...

  5. HDC2021技术分论坛:鸿蒙智联设备开发,这五大法宝你应该拥有

    作者:zhaowenguang,dinglu, 华为高级工程师 Huawei LiteOS是轻量级的开源物联网操作系统.智能硬件使能平台,可广泛应用于智能家居.穿戴式.车联网.制造业等领域,使物联网终 ...

  6. Python2同时输出中文和变量时中文乱码

    Python2同时输出中文和变量时中文乱码 一.问题描述 最近在学习tensorflow时,突然发现一个问题,python2在输出中文和变量时会出现一点问题,如下图: 可以看到,输出并不是想要的结果, ...

  7. 重新点亮linux 命令树————内存与文件系统的查看[二十七]

    前言 简单介绍一下内存的查看. 正文 常用命令. free top 查看磁盘使用率: fdisk df du du和ls的区别 free -h 查看内存使用率: free -g 显示按G来显示,-m用 ...

  8. asp .net core 单页应用

    前言 单页应用其实就是 asp.net core 来作为js service,个人觉得这样更好调试,这种比较适合中小型业务. 正文 因为自己写过混合app是ionic,那么就尝试一下angular的单 ...

  9. PTA前三次题目集总结

    以下内容是我对PTA三次习题作业最后一题的思路,源码以及总结 学到的java知识大多都来自写题目集 这些题目对我对java的认知与学习起到了不小的帮助 答题判题程序-1 题目内容 设计实现答题程序,模 ...

  10. 【笔记】go语言--go语言的依赖管理

    [笔记]go语言--go语言的依赖管理 GO语言的依赖管理 依赖的概念,依赖就是第三方的库,即别人已经做好的库 依赖管理的三个阶段 GOPATH,GOVENDOR, go mod 三个阶段 - GOP ...