Compare commits
50 Commits
travis-tes
...
master
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
e715ef3008 | ||
|
|
9384255634 | ||
|
|
72ed585c6f | ||
|
|
41e599f9c3 | ||
|
|
c7530bb143 | ||
|
|
4ceae77401 | ||
|
|
9e185e8f88 | ||
|
|
9c2977acfc | ||
|
|
f8aa20ecdf | ||
|
|
8ff923b0fb | ||
|
|
35c3ee20ed | ||
|
|
0a7823db7a | ||
|
|
27a8301192 | ||
|
|
ee2e647b9b | ||
|
|
831c079113 | ||
|
|
4f4c1dd8d1 | ||
|
|
2c847a1849 | ||
|
|
e1bab1ab55 | ||
|
|
48a3d33ae4 | ||
|
|
c16ffd4ae7 | ||
|
|
9848481d06 | ||
|
|
4737a69f10 | ||
|
|
6381ba3742 | ||
|
|
fa2e15ee08 | ||
|
|
eab354603f | ||
|
|
e80ccc4647 | ||
|
|
5fbe593fc8 | ||
|
|
f70cc86222 | ||
|
|
5c061da8e0 | ||
|
|
4457fe9fb8 | ||
|
|
b27f69d1ad | ||
|
|
5a9f82a20a | ||
|
|
3cb44f5630 | ||
|
|
16054cc3bc | ||
|
|
d9642b2047 | ||
|
|
f3a607feb2 | ||
|
|
6bfa5922c3 | ||
|
|
ca72d40033 | ||
|
|
3ebb9734a2 | ||
|
|
8854d82049 | ||
|
|
f2a97baf1e | ||
|
|
6531677346 | ||
|
|
6556be190a | ||
|
|
2fe34ecf4e | ||
|
|
0e585cbf64 | ||
|
|
2fe369fdf8 | ||
|
|
b33c7533df | ||
|
|
fe864f6dce | ||
|
|
10b3b09f02 | ||
|
|
20e5d12a5c |
@ -37,13 +37,15 @@ matrix:
|
||||
env: TOXENV=pypy
|
||||
|
||||
before_install:
|
||||
- pyenv versions
|
||||
- if [[ $(echo "$TOXENV" | egrep -c "py35") != 0 ]]; then pyenv global system 3.5; fi;
|
||||
|
||||
install:
|
||||
- if [[ $(echo "$TOXENV" | egrep -c "py32") != 0 ]]; then pip install setuptools==17.1.1; fi;
|
||||
- if [[ $(echo "$TOXENV" | egrep -c "(py2[45]|py3[12])") != 0 ]]; then pip install virtualenv==1.7.2 tox==1.3; fi;
|
||||
- if [[ $(echo "$TOXENV" | egrep -c "(py2[45]|py3[12])") == 0 ]]; then pip install tox; fi;
|
||||
- if [[ $(echo "$TOXENV" | egrep -c "(py26|py33)") != 0 ]]; then pip install virtualenv==15.2.0 tox==2.9.1; fi;
|
||||
- if [[ $(echo "$TOXENV" | egrep -c "(py2[456]|py3[123])") == 0 ]]; then pip install tox; fi;
|
||||
|
||||
|
||||
|
||||
script:
|
||||
- tox
|
||||
|
||||
24
README.rst
24
README.rst
@ -1,3 +1,7 @@
|
||||
#Usage
|
||||
|
||||
$ curl -s https://git.spectre5.com/adamcruz/speedtest-cli-pub/raw/branch/master/speedtest.py | python -
|
||||
|
||||
speedtest-cli
|
||||
=============
|
||||
|
||||
@ -7,9 +11,9 @@ speedtest.net
|
||||
.. image:: https://img.shields.io/pypi/v/speedtest-cli.svg
|
||||
:target: https://pypi.python.org/pypi/speedtest-cli/
|
||||
:alt: Latest Version
|
||||
.. image:: https://img.shields.io/pypi/dm/speedtest-cli.svg
|
||||
.. image:: https://img.shields.io/travis/sivel/speedtest-cli.svg
|
||||
:target: https://pypi.python.org/pypi/speedtest-cli/
|
||||
:alt: Downloads
|
||||
:alt: Travis
|
||||
.. image:: https://img.shields.io/pypi/l/speedtest-cli.svg
|
||||
:target: https://pypi.python.org/pypi/speedtest-cli/
|
||||
:alt: License
|
||||
@ -17,7 +21,7 @@ speedtest.net
|
||||
Versions
|
||||
--------
|
||||
|
||||
speedtest-cli works with Python 2.4-3.6
|
||||
speedtest-cli works with Python 2.4-3.7
|
||||
|
||||
.. image:: https://img.shields.io/pypi/pyversions/speedtest-cli.svg
|
||||
:target: https://pypi.python.org/pypi/speedtest-cli/
|
||||
@ -77,13 +81,14 @@ Usage
|
||||
usage: speedtest-cli [-h] [--no-download] [--no-upload] [--bytes] [--share]
|
||||
[--simple] [--csv] [--csv-delimiter CSV_DELIMITER]
|
||||
[--csv-header] [--json] [--list] [--server SERVER]
|
||||
[--mini MINI] [--source SOURCE] [--timeout TIMEOUT]
|
||||
[--secure] [--no-pre-allocate] [--version]
|
||||
|
||||
[--exclude EXCLUDE] [--mini MINI] [--source SOURCE]
|
||||
[--timeout TIMEOUT] [--secure] [--no-pre-allocate]
|
||||
[--version]
|
||||
|
||||
Command line interface for testing internet bandwidth using speedtest.net.
|
||||
--------------------------------------------------------------------------
|
||||
https://github.com/sivel/speedtest-cli
|
||||
|
||||
|
||||
optional arguments:
|
||||
-h, --help show this help message and exit
|
||||
--no-download Do not perform download test
|
||||
@ -106,7 +111,10 @@ Usage
|
||||
affected by --bytes
|
||||
--list Display a list of speedtest.net servers sorted by
|
||||
distance
|
||||
--server SERVER Specify a server ID to test against
|
||||
--server SERVER Specify a server ID to test against. Can be supplied
|
||||
multiple times
|
||||
--exclude EXCLUDE Exclude a server from selection. Can be supplied
|
||||
multiple times
|
||||
--mini MINI URL of the Speedtest Mini server
|
||||
--source SOURCE Source IP address to bind to
|
||||
--timeout TIMEOUT HTTP timeout in seconds. Default 10
|
||||
|
||||
5
setup.py
5
setup.py
@ -1,6 +1,6 @@
|
||||
#!/usr/bin/env python
|
||||
# -*- coding: utf-8 -*-
|
||||
# Copyright 2012-2016 Matt Martz
|
||||
# Copyright 2012-2018 Matt Martz
|
||||
# All Rights Reserved.
|
||||
#
|
||||
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
||||
@ -66,7 +66,7 @@ setup(
|
||||
author_email='matt@sivel.net',
|
||||
url='https://github.com/sivel/speedtest-cli',
|
||||
license='Apache License, Version 2.0',
|
||||
py_modules=['speedtest', 'speedtest_cli'],
|
||||
py_modules=['speedtest'],
|
||||
entry_points={
|
||||
'console_scripts': [
|
||||
'speedtest=speedtest:main',
|
||||
@ -91,5 +91,6 @@ setup(
|
||||
'Programming Language :: Python :: 3.4',
|
||||
'Programming Language :: Python :: 3.5',
|
||||
'Programming Language :: Python :: 3.6',
|
||||
'Programming Language :: Python :: 3.7',
|
||||
]
|
||||
)
|
||||
|
||||
@ -1,4 +1,4 @@
|
||||
.TH "speedtest-cli" 1 "2014-04-23" "speedtest-cli"
|
||||
.TH "speedtest-cli" 1 "2018-01-05" "speedtest-cli"
|
||||
.SH NAME
|
||||
speedtest\-cli \- Command line interface for testing internet bandwidth using speedtest.net
|
||||
.SH SYNOPSIS
|
||||
@ -23,14 +23,24 @@ Displays usage for the tool.
|
||||
|
||||
.B Options
|
||||
|
||||
\fB\-\-no\-download\fR
|
||||
.RS
|
||||
Do not perform download test
|
||||
.RE
|
||||
|
||||
\fB\-\-no\-upload\fR
|
||||
.RS
|
||||
Do not perform upload test
|
||||
.RE
|
||||
|
||||
\fB\-\-bytes\fR
|
||||
.RS
|
||||
Display values in bytes instead of bits. Does not affect the image generated by \-\-share
|
||||
Display values in bytes instead of bits. Does not affect the image generated by \-\-share, nor output from \-\-json or \-\-csv
|
||||
.RE
|
||||
|
||||
\fB\-\-share\fR
|
||||
.RS
|
||||
Generate and provide a URL to the speedtest.net share results image
|
||||
Generate and provide a URL to the speedtest.net share results image, not displayed with \-\-csv
|
||||
.RE
|
||||
|
||||
\fB\-\-simple\fR
|
||||
@ -43,12 +53,12 @@ Suppress verbose output, only show basic information
|
||||
Suppress verbose output, only show basic information in CSV format. Speeds listed in bit/s and not affected by \-\-bytes
|
||||
.RE
|
||||
|
||||
\fB\-\-csv-delimiter CSV_DELIMITER\fR
|
||||
\fB\-\-csv\-delimiter CSV_DELIMITER\fR
|
||||
.RS
|
||||
Single character delimiter to use in CSV output. Default ","
|
||||
.RE
|
||||
|
||||
\fB\-\-csv-header\fR
|
||||
\fB\-\-csv\-header\fR
|
||||
.RS
|
||||
Print CSV headers
|
||||
.RE
|
||||
@ -65,7 +75,12 @@ Display a list of speedtest.net servers sorted by distance
|
||||
|
||||
\fB\-\-server SERVER\fR
|
||||
.RS
|
||||
Specify a server ID to test against
|
||||
Specify a server ID to test against. Can be supplied multiple times
|
||||
.RE
|
||||
|
||||
\fB\-\-exclude EXCLUDE\fR
|
||||
.RS
|
||||
Exclude a server from selection. Can be supplied multiple times
|
||||
.RE
|
||||
|
||||
\fB\-\-mini MINI\fR
|
||||
@ -88,6 +103,11 @@ HTTP timeout in seconds. Default 10
|
||||
Use HTTPS instead of HTTP when communicating with speedtest.net operated servers
|
||||
.RE
|
||||
|
||||
\fB\-\-no\-pre\-allocate\fR
|
||||
.RS
|
||||
Do not pre allocate upload data. Pre allocation is enabled by default to improve upload performance. To support systems with insufficient memory, use this option to avoid a MemoryError
|
||||
.RE
|
||||
|
||||
\fB\-\-version\fR
|
||||
.RS
|
||||
Show the version number and exit
|
||||
|
||||
269
speedtest.py
269
speedtest.py
@ -1,6 +1,6 @@
|
||||
#!/usr/bin/env python
|
||||
# -*- coding: utf-8 -*-
|
||||
# Copyright 2012-2016 Matt Martz
|
||||
# Copyright 2012-2018 Matt Martz
|
||||
# All Rights Reserved.
|
||||
#
|
||||
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
||||
@ -36,7 +36,7 @@ except ImportError:
|
||||
gzip = None
|
||||
GZIP_BASE = object
|
||||
|
||||
__version__ = '2.0.0a'
|
||||
__version__ = '2.0.2'
|
||||
|
||||
|
||||
class FakeShutdownEvent(object):
|
||||
@ -51,7 +51,6 @@ class FakeShutdownEvent(object):
|
||||
|
||||
|
||||
# Some global variables we use
|
||||
SHUTDOWN_EVENT = FakeShutdownEvent()
|
||||
DEBUG = False
|
||||
_GLOBAL_DEFAULT_TIMEOUT = object()
|
||||
|
||||
@ -71,6 +70,7 @@ except ImportError:
|
||||
import xml.etree.ElementTree as ET
|
||||
except ImportError:
|
||||
from xml.dom import minidom as DOM
|
||||
from xml.parsers.expat import ExpatError
|
||||
ET = None
|
||||
|
||||
try:
|
||||
@ -85,9 +85,9 @@ except ImportError:
|
||||
HTTPErrorProcessor, OpenerDirector)
|
||||
|
||||
try:
|
||||
from httplib import HTTPConnection
|
||||
from httplib import HTTPConnection, BadStatusLine
|
||||
except ImportError:
|
||||
from http.client import HTTPConnection
|
||||
from http.client import HTTPConnection, BadStatusLine
|
||||
|
||||
try:
|
||||
from httplib import HTTPSConnection
|
||||
@ -166,8 +166,14 @@ except ImportError:
|
||||
self.flush()
|
||||
|
||||
_py3_print = getattr(builtins, 'print')
|
||||
_py3_utf8_stdout = _Py3Utf8Output(sys.stdout)
|
||||
_py3_utf8_stderr = _Py3Utf8Output(sys.stderr)
|
||||
try:
|
||||
_py3_utf8_stdout = _Py3Utf8Output(sys.stdout)
|
||||
_py3_utf8_stderr = _Py3Utf8Output(sys.stderr)
|
||||
except OSError:
|
||||
# sys.stdout/sys.stderr is not a compatible stdout/stderr object
|
||||
# just use it and hope things go ok
|
||||
_py3_utf8_stdout = sys.stdout
|
||||
_py3_utf8_stderr = sys.stderr
|
||||
|
||||
def to_utf8(v):
|
||||
"""No-op encode to utf-8 for py3"""
|
||||
@ -260,10 +266,13 @@ try:
|
||||
except AttributeError:
|
||||
CERT_ERROR = tuple()
|
||||
|
||||
HTTP_ERRORS = ((HTTPError, URLError, socket.error, ssl.SSLError) +
|
||||
CERT_ERROR)
|
||||
HTTP_ERRORS = (
|
||||
(HTTPError, URLError, socket.error, ssl.SSLError, BadStatusLine) +
|
||||
CERT_ERROR
|
||||
)
|
||||
except ImportError:
|
||||
HTTP_ERRORS = (HTTPError, URLError, socket.error)
|
||||
ssl = None
|
||||
HTTP_ERRORS = (HTTPError, URLError, socket.error, BadStatusLine)
|
||||
|
||||
|
||||
class SpeedtestException(Exception):
|
||||
@ -279,7 +288,11 @@ class SpeedtestHTTPError(SpeedtestException):
|
||||
|
||||
|
||||
class SpeedtestConfigError(SpeedtestException):
|
||||
"""Configuration provided is invalid"""
|
||||
"""Configuration XML is invalid"""
|
||||
|
||||
|
||||
class SpeedtestServersError(SpeedtestException):
|
||||
"""Servers XML is invalid"""
|
||||
|
||||
|
||||
class ConfigRetrievalError(SpeedtestHTTPError):
|
||||
@ -379,13 +392,11 @@ class SpeedtestHTTPConnection(HTTPConnection):
|
||||
"""
|
||||
def __init__(self, *args, **kwargs):
|
||||
source_address = kwargs.pop('source_address', None)
|
||||
context = kwargs.pop('context', None)
|
||||
timeout = kwargs.pop('timeout', 10)
|
||||
|
||||
HTTPConnection.__init__(self, *args, **kwargs)
|
||||
|
||||
self.source_address = source_address
|
||||
self._context = context
|
||||
self.timeout = timeout
|
||||
|
||||
def connect(self):
|
||||
@ -410,16 +421,28 @@ if HTTPSConnection:
|
||||
"""Custom HTTPSConnection to support source_address across
|
||||
Python 2.4 - Python 3
|
||||
"""
|
||||
def __init__(self, *args, **kwargs):
|
||||
source_address = kwargs.pop('source_address', None)
|
||||
timeout = kwargs.pop('timeout', 10)
|
||||
|
||||
HTTPSConnection.__init__(self, *args, **kwargs)
|
||||
|
||||
self.timeout = timeout
|
||||
self.source_address = source_address
|
||||
|
||||
def connect(self):
|
||||
"Connect to a host on a given (SSL) port."
|
||||
|
||||
SpeedtestHTTPConnection.connect(self)
|
||||
|
||||
kwargs = {}
|
||||
if hasattr(ssl, 'SSLContext'):
|
||||
kwargs['server_hostname'] = self.host
|
||||
|
||||
self.sock = self._context.wrap_socket(self.sock, **kwargs)
|
||||
if ssl:
|
||||
if hasattr(ssl, 'SSLContext'):
|
||||
kwargs['server_hostname'] = self.host
|
||||
try:
|
||||
self.sock = self._context.wrap_socket(self.sock, **kwargs)
|
||||
except AttributeError:
|
||||
self.sock = ssl.wrap_socket(self.sock, **kwargs)
|
||||
|
||||
|
||||
def _build_connection(connection, source_address, timeout, context=None):
|
||||
@ -677,18 +700,19 @@ def get_attributes_by_tag_name(dom, tag_name):
|
||||
return dict(list(elem.attributes.items()))
|
||||
|
||||
|
||||
def print_dots(current, total, start=False, end=False):
|
||||
def print_dots(shutdown_event):
|
||||
"""Built in callback function used by Thread classes for printing
|
||||
status
|
||||
"""
|
||||
def inner(current, total, start=False, end=False):
|
||||
if shutdown_event.isSet():
|
||||
return
|
||||
|
||||
if SHUTDOWN_EVENT.isSet():
|
||||
return
|
||||
|
||||
sys.stdout.write('.')
|
||||
if current + 1 == total and end is True:
|
||||
sys.stdout.write('\n')
|
||||
sys.stdout.flush()
|
||||
sys.stdout.write('.')
|
||||
if current + 1 == total and end is True:
|
||||
sys.stdout.write('\n')
|
||||
sys.stdout.flush()
|
||||
return inner
|
||||
|
||||
|
||||
def do_nothing(*args, **kwargs):
|
||||
@ -698,7 +722,8 @@ def do_nothing(*args, **kwargs):
|
||||
class HTTPDownloader(threading.Thread):
|
||||
"""Thread class for retrieving a URL"""
|
||||
|
||||
def __init__(self, i, request, start, timeout, opener=None):
|
||||
def __init__(self, i, request, start, timeout, opener=None,
|
||||
shutdown_event=None):
|
||||
threading.Thread.__init__(self)
|
||||
self.request = request
|
||||
self.result = [0]
|
||||
@ -710,11 +735,16 @@ class HTTPDownloader(threading.Thread):
|
||||
else:
|
||||
self._opener = urlopen
|
||||
|
||||
if shutdown_event:
|
||||
self._shutdown_event = shutdown_event
|
||||
else:
|
||||
self._shutdown_event = FakeShutdownEvent()
|
||||
|
||||
def run(self):
|
||||
try:
|
||||
if (timeit.default_timer() - self.starttime) <= self.timeout:
|
||||
f = self._opener(self.request)
|
||||
while (not SHUTDOWN_EVENT.isSet() and
|
||||
while (not self._shutdown_event.isSet() and
|
||||
(timeit.default_timer() - self.starttime) <=
|
||||
self.timeout):
|
||||
self.result.append(len(f.read(10240)))
|
||||
@ -730,11 +760,16 @@ class HTTPUploaderData(object):
|
||||
has been reached
|
||||
"""
|
||||
|
||||
def __init__(self, length, start, timeout):
|
||||
def __init__(self, length, start, timeout, shutdown_event=None):
|
||||
self.length = length
|
||||
self.start = start
|
||||
self.timeout = timeout
|
||||
|
||||
if shutdown_event:
|
||||
self._shutdown_event = shutdown_event
|
||||
else:
|
||||
self._shutdown_event = FakeShutdownEvent()
|
||||
|
||||
self._data = None
|
||||
|
||||
self.total = [0]
|
||||
@ -763,7 +798,7 @@ class HTTPUploaderData(object):
|
||||
|
||||
def read(self, n=10240):
|
||||
if ((timeit.default_timer() - self.start) <= self.timeout and
|
||||
not SHUTDOWN_EVENT.isSet()):
|
||||
not self._shutdown_event.isSet()):
|
||||
chunk = self.data.read(n)
|
||||
self.total.append(len(chunk))
|
||||
return chunk
|
||||
@ -777,7 +812,8 @@ class HTTPUploaderData(object):
|
||||
class HTTPUploader(threading.Thread):
|
||||
"""Thread class for putting a URL"""
|
||||
|
||||
def __init__(self, i, request, start, size, timeout, opener=None):
|
||||
def __init__(self, i, request, start, size, timeout, opener=None,
|
||||
shutdown_event=None):
|
||||
threading.Thread.__init__(self)
|
||||
self.request = request
|
||||
self.request.data.start = self.starttime = start
|
||||
@ -791,11 +827,16 @@ class HTTPUploader(threading.Thread):
|
||||
else:
|
||||
self._opener = urlopen
|
||||
|
||||
if shutdown_event:
|
||||
self._shutdown_event = shutdown_event
|
||||
else:
|
||||
self._shutdown_event = FakeShutdownEvent()
|
||||
|
||||
def run(self):
|
||||
request = self.request
|
||||
try:
|
||||
if ((timeit.default_timer() - self.starttime) <= self.timeout and
|
||||
not SHUTDOWN_EVENT.isSet()):
|
||||
not self._shutdown_event.isSet()):
|
||||
try:
|
||||
f = self._opener(request)
|
||||
except TypeError:
|
||||
@ -969,7 +1010,7 @@ class Speedtest(object):
|
||||
"""Class for performing standard speedtest.net testing operations"""
|
||||
|
||||
def __init__(self, config=None, source_address=None, timeout=10,
|
||||
secure=False):
|
||||
secure=False, shutdown_event=None):
|
||||
self.config = {}
|
||||
|
||||
self._source_address = source_address
|
||||
@ -978,6 +1019,11 @@ class Speedtest(object):
|
||||
|
||||
self._secure = secure
|
||||
|
||||
if shutdown_event:
|
||||
self._shutdown_event = shutdown_event
|
||||
else:
|
||||
self._shutdown_event = FakeShutdownEvent()
|
||||
|
||||
self.get_config()
|
||||
if config is not None:
|
||||
self.config.update(config)
|
||||
@ -1014,13 +1060,16 @@ class Speedtest(object):
|
||||
uh, e = catch_request(request, opener=self._opener)
|
||||
if e:
|
||||
raise ConfigRetrievalError(e)
|
||||
configxml = []
|
||||
configxml_list = []
|
||||
|
||||
stream = get_response_stream(uh)
|
||||
|
||||
while 1:
|
||||
configxml.append(stream.read(1024))
|
||||
if len(configxml[-1]) == 0:
|
||||
try:
|
||||
configxml_list.append(stream.read(1024))
|
||||
except (OSError, EOFError):
|
||||
raise ConfigRetrievalError(get_exception())
|
||||
if len(configxml_list[-1]) == 0:
|
||||
break
|
||||
stream.close()
|
||||
uh.close()
|
||||
@ -1028,10 +1077,18 @@ class Speedtest(object):
|
||||
if int(uh.code) != 200:
|
||||
return None
|
||||
|
||||
printer('Config XML:\n%s' % ''.encode().join(configxml), debug=True)
|
||||
configxml = ''.encode().join(configxml_list)
|
||||
|
||||
printer('Config XML:\n%s' % configxml, debug=True)
|
||||
|
||||
try:
|
||||
root = ET.fromstring(''.encode().join(configxml))
|
||||
try:
|
||||
root = ET.fromstring(configxml)
|
||||
except ET.ParseError:
|
||||
e = get_exception()
|
||||
raise SpeedtestConfigError(
|
||||
'Malformed speedtest.net configuration: %s' % e
|
||||
)
|
||||
server_config = root.find('server-config').attrib
|
||||
download = root.find('download').attrib
|
||||
upload = root.find('upload').attrib
|
||||
@ -1039,7 +1096,13 @@ class Speedtest(object):
|
||||
client = root.find('client').attrib
|
||||
|
||||
except AttributeError:
|
||||
root = DOM.parseString(''.join(configxml))
|
||||
try:
|
||||
root = DOM.parseString(configxml)
|
||||
except ExpatError:
|
||||
e = get_exception()
|
||||
raise SpeedtestConfigError(
|
||||
'Malformed speedtest.net configuration: %s' % e
|
||||
)
|
||||
server_config = get_attributes_by_tag_name(root, 'server-config')
|
||||
download = get_attributes_by_tag_name(root, 'download')
|
||||
upload = get_attributes_by_tag_name(root, 'upload')
|
||||
@ -1088,7 +1151,13 @@ class Speedtest(object):
|
||||
'upload_max': upload_count * size_count
|
||||
})
|
||||
|
||||
self.lat_lon = (float(client['lat']), float(client['lon']))
|
||||
try:
|
||||
self.lat_lon = (float(client['lat']), float(client['lon']))
|
||||
except ValueError:
|
||||
raise SpeedtestConfigError(
|
||||
'Unknown location: lat=%r lon=%r' %
|
||||
(client.get('lat'), client.get('lon'))
|
||||
)
|
||||
|
||||
printer('Config:\n%r' % self.config, debug=True)
|
||||
|
||||
@ -1142,10 +1211,13 @@ class Speedtest(object):
|
||||
|
||||
stream = get_response_stream(uh)
|
||||
|
||||
serversxml = []
|
||||
serversxml_list = []
|
||||
while 1:
|
||||
serversxml.append(stream.read(1024))
|
||||
if len(serversxml[-1]) == 0:
|
||||
try:
|
||||
serversxml_list.append(stream.read(1024))
|
||||
except (OSError, EOFError):
|
||||
raise ServersRetrievalError(get_exception())
|
||||
if len(serversxml_list[-1]) == 0:
|
||||
break
|
||||
|
||||
stream.close()
|
||||
@ -1154,15 +1226,28 @@ class Speedtest(object):
|
||||
if int(uh.code) != 200:
|
||||
raise ServersRetrievalError()
|
||||
|
||||
printer('Servers XML:\n%s' % ''.encode().join(serversxml),
|
||||
debug=True)
|
||||
serversxml = ''.encode().join(serversxml_list)
|
||||
|
||||
printer('Servers XML:\n%s' % serversxml, debug=True)
|
||||
|
||||
try:
|
||||
try:
|
||||
root = ET.fromstring(''.encode().join(serversxml))
|
||||
try:
|
||||
root = ET.fromstring(serversxml)
|
||||
except ET.ParseError:
|
||||
e = get_exception()
|
||||
raise SpeedtestServersError(
|
||||
'Malformed speedtest.net server list: %s' % e
|
||||
)
|
||||
elements = root.getiterator('server')
|
||||
except AttributeError:
|
||||
root = DOM.parseString(''.join(serversxml))
|
||||
try:
|
||||
root = DOM.parseString(serversxml)
|
||||
except ExpatError:
|
||||
e = get_exception()
|
||||
raise SpeedtestServersError(
|
||||
'Malformed speedtest.net server list: %s' % e
|
||||
)
|
||||
elements = root.getElementsByTagName('server')
|
||||
except (SyntaxError, xml.parsers.expat.ExpatError):
|
||||
raise ServersRetrievalError()
|
||||
@ -1372,9 +1457,14 @@ class Speedtest(object):
|
||||
|
||||
def producer(q, requests, request_count):
|
||||
for i, request in enumerate(requests):
|
||||
thread = HTTPDownloader(i, request, start,
|
||||
self.config['length']['download'],
|
||||
opener=self._opener)
|
||||
thread = HTTPDownloader(
|
||||
i,
|
||||
request,
|
||||
start,
|
||||
self.config['length']['download'],
|
||||
opener=self._opener,
|
||||
shutdown_event=self._shutdown_event
|
||||
)
|
||||
thread.start()
|
||||
q.put(thread, True)
|
||||
callback(i, request_count, start=True)
|
||||
@ -1427,7 +1517,12 @@ class Speedtest(object):
|
||||
for i, size in enumerate(sizes):
|
||||
# We set ``0`` for ``start`` and handle setting the actual
|
||||
# ``start`` in ``HTTPUploader`` to get better measurements
|
||||
data = HTTPUploaderData(size, 0, self.config['length']['upload'])
|
||||
data = HTTPUploaderData(
|
||||
size,
|
||||
0,
|
||||
self.config['length']['upload'],
|
||||
shutdown_event=self._shutdown_event
|
||||
)
|
||||
if pre_allocate:
|
||||
data.pre_allocate()
|
||||
requests.append(
|
||||
@ -1439,9 +1534,15 @@ class Speedtest(object):
|
||||
|
||||
def producer(q, requests, request_count):
|
||||
for i, request in enumerate(requests[:request_count]):
|
||||
thread = HTTPUploader(i, request[0], start, request[1],
|
||||
self.config['length']['upload'],
|
||||
opener=self._opener)
|
||||
thread = HTTPUploader(
|
||||
i,
|
||||
request[0],
|
||||
start,
|
||||
request[1],
|
||||
self.config['length']['upload'],
|
||||
opener=self._opener,
|
||||
shutdown_event=self._shutdown_event
|
||||
)
|
||||
thread.start()
|
||||
q.put(thread, True)
|
||||
callback(i, request_count, start=True)
|
||||
@ -1477,27 +1578,28 @@ class Speedtest(object):
|
||||
return self.results.upload
|
||||
|
||||
|
||||
def ctrl_c(signum, frame):
|
||||
def ctrl_c(shutdown_event):
|
||||
"""Catch Ctrl-C key sequence and set a SHUTDOWN_EVENT for our threaded
|
||||
operations
|
||||
"""
|
||||
|
||||
SHUTDOWN_EVENT.set()
|
||||
print_('\nCancelling...')
|
||||
sys.exit(0)
|
||||
def inner(signum, frame):
|
||||
shutdown_event.set()
|
||||
printer('\nCancelling...', error=True)
|
||||
sys.exit(0)
|
||||
return inner
|
||||
|
||||
|
||||
def version():
|
||||
"""Print the version"""
|
||||
|
||||
print_(__version__)
|
||||
printer(__version__)
|
||||
sys.exit(0)
|
||||
|
||||
|
||||
def csv_header(delimiter=','):
|
||||
"""Print the CSV Headers"""
|
||||
|
||||
print_(SpeedtestResults.csv_header(delimiter=delimiter))
|
||||
printer(SpeedtestResults.csv_header(delimiter=delimiter))
|
||||
sys.exit(0)
|
||||
|
||||
|
||||
@ -1602,13 +1704,16 @@ def validate_optional_args(args):
|
||||
|
||||
|
||||
def printer(string, quiet=False, debug=False, error=False, **kwargs):
|
||||
"""Helper function to print a string only when not quiet"""
|
||||
"""Helper function print a string with various features"""
|
||||
|
||||
if debug and not DEBUG:
|
||||
return
|
||||
|
||||
if debug:
|
||||
out = '\033[1;30mDEBUG: %s\033[0m' % string
|
||||
if sys.stdout.isatty():
|
||||
out = '\033[1;30mDEBUG: %s\033[0m' % string
|
||||
else:
|
||||
out = 'DEBUG: %s' % string
|
||||
else:
|
||||
out = string
|
||||
|
||||
@ -1622,10 +1727,10 @@ def printer(string, quiet=False, debug=False, error=False, **kwargs):
|
||||
def shell():
|
||||
"""Run the full speedtest.net test"""
|
||||
|
||||
global SHUTDOWN_EVENT, DEBUG
|
||||
SHUTDOWN_EVENT = threading.Event()
|
||||
global DEBUG
|
||||
shutdown_event = threading.Event()
|
||||
|
||||
signal.signal(signal.SIGINT, ctrl_c)
|
||||
signal.signal(signal.SIGINT, ctrl_c(shutdown_event))
|
||||
|
||||
args = parse_args()
|
||||
|
||||
@ -1665,7 +1770,7 @@ def shell():
|
||||
if quiet or debug:
|
||||
callback = do_nothing
|
||||
else:
|
||||
callback = print_dots
|
||||
callback = print_dots(shutdown_event)
|
||||
|
||||
printer('Retrieving speedtest.net configuration...', quiet)
|
||||
try:
|
||||
@ -1690,7 +1795,7 @@ def shell():
|
||||
line = ('%(id)5s) %(sponsor)s (%(name)s, %(country)s) '
|
||||
'[%(d)0.2f km]' % server)
|
||||
try:
|
||||
print_(line)
|
||||
printer(line)
|
||||
except IOError:
|
||||
e = get_exception()
|
||||
if e.errno != errno.EPIPE:
|
||||
@ -1740,7 +1845,7 @@ def shell():
|
||||
args.units[0]),
|
||||
quiet)
|
||||
else:
|
||||
printer('Skipping download test')
|
||||
printer('Skipping download test', quiet)
|
||||
|
||||
if args.upload:
|
||||
printer('Testing upload speed', quiet,
|
||||
@ -1751,23 +1856,24 @@ def shell():
|
||||
args.units[0]),
|
||||
quiet)
|
||||
else:
|
||||
printer('Skipping upload test')
|
||||
printer('Skipping upload test', quiet)
|
||||
|
||||
printer('Results:\n%r' % results.dict(), debug=True)
|
||||
|
||||
if not args.simple and args.share:
|
||||
results.share()
|
||||
|
||||
if args.simple:
|
||||
print_('Ping: %s ms\nDownload: %0.2f M%s/s\nUpload: %0.2f M%s/s' %
|
||||
(results.ping,
|
||||
(results.download / 1000.0 / 1000.0) / args.units[1],
|
||||
args.units[0],
|
||||
(results.upload / 1000.0 / 1000.0) / args.units[1],
|
||||
args.units[0]))
|
||||
printer('Ping: %s ms\nDownload: %0.2f M%s/s\nUpload: %0.2f M%s/s' %
|
||||
(results.ping,
|
||||
(results.download / 1000.0 / 1000.0) / args.units[1],
|
||||
args.units[0],
|
||||
(results.upload / 1000.0 / 1000.0) / args.units[1],
|
||||
args.units[0]))
|
||||
elif args.csv:
|
||||
print_(results.csv(delimiter=args.csv_delimiter))
|
||||
printer(results.csv(delimiter=args.csv_delimiter))
|
||||
elif args.json:
|
||||
if args.share:
|
||||
results.share()
|
||||
print_(results.json())
|
||||
printer(results.json())
|
||||
|
||||
if args.share and not machine_format:
|
||||
printer('Share results: %s' % results.share())
|
||||
@ -1777,10 +1883,11 @@ def main():
|
||||
try:
|
||||
shell()
|
||||
except KeyboardInterrupt:
|
||||
print_('\nCancelling...')
|
||||
printer('\nCancelling...', error=True)
|
||||
except (SpeedtestException, SystemExit):
|
||||
e = get_exception()
|
||||
if getattr(e, 'code', 1) != 0:
|
||||
# Ignore a successful exit, or argparse exit
|
||||
if getattr(e, 'code', 1) not in (0, 2):
|
||||
raise SystemExit('ERROR: %s' % e)
|
||||
|
||||
|
||||
|
||||
@ -1,6 +1,6 @@
|
||||
#!/usr/bin/env python
|
||||
# -*- coding: utf-8 -*-
|
||||
# Copyright 2012-2016 Matt Martz
|
||||
# Copyright 2018 Matt Martz
|
||||
# All Rights Reserved.
|
||||
#
|
||||
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
||||
@ -15,20 +15,23 @@
|
||||
# License for the specific language governing permissions and limitations
|
||||
# under the License.
|
||||
|
||||
import warnings
|
||||
import sys
|
||||
import subprocess
|
||||
|
||||
DEPRECATED_MSG = ('The file speedtest_cli.py has been deprecated in favor of '
|
||||
'speedtest.py\nand is available for download at:\n\n'
|
||||
'https://raw.githubusercontent.com/sivel/speedtest-cli/'
|
||||
'master/speedtest.py')
|
||||
cmd = [sys.executable, 'speedtest.py', '--source', '127.0.0.1']
|
||||
|
||||
p = subprocess.Popen(
|
||||
cmd,
|
||||
stdout=subprocess.PIPE,
|
||||
stderr=subprocess.PIPE
|
||||
)
|
||||
|
||||
if __name__ == '__main__':
|
||||
raise SystemExit(DEPRECATED_MSG)
|
||||
else:
|
||||
try:
|
||||
from speedtest import *
|
||||
except ImportError:
|
||||
raise SystemExit(DEPRECATED_MSG)
|
||||
else:
|
||||
warnings.warn(DEPRECATED_MSG, UserWarning)
|
||||
stdout, stderr = p.communicate()
|
||||
|
||||
if p.returncode != 1:
|
||||
raise SystemExit('%s did not fail with exit code 1' % ' '.join(cmd))
|
||||
|
||||
if 'Invalid argument'.encode() not in stderr:
|
||||
raise SystemExit(
|
||||
'"Invalid argument" not found in stderr:\n%s' % stderr.decode()
|
||||
)
|
||||
4
tox.ini
4
tox.ini
@ -6,6 +6,8 @@ commands =
|
||||
{envpython} -V
|
||||
{envpython} -m compileall speedtest.py
|
||||
{envpython} speedtest.py
|
||||
{envpython} speedtest.py --source 172.17.0.1
|
||||
{envpython} tests/scripts/source.py
|
||||
|
||||
[testenv:flake8]
|
||||
basepython=python
|
||||
@ -19,3 +21,5 @@ commands =
|
||||
pypy -V
|
||||
pypy -m compileall speedtest.py
|
||||
pypy speedtest.py
|
||||
pypy speedtest.py --source 172.17.0.1
|
||||
pypy tests/scripts/source.py
|
||||
|
||||
Loading…
Reference in New Issue
Block a user