view adslstats.py @ 3:3748cec0e322

merge
author Daniel O'Connor <darius@dons.net.au>
date Wed, 20 Nov 2013 23:56:29 +1030
parents b1048f889ef8 a795b6cd8b1a
children 98d351a87043
line wrap: on
line source

#!/usr/bin/env python
############################################################################
#
# Parse ADSL link stats for Billion 7300G & generate RRD archives & graphs
#
############################################################################
#
# Copyright (C) 2013 Daniel O'Connor. All rights reserved.
#
# Redistribution and use in source and binary forms, with or without
# modification, are permitted provided that the following conditions
# are met:
# 1. Redistributions of source code must retain the above copyright
#    notice, this list of conditions and the following disclaimer.
# 2. Redistributions in binary form must reproduce the above copyright
#    notice, this list of conditions and the following disclaimer in the
#    documentation and/or other materials provided with the distribution.
#
# THIS SOFTWARE IS PROVIDED BY AUTHOR AND CONTRIBUTORS ``AS IS'' AND
# ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
# ARE DISCLAIMED.  IN NO EVENT SHALL AUTHOR OR CONTRIBUTORS BE LIABLE
# FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
# DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
# OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
# HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
# OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
# SUCH DAMAGE.
#
############################################################################

import ConfigParser
import optparse
import os
import re
import rrdtool
import time
import urllib
from bs4 import BeautifulSoup

conf = ConfigParser.ConfigParser()
conf.add_section('global')
conf.set('global', 'username', 'admin')
conf.set('global', 'password', 'admin')
conf.set('global', 'name', 'dsl.dons.net.au')

conflist = ['adslstats.ini']
if ('HOME' in os.environ):
    conflist.append(os.path.expanduser('~/.adslstats.ini'))
conf.read(conflist)

usage = '''%prog [options]'''
opts = optparse.OptionParser(usage)
opts.add_option('-v', '--verbose', action="store_true", default=False,
                    help="Enable debug output")
opts.add_option('-g', '--graph', action="store_true", default=False,
                    help="Generate a graph")
opts.add_option('-u', '--update', action="store_true", default=False,
                    help="Update RRD")
opts.add_option('-a', '--authname', action="store", default=conf.get('global', 'username'),
                    help="Username to login to modem")
opts.add_option('-p', '--password', action="store", default=conf.get('global', 'password'),
                    help="Password to login to modem")
opts.add_option('-n', '--name', action="store", default=conf.get('global', 'name'),
                    help="Hostname of modem")
opts.add_option('-b', '--base', action="store", default="/home/darius/projects/adslstats/adslstats",
                    help="Base directory for RRD & PNGs")

(options, args) = opts.parse_args()

statsurl = "http://%s/status/status_deviceinfo.htm" % (options.name)
rrdname = "%s.rrd" % (options.base)
graphbasename = options.base

class ADSLStats(object):
    def __str__(self):
        return """Line Rate - Up: %d kbits, Down %d kbits
Noise Margin - Up: %.1f dB, Down %.1f dB
Attenuation - Up: %.1f dB, Down %.1f dB""" % (self.upstream, self.downstream,
                                              self.nmup, self.nmdown,
                                              self.attenup, self.attendown)

def getstats(f):
    s = BeautifulSoup(f)
    a = s.findAll('td')
    
    # Sanity check in case the firmware changes page layout
    assert(a[122]('font')[0].contents[0] == 'SNR Margin')
    assert(a[129]('font')[0].contents[0] == 'Line Attenuation')
    assert(a[136]('font')[0].contents[0] == 'Data Rate')

    # Check if the modem is offline
    if a[124].contents[0].strip() == 'N/A':
	return None

    stats = ADSLStats()

    stats.nmdown = float(a[124].contents[0])		# dB
    stats.nmup = float(a[125].contents[0])		# dB

    stats.attendown = float(a[131].contents[0])		# dB
    stats.attenup = float(a[132].contents[0])		# dB

    stats.downstream = float(a[138].contents[0])	# kbit/sec
    stats.upstream = float(a[139].contents[0])		# kbit/sec
    
    # Check if the modem is offline
    if a[9].td.findNext('td').contents[0].contents[0].find('N/A') != -1:
	return None
    stats.upstream = cleannum(a[7].td.findNext('td').contents[0].contents[0])   # kbits
    stats.downstream = cleannum(a[8].td.findNext('td').contents[0].contents[0]) # kbits
    stats.nmup = cleannum(a[9].td.findNext('td').contents[0].contents[0])       # dB
    stats.nmdown = cleannum(a[10].td.findNext('td').contents[0].contents[0])    # dB
    stats.attenup = cleannum(a[11].td.findNext('td').contents[0].contents[0])   # dB
    stats.attendown = cleannum(a[12].td.findNext('td').contents[0].contents[0]) # dB
>>>>>>> other

    return stats

# Setup RRD
# We expect data to be logged every 5 minutes
# Average 12 5 minute points -> hourly stats (keep 168 - a weeks worth)
# Average 288 5 minute points -> daily stats (keep 365 - a years worth)
# Detemine minimum & maximum for an hour and keep a weeks worth.
def makerrd(filename):
    rrdtool.create(filename,
                   '--step', '300',
                   'DS:upstream:GAUGE:3600:32:25000',   # Upstream (kbits) - 24mbit is ADSL2+ max
                   'DS:downstream:GAUGE:3600:32:25000', # Downstream (kbits)
                   'DS:nmup:GAUGE:3600:0:100',          # Upstream Noise margin (dB)
                   'DS:nmdown:GAUGE:3600:0:100',        # Downstream Noise margin (dB)
                   'DS:attenup:GAUGE:3600:0:100',       # Upstream Attenuation (dB)
                   'DS:attendown:GAUGE:3600:0:100',     # Downstream Attenuation (dB)
                   'RRA:AVERAGE:0.1:12:168',
                   'RRA:AVERAGE:0.1:288:365',
                   'RRA:MIN:0.1:12:168',
                   'RRA:MAX:0.1:12:168')

# Update the RRD (format stats as expected)
def updaterrd(filename, tstamp, stats):
    rrdtool.update(filename,
                   '%d:%d:%d:%f:%f:%f:%f' % (tstamp,
                                             stats.upstream,
                                             stats.downstream,
                                             stats.nmup,
                                             stats.nmdown,
                                             stats.attenup,
                                             stats.attendown))

# Open the URL and call the parser, the update the RRD
def doupdate():
    opener = urllib.FancyURLopener()
    opener.prompt_user_passwd = lambda host, realm: (options.authname, options.password)
    f = opener.open(statsurl)
    #f = open("adsl.html")
    stats = getstats(f)
    if stats == None:
	if options.verbose:
	    print "Modem is offline"
	return
    if options.verbose:
        print str(stats)
    updaterrd(rrdname, int(time.time()), stats)

# Generate a graph
def gengraph():

    linkargs = (
        '-a', 'PNG',
        '-X', '0',
        '--vertical-label', 'kbit/sec',
        '--slope-mode',

        'DEF:upstream=%s:upstream:AVERAGE' % rrdname,
        'DEF:upstreammin=%s:upstream:MIN' % rrdname,
        'DEF:upstreammax=%s:upstream:MAX' % rrdname,
        
        'DEF:downstream=%s:downstream:AVERAGE' % rrdname,
        'DEF:downstreammin=%s:downstream:MIN' % rrdname,
        'DEF:downstreammax=%s:downstream:MAX' % rrdname,

        'CDEF:upstreamdif=upstreammax,upstreammin,-',
        'CDEF:downstreamdif=downstreammax,downstreammin,-',
        
        'LINE0:upstreammin#000000:',
        'AREA:upstreamdif#00dc76::STACK',
        'LINE1:upstream#00ff00:Upstream',
        
        'LINE0:downstreammin#000000:',
        'AREA:downstreamdif#ff8686::STACK',
        'LINE1:downstream#ff0000:Downstream')

    signalargs = (
        '-a', 'PNG',
        '--vertical-label', 'dB',
        '--slope-mode',

        'DEF:upstream=%s:upstream:AVERAGE' % rrdname,
        'DEF:downstream=%s:downstream:AVERAGE' % rrdname,
        
        'DEF:nmup_=%s:nmup:AVERAGE' % rrdname,
        'DEF:nmupmin_=%s:nmup:MIN' % rrdname,
        'DEF:nmupmax_=%s:nmup:MAX' % rrdname,
        
        'DEF:nmdown_=%s:nmdown:AVERAGE' % rrdname,
        'DEF:nmdownmin_=%s:nmdown:MIN' % rrdname,
        'DEF:nmdownmax_=%s:nmdown:MAX' % rrdname,
        
        'DEF:attenup=%s:attenup:AVERAGE' % rrdname,
        'DEF:attenupmin=%s:attenup:MIN' % rrdname,
        'DEF:attenupmax=%s:attenup:MAX' % rrdname,
        
        'DEF:attendown=%s:attendown:AVERAGE' % rrdname,
        'DEF:attendownmin=%s:attendown:MIN' % rrdname,
        'DEF:attendownmax=%s:attendown:MAX' % rrdname,
        
        'CDEF:nmup=nmup_,10,*',
        'CDEF:nmupmin=nmupmin_,10,*',
        'CDEF:nmupmax=nmupmax_,10,*',
        'CDEF:nmupdif=nmupmax,nmupmin,-',
        
        'CDEF:nmdown=nmdown_,10,*',
        'CDEF:nmdownmin=nmdownmin_,10,*',
        'CDEF:nmdownmax=nmdownmax_,10,*',
        'CDEF:nmdowndif=nmdownmax,nmdownmin,-',
        
        'CDEF:attenupdif=attenupmax,attenupmin,-',
        
        'CDEF:attendowndif=attendownmax,attendownmin,-',
        
        'LINE0:nmupmin#000000:',
        'AREA:nmupdif#5c5cff::STACK',
        'LINE1:nmup#0000ff:Noise Margin - Up (1/10 dB)',
        
        'LINE0:nmdownmin#000000:',
        'AREA:nmdowndif#009a00::STACK',
        'LINE1:nmdown#00ff00:Noise Margin - Down (1/10 dB)',
        
        'LINE0:attenupmin#000000:',
        'AREA:attenupdif#f98100::STACK',
        'LINE1:attenup#ff0000:Attenuation - Up',
        
        'LINE0:attendownmin#000000:',
        'AREA:attendowndif#aaaaaa::STACK',
        'LINE1:attendown#000000:Attenuation - Down')
    
    rrdtool.graph("%s-hour-link.png" % (graphbasename),
                  '--width', '1024',
                  '--height', '256',
                  '--start', 'end - 7d',
                  '--end', 'now',
                  *linkargs)

    rrdtool.graph("%s-daily-link.png" % (graphbasename),
                  '--width', '1024',
                  '--height', '256',
                  '--start', 'end - 365d',
                  '--end', 'now',
                  *linkargs)


    rrdtool.graph("%s-hour-signal.png" % (graphbasename),
                  '--width', '1024',
                  '--height', '256',
                  '--start', 'end - 7d',
                  '--end', 'now',
                  *signalargs)

    rrdtool.graph("%s-daily-signal.png" % (graphbasename),
                  '--width', '1024',
                  '--height', '256',
                  '--start', 'end - 365d',
                  '--end', 'now',
                  *signalargs)

                  
if __name__ == "__main__":
    if options.update:
        try:
            os.stat(rrdname)
        except OSError, e:
            if e.errno == 2:
                print "rrd not found, creating.."
                makerrd(rrdname)

        doupdate()

    if options.graph:
        gengraph()