-
Notifications
You must be signed in to change notification settings - Fork 17
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
[change] NetJSON DeviceMonitoring compliance #2 #62
base: master
Are you sure you want to change the base?
Changes from 19 commits
d46b568
9168a27
cfa6650
c9962e2
8cd7089
da39965
1800332
cda0295
0edc93a
acafe93
34f05c9
034475c
995b703
83a25de
0f37717
5dc3cc6
d746f2d
56ea551
4e6f1f1
5424532
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,149 @@ | ||
|
||
************ | ||
SNMP backend | ||
************ | ||
|
||
SNMP | ||
==== | ||
|
||
SNMP (Simple Network Management Protocol) is a network protocol very useful for retrieving info from a device. | ||
All the information is retrieved by using codes called MIBs. All MIBs have a tree-like structure, every main information is the root and by adding more detail to the info | ||
the tree gains more depth. | ||
Obviously, by getting the smallest MIB which is "1" or simply " . " one can get all the tree. | ||
|
||
The base SNMP backend contains the following methods (some internal methods are not documented and are subject to change in the future): | ||
|
||
+--------------+------------------------------------------------------------------------------------------------------------------------------------------+ | ||
| **to_dict** | Returns a dict containing monitoring information depending on the type of the device. | | ||
| | It follows the `NetJSON Devicemonitoring <https://netjson.org/rfc.html#name-devicemonitoring>`_ spec | | ||
+--------------+------------------------------------------------------------------------------------------------------------------------------------------+ | ||
| **to_json** | Calls the `to_dict` method and returns a JSON string of the dict | | ||
+--------------+------------------------------------------------------------------------------------------------------------------------------------------+ | ||
| **validate** | Checks if connection with the device is working and raises `NetengineError` in case something is wrong | | ||
+--------------+------------------------------------------------------------------------------------------------------------------------------------------+ | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I suggest to do the same here and drop the table syntax which is painful to maintain |
||
|
||
Initializing an SNMP backend class requires the following arguments: | ||
|
||
+---------------+---------------------------------------------------------------------+ | ||
| **host** | Management ip or hostname of the device | | ||
+---------------+---------------------------------------------------------------------+ | ||
| **community** | Community string for the SNMP connection. Default value is 'public' | | ||
+---------------+---------------------------------------------------------------------+ | ||
| **agent** | Agent string for the SNMP connection | | ||
+---------------+---------------------------------------------------------------------+ | ||
| **port** | Port for the SNMP connection. Default value is `161` | | ||
+---------------+---------------------------------------------------------------------+ | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Are these parameters all mandatory? I mean this: See the source for more info: https://raw.githubusercontent.com/openwisp/netjsonconfig/master/docs/source/backends/openwrt.rst The doc text is embedded in a docstring in the method itself. This is a good practice, the docstring is also shown in some rich shells (bpython should show it, not sure about ipython). |
||
|
||
The SNMP backend provides support for 2 firmwares: | ||
* AirOS | ||
* OpenWRT | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. please list OpenWrt first. Please also link their respective pages |
||
|
||
.. note:: | ||
|
||
The data collected by Netengine is dependant on the OIDs available on your device. Some proprietary manufacturers may not | ||
provide the same information as others. | ||
|
||
***** | ||
AirOS | ||
***** | ||
|
||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. as in netjsonconfig, I suggest to make this a dedicated page, which should point to the base snmp page for the general concepts. Some information regarding methods can be duplicated (no problem with this if we use autodoc). |
||
With AirOS, Netengine is able to collect the following information which is returned in the | ||
`NetJSON Devicemonitoring <https://netjson.org/rfc.html#name-devicemonitoring>`_ format: | ||
|
||
+------------+------------------------------------------------------------------------------------------------------------+ | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. please let's switch from table sytnax which is painful to maintain to simple headings and list after headings. |
||
| general | - uptime: uptime of the device in seconds | | ||
| | - local_time: local time of the device in timestamp | | ||
+------------+------------------------------------------------------------------------------------------------------------+ | ||
| resources | - load: array containing average load on the cpu in the past minute, 5 minutes and 15 minutes respectively | | ||
| | - memory: | | ||
| | - total: total memory in bytes | | ||
| | - buffered: buffered memory in bytes | | ||
| | - free: free memory in bytes | | ||
| | - cached: cached memory in bytes | | ||
| | - swap: | | ||
| | - total: total swap storage in bytes | | ||
| | - free: free swap storage in bytes | | ||
+------------+------------------------------------------------------------------------------------------------------------+ | ||
| interfaces | Each interface is listed with the following information: | | ||
| | | | ||
| | - name | | ||
| | - type | | ||
| | - statistics: | | ||
| | | | ||
| | - rx_bytes | | ||
| | - tx_bytes | | ||
+------------+------------------------------------------------------------------------------------------------------------+ | ||
|
||
AirOS example | ||
============= | ||
|
||
:: | ||
|
||
from netengine.backends.snmp import AirOS | ||
device = AirOS("10.40.0.130") | ||
device.name | ||
'RM5PomeziaSNode' | ||
device.uptime_tuple | ||
(121, 0, 5) # a tuple containing device uptime hours, mins and seconds | ||
|
||
We have just called two simple properties on **device**, but we can ask **device** for more specific values or portions of the SNMP tree not included in the API, just type:: | ||
device.next("1.3.6") | ||
|
||
Otherwise, if you want simply a value of the tree just type:: | ||
device.get_value("oid_you_want_to_ask_for") | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. these methods are not documented above so why are we showing them in the example? I guess we can document (and keep public) |
||
|
||
To collect the whole json:: | ||
device.to_json() | ||
|
||
******* | ||
OpenWRT | ||
******* | ||
|
||
With OpenWRT, Netengine is able to collect the following information which is returned in the | ||
`NetJSON Devicemonitoring <https://netjson.org/rfc.html#name-devicemonitoring>`_ format: | ||
|
||
+------------+----------------------------------------------------------+ | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. please let's switch from table sytnax which is painful to maintain to simple headings and list after headings. |
||
| general | - uptime: uptime of the device in seconds | | ||
| | - local_time: local time of the device in timestamp | | ||
+------------+----------------------------------------------------------+ | ||
| resources | - cpus: number of cpus on the device | | ||
| | - memory: | | ||
| | - total: total memory in bytes | | ||
| | - shared: shared memory in bytes | | ||
| | - used: used memory in bytes | | ||
| | - free: free memory in bytes | | ||
| | - cached: cached memory in bytes | | ||
| | - swap: | | ||
| | - total: total swap storage in bytes | | ||
| | - free: free swap storage in bytes | | ||
+------------+----------------------------------------------------------+ | ||
| interfaces | Each interface is listed with the following information: | | ||
| | | | ||
| | - name: name of the interface (example: "eth0") | | ||
| | - statistics: | | ||
| | | | ||
| | - mac | | ||
| | - type | | ||
| | - up | | ||
| | - rx_bytes | | ||
| | - tx_bytes | | ||
| | - mtu | | ||
| | - addresses: | | ||
| | | | ||
| | - family | | ||
| | - address | | ||
| | - mask | | ||
+------------+----------------------------------------------------------+ | ||
| neighbors | Each neighbor is listed with the following information: | | ||
| | - mac: mac address of the neighbor | | ||
| | - state: state of the neighbor (REACHABLE/STALE/DELAY) | | ||
| | - interface: interface of the neighbor | | ||
| | - ip: ip address of the neighbor | | ||
+------------+----------------------------------------------------------+ | ||
|
||
OpenWRT example | ||
=============== | ||
|
||
The same instructions typed above can be applied to OpenWRT itself, just remember to import the correct firmware by typing:: | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. it's not a firmware, it's a backend. |
||
|
||
from netengine.backends.snmp import OpenWRT |
This file was deleted.
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,152 @@ | ||
# NetJSON DeviceMonitoring schema, | ||
# https://github.com/netjson/netjson/blob/master/schema/device-monitoring.json | ||
schema = { | ||
'$schema': 'http://json-schema.org/draft-07/schema#', | ||
'$id': 'https://raw.githubusercontent.com/netjson/netjson/master/schema/device-monitoring.json', | ||
'title': 'NetJSON Device Monitoring', | ||
'description': 'Monitoring information sent by a device.', | ||
'type': 'object', | ||
'additionalProperties': True, | ||
'required': ['type'], | ||
'properties': { | ||
'type': {'type': 'string', 'enum': ['DeviceMonitoring']}, | ||
'general': { | ||
'type': 'object', | ||
'title': 'General', | ||
'additionalProperties': True, | ||
'properties': { | ||
'local_time': {'type': 'integer'}, | ||
'uptime': {'type': 'integer'}, | ||
}, | ||
}, | ||
'resources': { | ||
'type': 'object', | ||
'title': 'Resources', | ||
'additionalProperties': True, | ||
'properties': { | ||
'load': { | ||
'type': 'array', | ||
'items': {'type': 'number', 'minItems': 3, 'maxItems': 3}, | ||
}, | ||
'memory': { | ||
'id': 'memory', | ||
'type': 'object', | ||
'properties': { | ||
'total': {'type': 'integer'}, | ||
'free': {'type': 'integer'}, | ||
'buffered': {'type': 'integer'}, | ||
'cache': {'type': 'integer'}, | ||
}, | ||
}, | ||
'swap': { | ||
'type': 'object', | ||
'properties': { | ||
'total': {'type': 'integer'}, | ||
'free': {'type': 'integer'}, | ||
}, | ||
}, | ||
'connections': { | ||
'type': 'object', | ||
'properties': { | ||
'ipv4': { | ||
'type': 'object', | ||
'properties': { | ||
'tcp': {'type': 'integer'}, | ||
'udp': {'type': 'integer'}, | ||
}, | ||
}, | ||
'ipv6': { | ||
'type': 'object', | ||
'properties': { | ||
'tcp': {'type': 'integer'}, | ||
'udp': {'type': 'integer'}, | ||
}, | ||
}, | ||
}, | ||
}, | ||
'processes': { | ||
'type': 'object', | ||
'properties': { | ||
'running': {'type': 'integer'}, | ||
'sleeping': {'type': 'integer'}, | ||
'blocked': {'type': 'integer'}, | ||
'zombie': {'type': 'integer'}, | ||
'stopped': {'type': 'integer'}, | ||
'paging': {'type': 'integer'}, | ||
}, | ||
}, | ||
'cpu': { | ||
'type': 'object', | ||
'properties': { | ||
'frequency': {'type': 'integer'}, | ||
'user': {'type': 'integer'}, | ||
'system': {'type': 'integer'}, | ||
'nice': {'type': 'integer'}, | ||
'idle': {'type': 'integer'}, | ||
'iowait': {'type': 'integer'}, | ||
'irq': {'type': 'integer'}, | ||
'softirq': {'type': 'integer'}, | ||
}, | ||
}, | ||
'flash': { | ||
'type': 'object', | ||
'properties': { | ||
'total': {'type': 'integer'}, | ||
'free': {'type': 'integer'}, | ||
}, | ||
}, | ||
'storage': { | ||
'type': 'object', | ||
'properties': { | ||
'total': {'type': 'integer'}, | ||
'free': {'type': 'integer'}, | ||
}, | ||
}, | ||
}, | ||
}, | ||
'interfaces': { | ||
'type': 'array', | ||
'title': 'Interfaces', | ||
'uniqueItems': True, | ||
'additionalItems': True, | ||
'items': { | ||
'type': 'object', | ||
'title': 'Interface', | ||
'additionalProperties': True, | ||
'required': ['name'], | ||
'properties': { | ||
'name': {'type': 'string'}, | ||
'uptime': {'type': 'integer'}, | ||
'statistics': { | ||
'type': 'object', | ||
'properties': { | ||
'collisions': {'type': 'integer'}, | ||
'rx_frame_errors': {'type': 'integer'}, | ||
'tx_compressed': {'type': 'integer'}, | ||
'multicast': {'type': 'integer'}, | ||
'rx_length_errors': {'type': 'integer'}, | ||
'tx_dropped': {'type': 'integer'}, | ||
'rx_bytes': {'type': 'integer'}, | ||
'rx_missed_errors': {'type': 'integer'}, | ||
'tx_errors': {'type': 'integer'}, | ||
'rx_compressed': {'type': 'integer'}, | ||
'rx_over_errors': {'type': 'integer'}, | ||
'tx_fifo_errors': {'type': 'integer'}, | ||
'rx_crc_errors': {'type': 'integer'}, | ||
'rx_packets': {'type': 'integer'}, | ||
'tx_heartbeat_errors': {'type': 'integer'}, | ||
'rx_dropped': {'type': 'integer'}, | ||
'tx_aborted_errors': {'type': 'integer'}, | ||
'tx_packets': {'type': 'integer'}, | ||
'rx_errors': {'type': 'integer'}, | ||
'tx_bytes': {'type': 'integer'}, | ||
'tx_window_errors': {'type': 'integer'}, | ||
'rx_fifo_errors': {'type': 'integer'}, | ||
'tx_carrier_errors': {'type': 'integer'}, | ||
}, | ||
}, | ||
}, | ||
}, | ||
}, | ||
}, | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think we can avoid explaining what SNMP is, we assume peopel using this library will know.
We can however include information about how all the data is retrieved and give a few hints on how to use the base backend to create new SNMP backends for other systems (eg: indicate the python path for the base backend and explain anything important regarding it).