Add --json option to produce JSON results

This commit is contained in:
dana 2016-10-23 22:59:54 -04:00
parent 7b09d8759f
commit 7770673e2c
3 changed files with 60 additions and 7 deletions

View File

@ -76,7 +76,7 @@ Usage
$ speedtest-cli -h $ speedtest-cli -h
usage: speedtest-cli [-h] [--bytes] [--share] [--simple] [--list] usage: speedtest-cli [-h] [--bytes] [--share] [--simple] [--list]
[--server SERVER] [--mini MINI] [--source SOURCE] [--server SERVER] [--mini MINI] [--source SOURCE]
[--timeout TIMEOUT] [--secure] [--version] [--timeout TIMEOUT] [--secure] [--version] [--json]
Command line interface for testing internet bandwidth using speedtest.net. Command line interface for testing internet bandwidth using speedtest.net.
-------------------------------------------------------------------------- --------------------------------------------------------------------------
@ -98,6 +98,7 @@ Usage
--secure Use HTTPS instead of HTTP when communicating with --secure Use HTTPS instead of HTTP when communicating with
speedtest.net operated servers speedtest.net operated servers
--version Show the version number and exit --version Show the version number and exit
--json Output results in JSON format, requires --simple
Inconsistency Inconsistency
------------- -------------

View File

@ -63,6 +63,11 @@ Source IP address to bind to
Show the version number and exit Show the version number and exit
.RE .RE
\fB\-\-json\fR
.RS
Output results in JSON format, requires \-\-simple and python 2.6 or greater
.RE
.SH EXAMPLES .SH EXAMPLES
\fBAutomatically find closest server and start testing\fR \fBAutomatically find closest server and start testing\fR

View File

@ -32,6 +32,7 @@ user_agent = None
source = None source = None
shutdown_event = None shutdown_event = None
scheme = 'http' scheme = 'http'
json_available = True
# Used for bound_interface # Used for bound_interface
@ -149,6 +150,11 @@ else:
print_ = getattr(builtins, 'print') print_ = getattr(builtins, 'print')
del builtins del builtins
try:
import json
except ImportError:
json_available = False
class SpeedtestCliServerListError(Exception): class SpeedtestCliServerListError(Exception):
"""Internal Exception class used to indicate to move on to the next """Internal Exception class used to indicate to move on to the next
@ -547,11 +553,18 @@ def version():
raise SystemExit(__version__) raise SystemExit(__version__)
def badArgument(reason):
"""Show argument error and exit"""
raise SystemExit(reason)
def speedtest(): def speedtest():
"""Run the full speedtest.net test""" """Run the full speedtest.net test"""
global shutdown_event, source, scheme global shutdown_event, source, scheme, json_available
shutdown_event = threading.Event() shutdown_event = threading.Event()
json_result = {}
signal.signal(signal.SIGINT, ctrl_c) signal.signal(signal.SIGINT, ctrl_c)
@ -592,6 +605,10 @@ def speedtest():
'with speedtest.net operated servers') 'with speedtest.net operated servers')
parser.add_argument('--version', action='store_true', parser.add_argument('--version', action='store_true',
help='Show the version number and exit') help='Show the version number and exit')
if json_available:
parser.add_argument('--json', action='store_true',
help='Output results in JSON format, '
'requires --simple')
options = parser.parse_args() options = parser.parse_args()
if isinstance(options, tuple): if isinstance(options, tuple):
@ -604,6 +621,9 @@ def speedtest():
if args.version: if args.version:
version() version()
if json_available and args.json and not args.simple:
badArgument('error: --json requires --simple')
socket.setdefaulttimeout(args.timeout) socket.setdefaulttimeout(args.timeout)
# Pre-cache the user agent string # Pre-cache the user agent string
@ -705,6 +725,16 @@ def speedtest():
if not args.simple: if not args.simple:
print_(('Hosted by %(sponsor)s (%(name)s) [%(d)0.2f km]: ' print_(('Hosted by %(sponsor)s (%(name)s) [%(d)0.2f km]: '
'%(latency)s ms' % best).encode('utf-8', 'ignore')) '%(latency)s ms' % best).encode('utf-8', 'ignore'))
else:
if json_available and args.json:
json_result['host'] = {
'sponsor': best['sponsor'],
'name': best['name']
}
json_result['ping'] = {
'value': best['latency'],
'units': 'ms'
}
else: else:
print_('Ping: %(latency)s ms' % best) print_('Ping: %(latency)s ms' % best)
@ -719,6 +749,13 @@ def speedtest():
dlspeed = downloadSpeed(urls, args.simple) dlspeed = downloadSpeed(urls, args.simple)
if not args.simple: if not args.simple:
print_() print_()
if json_available and args.json:
json_result['download'] = {
'value': ((dlspeed / 1000 / 1000) * args.units[1]),
'units': 'M%s/s' % args.units[0]
}
else:
print_('Download: %0.2f M%s/s' % print_('Download: %0.2f M%s/s' %
((dlspeed / 1000 / 1000) * args.units[1], args.units[0])) ((dlspeed / 1000 / 1000) * args.units[1], args.units[0]))
@ -732,6 +769,13 @@ def speedtest():
ulspeed = uploadSpeed(best['url'], sizes, args.simple) ulspeed = uploadSpeed(best['url'], sizes, args.simple)
if not args.simple: if not args.simple:
print_() print_()
if json_available and args.json:
json_result['upload'] = {
'value': ((ulspeed / 1000 / 1000) * args.units[1]),
'units': 'M%s/s' % args.units[0]
}
else:
print_('Upload: %0.2f M%s/s' % print_('Upload: %0.2f M%s/s' %
((ulspeed / 1000 / 1000) * args.units[1], args.units[0])) ((ulspeed / 1000 / 1000) * args.units[1], args.units[0]))
@ -784,6 +828,9 @@ def speedtest():
print_('Share results: %s://www.speedtest.net/result/%s.png' % print_('Share results: %s://www.speedtest.net/result/%s.png' %
(scheme, resultid[0])) (scheme, resultid[0]))
if json_available and args.json:
print_(json.dumps(json_result))
def main(): def main():
try: try: