Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: Add SNMP Monitor #4717

Merged
merged 75 commits into from
Jul 15, 2024
Merged
Changes from 1 commit
Commits
Show all changes
75 commits
Select commit Hold shift + click to select a range
d92003e
SNMP Initial Commits
mattv8 Apr 27, 2024
a3cdd69
Use net-snmp instead of snmp-native
mattv8 Apr 29, 2024
ff5890a
Updated a comment
mattv8 Apr 29, 2024
4a882be
Further SNMP monitor development
mattv8 Apr 29, 2024
99dc4cf
Wrong variable used
mattv8 Apr 30, 2024
138075a
Update db migration: allow nulls
mattv8 Apr 30, 2024
9c8024c
Update db migration: down function
mattv8 Apr 30, 2024
9d28fcf
Update bean model backend
mattv8 Apr 30, 2024
4593afb
Frontend input validation
mattv8 Apr 30, 2024
9848ce4
Minor frontend styling
mattv8 Apr 30, 2024
704ffd3
Finalized SNMP monitor
mattv8 Apr 30, 2024
b4bd003
Merge branch 'master' into snmp-monitor
CommanderStorm Apr 30, 2024
ba47aca
Apply suggestions from code review
mattv8 Apr 30, 2024
7459654
ES Lint Compliant
mattv8 May 1, 2024
e944492
Corrected down function
mattv8 May 1, 2024
97a9094
ES Lint Compliant
mattv8 May 1, 2024
9ba0f68
Remove supurfluous log.debug
mattv8 May 1, 2024
ba84f01
Delete .EditMonitor.vue.swp
mattv8 May 1, 2024
4699a1c
ES Lint Compliant
mattv8 May 1, 2024
d83c2b9
Revert unintentional changes to EditMonitor.vue
mattv8 May 2, 2024
f059d54
Use frontend timeout
mattv8 May 2, 2024
8e56a81
Refactor how strings/numerics are parsed
mattv8 May 2, 2024
c87ac2f
Move getKey() to util.ts
mattv8 May 3, 2024
09fd816
Updated code comments
mattv8 May 3, 2024
407f729
New dependency for net-snmp
mattv8 May 3, 2024
9053b48
Merge branch 'louislam:master' into snmp-monitor
mattv8 May 3, 2024
4386d0a
Apply suggestions from code review
mattv8 May 5, 2024
0280b2a
A comment about varbinds[0] for clarification
mattv8 May 6, 2024
86b997c
Limit to <= SNMPv2c for now
mattv8 May 6, 2024
0384b34
Remove unnecessary func getKey
mattv8 May 6, 2024
997791b
Default: invalid condition error
mattv8 May 6, 2024
1fe1bb5
Given that above throws, the else case is not nessesary
mattv8 May 6, 2024
433e317
Simplify error catch
mattv8 May 6, 2024
6037912
Consistent placeholder text
mattv8 May 6, 2024
c68b1c6
Remove unnecessary func getKey
mattv8 May 6, 2024
e9b52eb
Separate error cases for SNMP varbind returns
mattv8 May 6, 2024
4ef66b3
SNMP version helptext
mattv8 May 6, 2024
19f21a9
SNMP OID helptext
mattv8 May 6, 2024
56e7fa8
Helptext ALL THE THINGS
mattv8 May 6, 2024
f4842ea
Translation key for OID
mattv8 May 6, 2024
2b5d100
Ensure SNMP session is closed properly
mattv8 May 6, 2024
e5fb726
Missed changes leftover from removal of getKey()
mattv8 May 7, 2024
2015142
Maybe don't helptext all the things...
mattv8 May 7, 2024
8b4b27f
Final cleanup of changes to EditMonitor.vue
mattv8 May 7, 2024
da8f0d1
Apply suggestions from code review
mattv8 May 8, 2024
1c47407
Re-use monitor.radiusPassword for community string
mattv8 May 8, 2024
c475994
Fix ES Lint
mattv8 May 8, 2024
d25ee8f
Using JSON Query Expressions
mattv8 May 10, 2024
7eee5db
Variable changes
mattv8 Jun 5, 2024
b2d76bc
Refactor line for conciseness
mattv8 Jun 5, 2024
2d2c186
Fix: a typo
mattv8 Jun 5, 2024
efb1642
Blend json-query and snmp monitors
mattv8 Jun 5, 2024
36dc94b
Better type handling
mattv8 Jun 6, 2024
10d3188
Query json directly rather than with $.value
mattv8 Jun 6, 2024
eaa935c
Also return result of the evaluation
mattv8 Jun 6, 2024
fdc145b
Added Robustness
mattv8 Jun 7, 2024
23f844d
Error handling robustness
mattv8 Jun 7, 2024
8235291
Fix: Cast to string then eval
mattv8 Jun 10, 2024
e2e8109
Helpful error when query returns object or array
mattv8 Jun 10, 2024
43bd09b
Update 2024-04-26-0000-snmp-monitor.js
mattv8 Jun 10, 2024
69c22ed
Removed "Control Value"
mattv8 Jun 12, 2024
5dc4bb6
Merge branch 'master' into snmp-monitor
CommanderStorm Jun 12, 2024
b5a73e5
Apply suggestions from code review
mattv8 Jun 12, 2024
248aec8
Formtting fix
CommanderStorm Jun 12, 2024
c124f3a
Formtting fix
CommanderStorm Jun 12, 2024
9820f57
Truncate long responses
mattv8 Jun 13, 2024
6fc0cbf
ES Lint
mattv8 Jun 13, 2024
092688a
ES Lint
mattv8 Jun 13, 2024
46ecb82
"Hostname or IP Address" back to "Hostname"
mattv8 Jun 13, 2024
a1f31f9
Remove jsonQueryDescription from lang files
mattv8 Jun 13, 2024
e237d66
"Hostname or IP Address" back to "Hostname"
mattv8 Jun 13, 2024
a037448
C&P typo from review
mattv8 Jun 14, 2024
8d8ce23
Robustness and edge-case handling
mattv8 Jun 14, 2024
71f9384
Merge branch 'master' into snmp-monitor
CommanderStorm Jul 15, 2024
c82de20
fixed merge-typo
CommanderStorm Jul 15, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
ES Lint Compliant
mattv8 committed May 1, 2024
commit 7459654e11b5c580989246f3127a451f625ea22e
4 changes: 2 additions & 2 deletions db/knex_migrations/2024-04-26-0000-snmp-monitor.js
Original file line number Diff line number Diff line change
@@ -3,12 +3,12 @@ exports.up = function (knex) {
.alterTable("monitor", function (table) {
table.string("snmp_community_string", 255).defaultTo("public"); // Add snmp_community_string column
table.string("snmp_oid").defaultTo(null); // Add oid column
table.enum("snmp_version", ["1", "2c", "3"]).defaultTo("2c"); // Add snmp_version column with enum values
table.enum("snmp_version", [ "1", "2c", "3" ]).defaultTo("2c"); // Add snmp_version column with enum values
table.float("snmp_control_value").defaultTo(null); // Add control_value column as float
table.string("snmp_condition").defaultTo(null); // Add oid column
});
};

exports.down = function (knex) {
// Nothing to do here
mattv8 marked this conversation as resolved.
Show resolved Hide resolved
};
};
30 changes: 18 additions & 12 deletions server/monitor-types/snmp.js
Original file line number Diff line number Diff line change
@@ -11,12 +11,18 @@ class SNMPMonitorType extends MonitorType {
async check(monitor, heartbeat, _server) {

const options = {
port: monitor.port || '161',
port: monitor.port || "161",
retries: monitor.maxretries,
timeout: 1000,
mattv8 marked this conversation as resolved.
Show resolved Hide resolved
version: getKey(snmp.Version, monitor.snmpVersion) || snmp.Version2c,
mattv8 marked this conversation as resolved.
Show resolved Hide resolved
};

/**
* Retrieves the key from the provided object corresponding to the given value.
* @param {object} obj - The object to search.
* @param {*} value - The value to search for.
* @returns {string|null} - The key associated with the value, or null if not found.
*/
function getKey(obj, value) {
mattv8 marked this conversation as resolved.
Show resolved Hide resolved
return Object.keys(obj).find(key => obj[key] === value) || null;
}
@@ -25,14 +31,14 @@ class SNMPMonitorType extends MonitorType {
const session = snmp.createSession(monitor.hostname, monitor.snmpCommunityString, options);

// Handle errors during session creation
session.on('error', (error) => {
session.on("error", (error) => {
heartbeat.status = DOWN;
heartbeat.msg = `SNMP: Error creating SNMP session: ${error.message}`;
log.debug("monitor", heartbeat.msg);
});

const varbinds = await new Promise((resolve, reject) => {
mattv8 marked this conversation as resolved.
Show resolved Hide resolved
session.get([monitor.snmpOid], (error, varbinds) => {
session.get([ monitor.snmpOid ], (error, varbinds) => {
if (error) {
reject(error);
} else {
@@ -42,27 +48,27 @@ class SNMPMonitorType extends MonitorType {
});
});

if (varbinds.length === 0 || getKey(snmp.ObjectType, varbinds[0].type) === 'NoSuchInstance') {
if (varbinds.length === 0 || getKey(snmp.ObjectType, varbinds[0].type) === "NoSuchInstance") {
throw new Error(`No varbinds returned from SNMP session (OID: ${monitor.snmpOid})`);
mattv8 marked this conversation as resolved.
Show resolved Hide resolved
} else {
const value = varbinds[0].value;
const numericValue = parseInt(value);
const stringValue = value.toString('ascii');
const stringValue = value.toString("ascii");

switch (monitor.snmpCondition) {
case '>':
case ">":
heartbeat.status = numericValue > monitor.snmpControlValue ? UP : DOWN;
break;
case '>=':
case ">=":
heartbeat.status = numericValue >= monitor.snmpControlValue ? UP : DOWN;
break;
case '<':
case "<":
heartbeat.status = numericValue < monitor.snmpControlValue ? UP : DOWN;
break;
case '<=':
case "<=":
heartbeat.status = numericValue <= monitor.snmpControlValue ? UP : DOWN;
break;
case '==':
case "==":
if (!isNaN(value) && !isNaN(monitor.snmpControlValue)) {
// Both values are numeric, parse them as numbers
heartbeat.status = parseFloat(value) === parseFloat(monitor.snmpControlValue) ? UP : DOWN;
@@ -71,15 +77,15 @@ class SNMPMonitorType extends MonitorType {
heartbeat.status = value.toString() === monitor.snmpControlValue.toString() ? UP : DOWN;
}
break;
case 'contains':
case "contains":
heartbeat.status = stringValue.includes(monitor.snmpControlValue) ? UP : DOWN;
break;
default:
heartbeat.status = DOWN;
heartbeat.msg = `Invalid condition: ${monitor.snmpCondition}`;
break;
}
heartbeat.msg = `SNMP value ` + (heartbeat.status ? `passes` : `does not pass`) + ` comparison: ${value.toString('ascii')} ${monitor.snmpCondition} ${monitor.snmpControlValue}`;
heartbeat.msg = "SNMP value " + (heartbeat.status ? "passes" : "does not pass") + ` comparison: ${value.toString("ascii")} ${monitor.snmpCondition} ${monitor.snmpControlValue}`;

}
session.close();
mattv8 marked this conversation as resolved.
Show resolved Hide resolved