Compare commits
185 Commits
Author | SHA1 | Date |
---|---|---|
George Sokianos | a8f034ef62 | |
Matt Martz | 22210ca352 | |
Matt Martz | 42e96b13dd | |
Matt Martz | cadc68b5ae | |
Matt Martz | db46af8bcd | |
Matt Martz | c58ad3367b | |
Matt Martz | 266e53c256 | |
Matt Martz | 7ebb9965dd | |
Matt Martz | 2658bd50b4 | |
Matt Martz | 81bba6070c | |
Matt Martz | 681cdf20a5 | |
Matt Martz | cdf6002865 | |
Matt Martz | 9af203652b | |
Matt Martz | 2d5a9ef364 | |
Matt Martz | 3109fcf407 | |
Matt Martz | 69ddff1a11 | |
Matt Martz | fb0569946d | |
Matt Martz | f356c7b02d | |
Matt Martz | 6cf43b2ff7 | |
Matt Martz | 217ce8eff1 | |
Matt Martz | b43334f1ec | |
Matt Martz | b0b826c870 | |
Matt Martz | 9ac1091eae | |
Matt Martz | ca2250f700 | |
Matt Martz | ddb8db0c94 | |
Matt Martz | 72bf53affa | |
liuxu | a8a3265001 | |
Alex Ward | b2654de410 | |
Matt Martz | 72ed585c6f | |
Matt Martz | 41e599f9c3 | |
Matt Martz | c7530bb143 | |
Matt Martz | 4ceae77401 | |
Matt Martz | 9e185e8f88 | |
Matt Martz | 9c2977acfc | |
Matt Martz | f8aa20ecdf | |
Matt Martz | 8ff923b0fb | |
Matt Martz | 35c3ee20ed | |
Matt Martz | 0a7823db7a | |
Matt Martz | 27a8301192 | |
Matt Martz | ee2e647b9b | |
Matt Martz | 831c079113 | |
Matt Martz | 4f4c1dd8d1 | |
Matt Martz | 2c847a1849 | |
Matt Martz | e1bab1ab55 | |
Matt Martz | 48a3d33ae4 | |
Matt Martz | c16ffd4ae7 | |
Matt Martz | 9848481d06 | |
Matt Martz | 4737a69f10 | |
Matt Martz | 6381ba3742 | |
Matt Martz | fa2e15ee08 | |
Matt Martz | eab354603f | |
Matt Martz | e80ccc4647 | |
Matt Martz | 5fbe593fc8 | |
Matt Martz | f70cc86222 | |
Matt Martz | 5c061da8e0 | |
Matt Martz | 4457fe9fb8 | |
Matt Martz | b27f69d1ad | |
Matt Martz | 5a9f82a20a | |
Matt Martz | 3cb44f5630 | |
Matt Martz | 16054cc3bc | |
Matt Martz | d9642b2047 | |
Matt Martz | f3a607feb2 | |
Matt Martz | 6bfa5922c3 | |
Matt Martz | ca72d40033 | |
Matt Martz | 3ebb9734a2 | |
Matt Martz | 8854d82049 | |
Matt Martz | f2a97baf1e | |
Matt Martz | 6531677346 | |
Matt Martz | 6556be190a | |
Matt Martz | 2fe34ecf4e | |
Matt Martz | 0e585cbf64 | |
Matt Martz | 2fe369fdf8 | |
Matt Martz | b33c7533df | |
Matt Martz | fe864f6dce | |
Matt Martz | 10b3b09f02 | |
Matt Martz | 20e5d12a5c | |
Matt Martz | 6603954e45 | |
Matt Martz | e982830350 | |
Matt Martz | 2c89c53a79 | |
Matt Martz | 401c469991 | |
Matt Martz | 3c1c9d3179 | |
Matt Martz | e2f815618b | |
Matt Martz | 955a756c96 | |
Matt Martz | ceef55488c | |
Matt Martz | 20eeadcb0c | |
Matt Martz | 4aebe01c3e | |
Matt Martz | 1871b26b9a | |
Matt Martz | 824c584658 | |
Matt Martz | 9806e401e0 | |
Matt Martz | 1642d0669f | |
Matt Martz | 2e79fbf1dc | |
Matt Martz | f20e8808a8 | |
Matt Martz | 3feb38d9d4 | |
Matt Martz | d712f947d9 | |
Matt Martz | 55b3cf14a3 | |
Matt Martz | 33e498beb3 | |
Matt Martz | 068d71597b | |
Matt Martz | 1863c35f6b | |
Matt Martz | 823d7dc2b7 | |
Matt Martz | 411f1609e8 | |
Matt Martz | 7b38e264bc | |
Matt Martz | 2acba6ecd7 | |
Matt Martz | 53ef57bd5e | |
Matt Martz | c512684ffa | |
Matt Martz | 6685d91729 | |
Matt Martz | 91270dbc67 | |
Matt Martz | b75ecc291c | |
Matt Martz | 1d6717e714 | |
Matt Martz | d41cfc0cb1 | |
Matt Martz | 59880107a7 | |
Matt Martz | 4280c448cf | |
Matt Martz | 9f44a72fdb | |
Matt Martz | b075152e3e | |
Matt Martz | 2be4d0a5e7 | |
Matt Martz | 9299e0860c | |
Matt Martz | 292e250990 | |
Matt Martz | fd8b8cfa92 | |
Matt Martz | 9e3a5b3a59 | |
Matt Martz | 53b760dfba | |
Matt Martz | d0c927e8ae | |
Matt Martz | 6fffcd5b63 | |
Matt Martz | f88c41f97f | |
Matt Martz | 64b03777da | |
Matt Martz | 01abb3ae71 | |
Matt Martz | 884c7fed87 | |
Matt Martz | 38870b69ea | |
Matt Martz | 4bd4b7dfec | |
Matt Martz | 5e0bd05c81 | |
Matt Martz | b3f9a48cbb | |
Matt Martz | 478f9affdd | |
Matt Martz | 9ccce5d861 | |
Matt Martz | 2a4990c96c | |
Matt Martz | 07c38d7194 | |
Matt Martz | c5f75f783e | |
Matt Martz | 050da542b3 | |
Matt Martz | e14f7ed108 | |
Matt Martz | a7e9bc695e | |
Matt Martz | 77db2ea8f4 | |
Matt Martz | e4218c7612 | |
Matt Martz | 69bae532c5 | |
Matt Martz | 4f7f367391 | |
Matt Martz | 08e87f4c54 | |
Matt Martz | aa52e550bf | |
Matt Martz | 537c5aeda0 | |
Matt Martz | 95fe038752 | |
Matt Martz | 1c0a029ca6 | |
Matt Martz | 9913b9915f | |
Matt Martz | a4cb217522 | |
Matt Martz | d09ec27cb2 | |
Matt Martz | 446e6eb27e | |
Matt Martz | 81182c1c94 | |
Matt Martz | 51014d5a70 | |
Matt Martz | 65145d9aae | |
Matt Martz | 308c530f07 | |
Matt Martz | abe85d85ff | |
Matt Martz | d1b1185bfc | |
Matt Martz | 713860a4b4 | |
Matt Martz | ff606d0ec1 | |
Matt Martz | 3f22a9d815 | |
Matt Martz | cb6dee8a77 | |
Matt Martz | 7b09d8759f | |
Matt Martz | 25d845362c | |
Matt Martz | 4b9662e0b3 | |
Matt Martz | 93951f1154 | |
Matt Martz | 0e6b85d4d5 | |
Matt Martz | 6ab5f27300 | |
Matt Martz | 2ee26bbf54 | |
Matt Martz | 514b310484 | |
Matt Martz | 918e70e66d | |
Matt Martz | bae642ccde | |
Matt Martz | 1df3e76b19 | |
Matt Martz | 1e44e9e2f1 | |
Matt Martz | 51d0d88b96 | |
Matt Martz | 47c17d4a49 | |
Matt Martz | d1be67be48 | |
Matt Martz | 075cfda9cf | |
Matt Martz | 3c04dfefd3 | |
Matt Martz | ffd2c7f963 | |
Matt Martz | aef4a78831 | |
Matt Martz | 72da41e4fc | |
Matt Martz | cb77da3d37 | |
Matt Martz | 790720b33a | |
Matt Martz | 3a31df31c1 | |
Matt Martz | 7383ad97af | |
Matt Martz | 3cc06168f5 |
72
.travis.yml
72
.travis.yml
|
@ -1,35 +1,61 @@
|
||||||
language: python
|
language: python
|
||||||
|
sudo: required
|
||||||
|
dist: xenial
|
||||||
|
|
||||||
python:
|
addons:
|
||||||
- 2.7
|
apt:
|
||||||
|
sources:
|
||||||
|
- deadsnakes
|
||||||
|
packages:
|
||||||
|
- python2.4
|
||||||
|
- python2.5
|
||||||
|
- python2.6
|
||||||
|
- python3.2
|
||||||
|
- python3.3
|
||||||
|
|
||||||
env:
|
matrix:
|
||||||
- TOXENV=py24
|
include:
|
||||||
- TOXENV=py25
|
- python: 2.7
|
||||||
- TOXENV=py26
|
env: TOXENV=flake8
|
||||||
- TOXENV=py27
|
- python: 2.7
|
||||||
- TOXENV=py31
|
env: TOXENV=py24
|
||||||
- TOXENV=py32
|
- python: 2.7
|
||||||
- TOXENV=py33
|
env: TOXENV=py25
|
||||||
- TOXENV=py34
|
- python: 2.7
|
||||||
- TOXENV=pypy
|
env: TOXENV=py26
|
||||||
- TOXENV=flake8
|
- python: 2.7
|
||||||
|
env: TOXENV=py27
|
||||||
|
- python: 2.7
|
||||||
|
env: TOXENV=py32
|
||||||
|
- python: 2.7
|
||||||
|
env: TOXENV=py33
|
||||||
|
- python: 3.4
|
||||||
|
env: TOXENV=py34
|
||||||
|
- python: 3.5
|
||||||
|
env: TOXENV=py35
|
||||||
|
- python: 3.6
|
||||||
|
env: TOXENV=py36
|
||||||
|
- python: 3.7
|
||||||
|
env: TOXENV=py37
|
||||||
|
- python: 3.8-dev
|
||||||
|
env: TOXENV=py38
|
||||||
|
- python: pypy
|
||||||
|
env: TOXENV=pypy
|
||||||
|
dist: trusty
|
||||||
|
|
||||||
before_install:
|
before_install:
|
||||||
- if [[ $(echo "$TOXENV" | egrep -c "(py2[45]|py3[14])") != 0 ]]; then sudo add-apt-repository -y ppa:fkrull/deadsnakes; fi;
|
- if [[ $(echo "$TOXENV" | egrep -c "py35") != 0 ]]; then pyenv global system 3.5; fi;
|
||||||
- if [[ $(echo "$TOXENV" | egrep -c "(py2[45]|py3[14])") != 0 ]]; then sudo apt-get update -qq; fi;
|
|
||||||
- if [[ "$TOXENV" == "py24" ]]; then sudo apt-get install -y python2.4; fi;
|
|
||||||
- if [[ "$TOXENV" == "py25" ]]; then sudo apt-get install -y python2.5; fi;
|
|
||||||
- if [[ "$TOXENV" == "py31" ]]; then sudo apt-get install -y python3.1; fi;
|
|
||||||
- if [[ "$TOXENV" == "py34" ]]; then sudo apt-get install -y python3.4; fi;
|
|
||||||
- if [[ "$TOXENV" == "pypy" ]]; then sudo apt-get install -y pypy; fi;
|
|
||||||
|
|
||||||
install:
|
install:
|
||||||
- if [[ $(echo "$TOXENV" | egrep -c "(py2[45]|py31)") != 0 ]]; then pip install virtualenv==1.7.2 tox==1.3; fi;
|
- if [[ $(echo "$TOXENV" | egrep -c "py32") != 0 ]]; then pip install setuptools==17.1.1; fi;
|
||||||
- if [[ $(echo "$TOXENV" | egrep -c "(py2[45]|py31)") == 0 ]]; then pip install tox; 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 "(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:
|
script:
|
||||||
- tox
|
- tox
|
||||||
|
|
||||||
notifications:
|
notifications:
|
||||||
email:
|
email:
|
||||||
|
|
|
@ -0,0 +1,23 @@
|
||||||
|
#
|
||||||
|
# Project: speedtest-cli
|
||||||
|
# Created by George "walkero" Sokianos
|
||||||
|
# 2022-07-31
|
||||||
|
#
|
||||||
|
|
||||||
|
release: clean
|
||||||
|
mkdir -p release/speedtest-cli
|
||||||
|
cp -r release_files/* release/speedtest-cli/
|
||||||
|
cp speedtest.py release/speedtest-cli/speedtest-cli
|
||||||
|
protect release/speedtest-cli/speedtest-cli srwed
|
||||||
|
cp -r simplejson release/speedtest-cli/
|
||||||
|
rm -rf release/speedtest-cli/simplejson/tests
|
||||||
|
cp README-OS4.md release/speedtest-cli/
|
||||||
|
cp README.rst release/speedtest-cli/
|
||||||
|
cp LICENSE release/speedtest-cli/
|
||||||
|
cp release_files/Icons/cha05e90/speedtest-cli.info release/speedtest-cli.info
|
||||||
|
lha -aeqr3 a speedtest-cli.lha release/
|
||||||
|
rm -rf release
|
||||||
|
|
||||||
|
clean:
|
||||||
|
rm -f simplejson/#?.pyc
|
||||||
|
|
|
@ -0,0 +1,24 @@
|
||||||
|
speedtest-cli
|
||||||
|
---------------------------------------------------------------------------
|
||||||
|
|
||||||
|
This is a simple terminal script that uses speedtest.net API. This is
|
||||||
|
based on Python, so it should work on any operating system supporting
|
||||||
|
Python v2.4 and above. This is created by Matt Martz and can be found
|
||||||
|
at his GitHub repository at https://github.com/sivel/speedtest-cli
|
||||||
|
|
||||||
|
I adapted it to work with AmigaOS 4. To use it just double click its
|
||||||
|
icon and a new shell window will open. Automatically a close server
|
||||||
|
will be chosen and it will measure the download and upload speed of
|
||||||
|
your system.
|
||||||
|
|
||||||
|
More parameters are available for the script and you can find them
|
||||||
|
in the included document README.rst.
|
||||||
|
|
||||||
|
The icons are designed by Frank Ruthe and can be found at
|
||||||
|
http://os4depot.net/?function=showfile&file=graphics/icon/cha05e90_icons_2012.lha
|
||||||
|
|
||||||
|
Changelog
|
||||||
|
--------------------------
|
||||||
|
v2.1.3r1 - 2022-07-31
|
||||||
|
* First release
|
||||||
|
|
86
README.rst
86
README.rst
|
@ -4,20 +4,24 @@ speedtest-cli
|
||||||
Command line interface for testing internet bandwidth using
|
Command line interface for testing internet bandwidth using
|
||||||
speedtest.net
|
speedtest.net
|
||||||
|
|
||||||
.. image:: https://pypip.in/v/speedtest-cli/badge.png
|
.. image:: https://img.shields.io/pypi/v/speedtest-cli.svg
|
||||||
:target: https://pypi.python.org/pypi/speedtest-cli/
|
:target: https://pypi.python.org/pypi/speedtest-cli/
|
||||||
:alt: Latest Version
|
:alt: Latest Version
|
||||||
.. image:: https://pypip.in/d/speedtest-cli/badge.png
|
.. image:: https://img.shields.io/travis/sivel/speedtest-cli.svg
|
||||||
:target: https://pypi.python.org/pypi/speedtest-cli/
|
:target: https://pypi.python.org/pypi/speedtest-cli/
|
||||||
:alt: Downloads
|
:alt: Travis
|
||||||
.. image:: https://pypip.in/license/speedtest-cli/badge.png
|
.. image:: https://img.shields.io/pypi/l/speedtest-cli.svg
|
||||||
:target: https://pypi.python.org/pypi/speedtest-cli/
|
:target: https://pypi.python.org/pypi/speedtest-cli/
|
||||||
:alt: License
|
:alt: License
|
||||||
|
|
||||||
Versions
|
Versions
|
||||||
--------
|
--------
|
||||||
|
|
||||||
speedtest-cli works with Python 2.4-3.4
|
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/
|
||||||
|
:alt: Versions
|
||||||
|
|
||||||
Installation
|
Installation
|
||||||
------------
|
------------
|
||||||
|
@ -47,21 +51,22 @@ or
|
||||||
::
|
::
|
||||||
|
|
||||||
git clone https://github.com/sivel/speedtest-cli.git
|
git clone https://github.com/sivel/speedtest-cli.git
|
||||||
python speedtest-cli/setup.py install
|
cd speedtest-cli
|
||||||
|
python setup.py install
|
||||||
|
|
||||||
Just download (Like the way it used to be)
|
Just download (Like the way it used to be)
|
||||||
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||||
|
|
||||||
::
|
::
|
||||||
|
|
||||||
wget -O speedtest-cli https://raw.github.com/sivel/speedtest-cli/master/speedtest_cli.py
|
wget -O speedtest-cli https://raw.githubusercontent.com/sivel/speedtest-cli/master/speedtest.py
|
||||||
chmod +x speedtest-cli
|
chmod +x speedtest-cli
|
||||||
|
|
||||||
or
|
or
|
||||||
|
|
||||||
::
|
::
|
||||||
|
|
||||||
curl -o speedtest-cli https://raw.github.com/sivel/speedtest-cli/master/speedtest_cli.py
|
curl -Lo speedtest-cli https://raw.githubusercontent.com/sivel/speedtest-cli/master/speedtest.py
|
||||||
chmod +x speedtest-cli
|
chmod +x speedtest-cli
|
||||||
|
|
||||||
Usage
|
Usage
|
||||||
|
@ -70,27 +75,62 @@ Usage
|
||||||
::
|
::
|
||||||
|
|
||||||
$ speedtest-cli -h
|
$ speedtest-cli -h
|
||||||
usage: speedtest-cli [-h] [--bytes] [--share] [--simple] [--list]
|
usage: speedtest-cli [-h] [--no-download] [--no-upload] [--single] [--bytes]
|
||||||
[--server SERVER] [--mini MINI] [--source SOURCE]
|
[--share] [--simple] [--csv]
|
||||||
[--version]
|
[--csv-delimiter CSV_DELIMITER] [--csv-header] [--json]
|
||||||
|
[--list] [--server SERVER] [--exclude EXCLUDE]
|
||||||
|
[--mini MINI] [--source SOURCE] [--timeout TIMEOUT]
|
||||||
|
[--secure] [--no-pre-allocate] [--version]
|
||||||
|
|
||||||
Command line interface for testing internet bandwidth using speedtest.net.
|
Command line interface for testing internet bandwidth using speedtest.net.
|
||||||
--------------------------------------------------------------------------
|
--------------------------------------------------------------------------
|
||||||
https://github.com/sivel/speedtest-cli
|
https://github.com/sivel/speedtest-cli
|
||||||
|
|
||||||
optional arguments:
|
optional arguments:
|
||||||
-h, --help show this help message and exit
|
-h, --help show this help message and exit
|
||||||
--bytes Display values in bytes instead of bits. Does not affect
|
--no-download Do not perform download test
|
||||||
the image generated by --share
|
--no-upload Do not perform upload test
|
||||||
--share Generate and provide a URL to the speedtest.net share
|
--single Only use a single connection instead of multiple. This
|
||||||
results image
|
simulates a typical file transfer.
|
||||||
--simple Suppress verbose output, only show basic information
|
--bytes Display values in bytes instead of bits. Does not
|
||||||
--list Display a list of speedtest.net servers sorted by distance
|
affect the image generated by --share, nor output from
|
||||||
--server SERVER Specify a server ID to test against
|
--json or --csv
|
||||||
--mini MINI URL of the Speedtest Mini server
|
--share Generate and provide a URL to the speedtest.net share
|
||||||
--source SOURCE Source IP address to bind to
|
results image, not displayed with --csv
|
||||||
--timeout TIMEOUT HTTP timeout in seconds. Default 10
|
--simple Suppress verbose output, only show basic information
|
||||||
--version Show the version number and exit
|
--csv Suppress verbose output, only show basic information
|
||||||
|
in CSV format. Speeds listed in bit/s and not affected
|
||||||
|
by --bytes
|
||||||
|
--csv-delimiter CSV_DELIMITER
|
||||||
|
Single character delimiter to use in CSV output.
|
||||||
|
Default ","
|
||||||
|
--csv-header Print CSV headers
|
||||||
|
--json Suppress verbose output, only show basic information
|
||||||
|
in JSON format. Speeds listed in bit/s and not
|
||||||
|
affected by --bytes
|
||||||
|
--list Display a list of speedtest.net servers sorted by
|
||||||
|
distance
|
||||||
|
--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
|
||||||
|
--secure Use HTTPS instead of HTTP when communicating with
|
||||||
|
speedtest.net operated servers
|
||||||
|
--no-pre-allocate 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
|
||||||
|
--version Show the version number and exit
|
||||||
|
|
||||||
|
|
||||||
|
Python API
|
||||||
|
----------
|
||||||
|
|
||||||
|
See the `wiki <https://github.com/sivel/speedtest-cli/wiki>`_.
|
||||||
|
|
||||||
|
|
||||||
Inconsistency
|
Inconsistency
|
||||||
-------------
|
-------------
|
||||||
|
|
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
16
setup.py
16
setup.py
|
@ -1,6 +1,6 @@
|
||||||
#!/usr/bin/env python
|
#!/usr/bin/env python
|
||||||
# -*- coding: utf-8 -*-
|
# -*- coding: utf-8 -*-
|
||||||
# Copyright 2012-2014 Matt Martz
|
# Copyright 2012 Matt Martz
|
||||||
# All Rights Reserved.
|
# All Rights Reserved.
|
||||||
#
|
#
|
||||||
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
||||||
|
@ -57,7 +57,7 @@ except:
|
||||||
|
|
||||||
setup(
|
setup(
|
||||||
name='speedtest-cli',
|
name='speedtest-cli',
|
||||||
version=find_version('speedtest_cli.py'),
|
version=find_version('speedtest.py'),
|
||||||
description=('Command line interface for testing internet bandwidth using '
|
description=('Command line interface for testing internet bandwidth using '
|
||||||
'speedtest.net'),
|
'speedtest.net'),
|
||||||
long_description=long_description,
|
long_description=long_description,
|
||||||
|
@ -66,11 +66,11 @@ setup(
|
||||||
author_email='matt@sivel.net',
|
author_email='matt@sivel.net',
|
||||||
url='https://github.com/sivel/speedtest-cli',
|
url='https://github.com/sivel/speedtest-cli',
|
||||||
license='Apache License, Version 2.0',
|
license='Apache License, Version 2.0',
|
||||||
py_modules=['speedtest_cli'],
|
py_modules=['speedtest'],
|
||||||
entry_points={
|
entry_points={
|
||||||
'console_scripts': [
|
'console_scripts': [
|
||||||
'speedtest=speedtest_cli:main',
|
'speedtest=speedtest:main',
|
||||||
'speedtest-cli=speedtest_cli:main'
|
'speedtest-cli=speedtest:main'
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
classifiers=[
|
classifiers=[
|
||||||
|
@ -89,5 +89,11 @@ setup(
|
||||||
'Programming Language :: Python :: 3.2',
|
'Programming Language :: Python :: 3.2',
|
||||||
'Programming Language :: Python :: 3.3',
|
'Programming Language :: Python :: 3.3',
|
||||||
'Programming Language :: Python :: 3.4',
|
'Programming Language :: Python :: 3.4',
|
||||||
|
'Programming Language :: Python :: 3.5',
|
||||||
|
'Programming Language :: Python :: 3.6',
|
||||||
|
'Programming Language :: Python :: 3.7',
|
||||||
|
'Programming Language :: Python :: 3.8',
|
||||||
|
'Programming Language :: Python :: 3.9',
|
||||||
|
'Programming Language :: Python :: 3.10',
|
||||||
]
|
]
|
||||||
)
|
)
|
||||||
|
|
|
@ -0,0 +1,577 @@
|
||||||
|
r"""JSON (JavaScript Object Notation) <http://json.org> is a subset of
|
||||||
|
JavaScript syntax (ECMA-262 3rd edition) used as a lightweight data
|
||||||
|
interchange format.
|
||||||
|
|
||||||
|
:mod:`simplejson` exposes an API familiar to users of the standard library
|
||||||
|
:mod:`marshal` and :mod:`pickle` modules. It is the externally maintained
|
||||||
|
version of the :mod:`json` library contained in Python 2.6, but maintains
|
||||||
|
compatibility back to Python 2.5 and (currently) has significant performance
|
||||||
|
advantages, even without using the optional C extension for speedups.
|
||||||
|
|
||||||
|
Encoding basic Python object hierarchies::
|
||||||
|
|
||||||
|
>>> import simplejson as json
|
||||||
|
>>> json.dumps(['foo', {'bar': ('baz', None, 1.0, 2)}])
|
||||||
|
'["foo", {"bar": ["baz", null, 1.0, 2]}]'
|
||||||
|
>>> print(json.dumps("\"foo\bar"))
|
||||||
|
"\"foo\bar"
|
||||||
|
>>> print(json.dumps(u'\u1234'))
|
||||||
|
"\u1234"
|
||||||
|
>>> print(json.dumps('\\'))
|
||||||
|
"\\"
|
||||||
|
>>> print(json.dumps({"c": 0, "b": 0, "a": 0}, sort_keys=True))
|
||||||
|
{"a": 0, "b": 0, "c": 0}
|
||||||
|
>>> from simplejson.compat import StringIO
|
||||||
|
>>> io = StringIO()
|
||||||
|
>>> json.dump(['streaming API'], io)
|
||||||
|
>>> io.getvalue()
|
||||||
|
'["streaming API"]'
|
||||||
|
|
||||||
|
Compact encoding::
|
||||||
|
|
||||||
|
>>> import simplejson as json
|
||||||
|
>>> obj = [1,2,3,{'4': 5, '6': 7}]
|
||||||
|
>>> json.dumps(obj, separators=(',',':'), sort_keys=True)
|
||||||
|
'[1,2,3,{"4":5,"6":7}]'
|
||||||
|
|
||||||
|
Pretty printing::
|
||||||
|
|
||||||
|
>>> import simplejson as json
|
||||||
|
>>> print(json.dumps({'4': 5, '6': 7}, sort_keys=True, indent=' '))
|
||||||
|
{
|
||||||
|
"4": 5,
|
||||||
|
"6": 7
|
||||||
|
}
|
||||||
|
|
||||||
|
Decoding JSON::
|
||||||
|
|
||||||
|
>>> import simplejson as json
|
||||||
|
>>> obj = [u'foo', {u'bar': [u'baz', None, 1.0, 2]}]
|
||||||
|
>>> json.loads('["foo", {"bar":["baz", null, 1.0, 2]}]') == obj
|
||||||
|
True
|
||||||
|
>>> json.loads('"\\"foo\\bar"') == u'"foo\x08ar'
|
||||||
|
True
|
||||||
|
>>> from simplejson.compat import StringIO
|
||||||
|
>>> io = StringIO('["streaming API"]')
|
||||||
|
>>> json.load(io)[0] == 'streaming API'
|
||||||
|
True
|
||||||
|
|
||||||
|
Specializing JSON object decoding::
|
||||||
|
|
||||||
|
>>> import simplejson as json
|
||||||
|
>>> def as_complex(dct):
|
||||||
|
... if '__complex__' in dct:
|
||||||
|
... return complex(dct['real'], dct['imag'])
|
||||||
|
... return dct
|
||||||
|
...
|
||||||
|
>>> json.loads('{"__complex__": true, "real": 1, "imag": 2}',
|
||||||
|
... object_hook=as_complex)
|
||||||
|
(1+2j)
|
||||||
|
>>> from decimal import Decimal
|
||||||
|
>>> json.loads('1.1', parse_float=Decimal) == Decimal('1.1')
|
||||||
|
True
|
||||||
|
|
||||||
|
Specializing JSON object encoding::
|
||||||
|
|
||||||
|
>>> import simplejson as json
|
||||||
|
>>> def encode_complex(obj):
|
||||||
|
... if isinstance(obj, complex):
|
||||||
|
... return [obj.real, obj.imag]
|
||||||
|
... raise TypeError('Object of type %s is not JSON serializable' %
|
||||||
|
... obj.__class__.__name__)
|
||||||
|
...
|
||||||
|
>>> json.dumps(2 + 1j, default=encode_complex)
|
||||||
|
'[2.0, 1.0]'
|
||||||
|
>>> json.JSONEncoder(default=encode_complex).encode(2 + 1j)
|
||||||
|
'[2.0, 1.0]'
|
||||||
|
>>> ''.join(json.JSONEncoder(default=encode_complex).iterencode(2 + 1j))
|
||||||
|
'[2.0, 1.0]'
|
||||||
|
|
||||||
|
|
||||||
|
Using simplejson.tool from the shell to validate and pretty-print::
|
||||||
|
|
||||||
|
$ echo '{"json":"obj"}' | python -m simplejson.tool
|
||||||
|
{
|
||||||
|
"json": "obj"
|
||||||
|
}
|
||||||
|
$ echo '{ 1.2:3.4}' | python -m simplejson.tool
|
||||||
|
Expecting property name: line 1 column 3 (char 2)
|
||||||
|
"""
|
||||||
|
from __future__ import absolute_import
|
||||||
|
__version__ = '3.16.0'
|
||||||
|
__all__ = [
|
||||||
|
'dump', 'dumps', 'load', 'loads',
|
||||||
|
'JSONDecoder', 'JSONDecodeError', 'JSONEncoder',
|
||||||
|
'OrderedDict', 'simple_first', 'RawJSON'
|
||||||
|
]
|
||||||
|
|
||||||
|
__author__ = 'Bob Ippolito <bob@redivi.com>'
|
||||||
|
|
||||||
|
from decimal import Decimal
|
||||||
|
|
||||||
|
from .errors import JSONDecodeError
|
||||||
|
from .raw_json import RawJSON
|
||||||
|
from .decoder import JSONDecoder
|
||||||
|
from .encoder import JSONEncoder, JSONEncoderForHTML
|
||||||
|
def _import_OrderedDict():
|
||||||
|
import collections
|
||||||
|
try:
|
||||||
|
return collections.OrderedDict
|
||||||
|
except AttributeError:
|
||||||
|
from . import ordered_dict
|
||||||
|
return ordered_dict.OrderedDict
|
||||||
|
OrderedDict = _import_OrderedDict()
|
||||||
|
|
||||||
|
def _import_c_make_encoder():
|
||||||
|
try:
|
||||||
|
from ._speedups import make_encoder
|
||||||
|
return make_encoder
|
||||||
|
except ImportError:
|
||||||
|
return None
|
||||||
|
|
||||||
|
_default_encoder = JSONEncoder(
|
||||||
|
skipkeys=False,
|
||||||
|
ensure_ascii=True,
|
||||||
|
check_circular=True,
|
||||||
|
allow_nan=True,
|
||||||
|
indent=None,
|
||||||
|
separators=None,
|
||||||
|
encoding='utf-8',
|
||||||
|
default=None,
|
||||||
|
use_decimal=True,
|
||||||
|
namedtuple_as_object=True,
|
||||||
|
tuple_as_array=True,
|
||||||
|
iterable_as_array=False,
|
||||||
|
bigint_as_string=False,
|
||||||
|
item_sort_key=None,
|
||||||
|
for_json=False,
|
||||||
|
ignore_nan=False,
|
||||||
|
int_as_string_bitcount=None,
|
||||||
|
)
|
||||||
|
|
||||||
|
def dump(obj, fp, skipkeys=False, ensure_ascii=True, check_circular=True,
|
||||||
|
allow_nan=True, cls=None, indent=None, separators=None,
|
||||||
|
encoding='utf-8', default=None, use_decimal=True,
|
||||||
|
namedtuple_as_object=True, tuple_as_array=True,
|
||||||
|
bigint_as_string=False, sort_keys=False, item_sort_key=None,
|
||||||
|
for_json=False, ignore_nan=False, int_as_string_bitcount=None,
|
||||||
|
iterable_as_array=False, **kw):
|
||||||
|
"""Serialize ``obj`` as a JSON formatted stream to ``fp`` (a
|
||||||
|
``.write()``-supporting file-like object).
|
||||||
|
|
||||||
|
If *skipkeys* is true then ``dict`` keys that are not basic types
|
||||||
|
(``str``, ``unicode``, ``int``, ``long``, ``float``, ``bool``, ``None``)
|
||||||
|
will be skipped instead of raising a ``TypeError``.
|
||||||
|
|
||||||
|
If *ensure_ascii* is false, then the some chunks written to ``fp``
|
||||||
|
may be ``unicode`` instances, subject to normal Python ``str`` to
|
||||||
|
``unicode`` coercion rules. Unless ``fp.write()`` explicitly
|
||||||
|
understands ``unicode`` (as in ``codecs.getwriter()``) this is likely
|
||||||
|
to cause an error.
|
||||||
|
|
||||||
|
If *check_circular* is false, then the circular reference check
|
||||||
|
for container types will be skipped and a circular reference will
|
||||||
|
result in an ``OverflowError`` (or worse).
|
||||||
|
|
||||||
|
If *allow_nan* is false, then it will be a ``ValueError`` to
|
||||||
|
serialize out of range ``float`` values (``nan``, ``inf``, ``-inf``)
|
||||||
|
in strict compliance of the original JSON specification, instead of using
|
||||||
|
the JavaScript equivalents (``NaN``, ``Infinity``, ``-Infinity``). See
|
||||||
|
*ignore_nan* for ECMA-262 compliant behavior.
|
||||||
|
|
||||||
|
If *indent* is a string, then JSON array elements and object members
|
||||||
|
will be pretty-printed with a newline followed by that string repeated
|
||||||
|
for each level of nesting. ``None`` (the default) selects the most compact
|
||||||
|
representation without any newlines. For backwards compatibility with
|
||||||
|
versions of simplejson earlier than 2.1.0, an integer is also accepted
|
||||||
|
and is converted to a string with that many spaces.
|
||||||
|
|
||||||
|
If specified, *separators* should be an
|
||||||
|
``(item_separator, key_separator)`` tuple. The default is ``(', ', ': ')``
|
||||||
|
if *indent* is ``None`` and ``(',', ': ')`` otherwise. To get the most
|
||||||
|
compact JSON representation, you should specify ``(',', ':')`` to eliminate
|
||||||
|
whitespace.
|
||||||
|
|
||||||
|
*encoding* is the character encoding for str instances, default is UTF-8.
|
||||||
|
|
||||||
|
*default(obj)* is a function that should return a serializable version
|
||||||
|
of obj or raise ``TypeError``. The default simply raises ``TypeError``.
|
||||||
|
|
||||||
|
If *use_decimal* is true (default: ``True``) then decimal.Decimal
|
||||||
|
will be natively serialized to JSON with full precision.
|
||||||
|
|
||||||
|
If *namedtuple_as_object* is true (default: ``True``),
|
||||||
|
:class:`tuple` subclasses with ``_asdict()`` methods will be encoded
|
||||||
|
as JSON objects.
|
||||||
|
|
||||||
|
If *tuple_as_array* is true (default: ``True``),
|
||||||
|
:class:`tuple` (and subclasses) will be encoded as JSON arrays.
|
||||||
|
|
||||||
|
If *iterable_as_array* is true (default: ``False``),
|
||||||
|
any object not in the above table that implements ``__iter__()``
|
||||||
|
will be encoded as a JSON array.
|
||||||
|
|
||||||
|
If *bigint_as_string* is true (default: ``False``), ints 2**53 and higher
|
||||||
|
or lower than -2**53 will be encoded as strings. This is to avoid the
|
||||||
|
rounding that happens in Javascript otherwise. Note that this is still a
|
||||||
|
lossy operation that will not round-trip correctly and should be used
|
||||||
|
sparingly.
|
||||||
|
|
||||||
|
If *int_as_string_bitcount* is a positive number (n), then int of size
|
||||||
|
greater than or equal to 2**n or lower than or equal to -2**n will be
|
||||||
|
encoded as strings.
|
||||||
|
|
||||||
|
If specified, *item_sort_key* is a callable used to sort the items in
|
||||||
|
each dictionary. This is useful if you want to sort items other than
|
||||||
|
in alphabetical order by key. This option takes precedence over
|
||||||
|
*sort_keys*.
|
||||||
|
|
||||||
|
If *sort_keys* is true (default: ``False``), the output of dictionaries
|
||||||
|
will be sorted by item.
|
||||||
|
|
||||||
|
If *for_json* is true (default: ``False``), objects with a ``for_json()``
|
||||||
|
method will use the return value of that method for encoding as JSON
|
||||||
|
instead of the object.
|
||||||
|
|
||||||
|
If *ignore_nan* is true (default: ``False``), then out of range
|
||||||
|
:class:`float` values (``nan``, ``inf``, ``-inf``) will be serialized as
|
||||||
|
``null`` in compliance with the ECMA-262 specification. If true, this will
|
||||||
|
override *allow_nan*.
|
||||||
|
|
||||||
|
To use a custom ``JSONEncoder`` subclass (e.g. one that overrides the
|
||||||
|
``.default()`` method to serialize additional types), specify it with
|
||||||
|
the ``cls`` kwarg. NOTE: You should use *default* or *for_json* instead
|
||||||
|
of subclassing whenever possible.
|
||||||
|
|
||||||
|
"""
|
||||||
|
# cached encoder
|
||||||
|
if (not skipkeys and ensure_ascii and
|
||||||
|
check_circular and allow_nan and
|
||||||
|
cls is None and indent is None and separators is None and
|
||||||
|
encoding == 'utf-8' and default is None and use_decimal
|
||||||
|
and namedtuple_as_object and tuple_as_array and not iterable_as_array
|
||||||
|
and not bigint_as_string and not sort_keys
|
||||||
|
and not item_sort_key and not for_json
|
||||||
|
and not ignore_nan and int_as_string_bitcount is None
|
||||||
|
and not kw
|
||||||
|
):
|
||||||
|
iterable = _default_encoder.iterencode(obj)
|
||||||
|
else:
|
||||||
|
if cls is None:
|
||||||
|
cls = JSONEncoder
|
||||||
|
iterable = cls(skipkeys=skipkeys, ensure_ascii=ensure_ascii,
|
||||||
|
check_circular=check_circular, allow_nan=allow_nan, indent=indent,
|
||||||
|
separators=separators, encoding=encoding,
|
||||||
|
default=default, use_decimal=use_decimal,
|
||||||
|
namedtuple_as_object=namedtuple_as_object,
|
||||||
|
tuple_as_array=tuple_as_array,
|
||||||
|
iterable_as_array=iterable_as_array,
|
||||||
|
bigint_as_string=bigint_as_string,
|
||||||
|
sort_keys=sort_keys,
|
||||||
|
item_sort_key=item_sort_key,
|
||||||
|
for_json=for_json,
|
||||||
|
ignore_nan=ignore_nan,
|
||||||
|
int_as_string_bitcount=int_as_string_bitcount,
|
||||||
|
**kw).iterencode(obj)
|
||||||
|
# could accelerate with writelines in some versions of Python, at
|
||||||
|
# a debuggability cost
|
||||||
|
for chunk in iterable:
|
||||||
|
fp.write(chunk)
|
||||||
|
|
||||||
|
|
||||||
|
def dumps(obj, skipkeys=False, ensure_ascii=True, check_circular=True,
|
||||||
|
allow_nan=True, cls=None, indent=None, separators=None,
|
||||||
|
encoding='utf-8', default=None, use_decimal=True,
|
||||||
|
namedtuple_as_object=True, tuple_as_array=True,
|
||||||
|
bigint_as_string=False, sort_keys=False, item_sort_key=None,
|
||||||
|
for_json=False, ignore_nan=False, int_as_string_bitcount=None,
|
||||||
|
iterable_as_array=False, **kw):
|
||||||
|
"""Serialize ``obj`` to a JSON formatted ``str``.
|
||||||
|
|
||||||
|
If ``skipkeys`` is false then ``dict`` keys that are not basic types
|
||||||
|
(``str``, ``unicode``, ``int``, ``long``, ``float``, ``bool``, ``None``)
|
||||||
|
will be skipped instead of raising a ``TypeError``.
|
||||||
|
|
||||||
|
If ``ensure_ascii`` is false, then the return value will be a
|
||||||
|
``unicode`` instance subject to normal Python ``str`` to ``unicode``
|
||||||
|
coercion rules instead of being escaped to an ASCII ``str``.
|
||||||
|
|
||||||
|
If ``check_circular`` is false, then the circular reference check
|
||||||
|
for container types will be skipped and a circular reference will
|
||||||
|
result in an ``OverflowError`` (or worse).
|
||||||
|
|
||||||
|
If ``allow_nan`` is false, then it will be a ``ValueError`` to
|
||||||
|
serialize out of range ``float`` values (``nan``, ``inf``, ``-inf``) in
|
||||||
|
strict compliance of the JSON specification, instead of using the
|
||||||
|
JavaScript equivalents (``NaN``, ``Infinity``, ``-Infinity``).
|
||||||
|
|
||||||
|
If ``indent`` is a string, then JSON array elements and object members
|
||||||
|
will be pretty-printed with a newline followed by that string repeated
|
||||||
|
for each level of nesting. ``None`` (the default) selects the most compact
|
||||||
|
representation without any newlines. For backwards compatibility with
|
||||||
|
versions of simplejson earlier than 2.1.0, an integer is also accepted
|
||||||
|
and is converted to a string with that many spaces.
|
||||||
|
|
||||||
|
If specified, ``separators`` should be an
|
||||||
|
``(item_separator, key_separator)`` tuple. The default is ``(', ', ': ')``
|
||||||
|
if *indent* is ``None`` and ``(',', ': ')`` otherwise. To get the most
|
||||||
|
compact JSON representation, you should specify ``(',', ':')`` to eliminate
|
||||||
|
whitespace.
|
||||||
|
|
||||||
|
``encoding`` is the character encoding for str instances, default is UTF-8.
|
||||||
|
|
||||||
|
``default(obj)`` is a function that should return a serializable version
|
||||||
|
of obj or raise TypeError. The default simply raises TypeError.
|
||||||
|
|
||||||
|
If *use_decimal* is true (default: ``True``) then decimal.Decimal
|
||||||
|
will be natively serialized to JSON with full precision.
|
||||||
|
|
||||||
|
If *namedtuple_as_object* is true (default: ``True``),
|
||||||
|
:class:`tuple` subclasses with ``_asdict()`` methods will be encoded
|
||||||
|
as JSON objects.
|
||||||
|
|
||||||
|
If *tuple_as_array* is true (default: ``True``),
|
||||||
|
:class:`tuple` (and subclasses) will be encoded as JSON arrays.
|
||||||
|
|
||||||
|
If *iterable_as_array* is true (default: ``False``),
|
||||||
|
any object not in the above table that implements ``__iter__()``
|
||||||
|
will be encoded as a JSON array.
|
||||||
|
|
||||||
|
If *bigint_as_string* is true (not the default), ints 2**53 and higher
|
||||||
|
or lower than -2**53 will be encoded as strings. This is to avoid the
|
||||||
|
rounding that happens in Javascript otherwise.
|
||||||
|
|
||||||
|
If *int_as_string_bitcount* is a positive number (n), then int of size
|
||||||
|
greater than or equal to 2**n or lower than or equal to -2**n will be
|
||||||
|
encoded as strings.
|
||||||
|
|
||||||
|
If specified, *item_sort_key* is a callable used to sort the items in
|
||||||
|
each dictionary. This is useful if you want to sort items other than
|
||||||
|
in alphabetical order by key. This option takes precendence over
|
||||||
|
*sort_keys*.
|
||||||
|
|
||||||
|
If *sort_keys* is true (default: ``False``), the output of dictionaries
|
||||||
|
will be sorted by item.
|
||||||
|
|
||||||
|
If *for_json* is true (default: ``False``), objects with a ``for_json()``
|
||||||
|
method will use the return value of that method for encoding as JSON
|
||||||
|
instead of the object.
|
||||||
|
|
||||||
|
If *ignore_nan* is true (default: ``False``), then out of range
|
||||||
|
:class:`float` values (``nan``, ``inf``, ``-inf``) will be serialized as
|
||||||
|
``null`` in compliance with the ECMA-262 specification. If true, this will
|
||||||
|
override *allow_nan*.
|
||||||
|
|
||||||
|
To use a custom ``JSONEncoder`` subclass (e.g. one that overrides the
|
||||||
|
``.default()`` method to serialize additional types), specify it with
|
||||||
|
the ``cls`` kwarg. NOTE: You should use *default* instead of subclassing
|
||||||
|
whenever possible.
|
||||||
|
|
||||||
|
"""
|
||||||
|
# cached encoder
|
||||||
|
if (not skipkeys and ensure_ascii and
|
||||||
|
check_circular and allow_nan and
|
||||||
|
cls is None and indent is None and separators is None and
|
||||||
|
encoding == 'utf-8' and default is None and use_decimal
|
||||||
|
and namedtuple_as_object and tuple_as_array and not iterable_as_array
|
||||||
|
and not bigint_as_string and not sort_keys
|
||||||
|
and not item_sort_key and not for_json
|
||||||
|
and not ignore_nan and int_as_string_bitcount is None
|
||||||
|
and not kw
|
||||||
|
):
|
||||||
|
return _default_encoder.encode(obj)
|
||||||
|
if cls is None:
|
||||||
|
cls = JSONEncoder
|
||||||
|
return cls(
|
||||||
|
skipkeys=skipkeys, ensure_ascii=ensure_ascii,
|
||||||
|
check_circular=check_circular, allow_nan=allow_nan, indent=indent,
|
||||||
|
separators=separators, encoding=encoding, default=default,
|
||||||
|
use_decimal=use_decimal,
|
||||||
|
namedtuple_as_object=namedtuple_as_object,
|
||||||
|
tuple_as_array=tuple_as_array,
|
||||||
|
iterable_as_array=iterable_as_array,
|
||||||
|
bigint_as_string=bigint_as_string,
|
||||||
|
sort_keys=sort_keys,
|
||||||
|
item_sort_key=item_sort_key,
|
||||||
|
for_json=for_json,
|
||||||
|
ignore_nan=ignore_nan,
|
||||||
|
int_as_string_bitcount=int_as_string_bitcount,
|
||||||
|
**kw).encode(obj)
|
||||||
|
|
||||||
|
|
||||||
|
_default_decoder = JSONDecoder(encoding=None, object_hook=None,
|
||||||
|
object_pairs_hook=None)
|
||||||
|
|
||||||
|
|
||||||
|
def load(fp, encoding=None, cls=None, object_hook=None, parse_float=None,
|
||||||
|
parse_int=None, parse_constant=None, object_pairs_hook=None,
|
||||||
|
use_decimal=False, namedtuple_as_object=True, tuple_as_array=True,
|
||||||
|
**kw):
|
||||||
|
"""Deserialize ``fp`` (a ``.read()``-supporting file-like object containing
|
||||||
|
a JSON document) to a Python object.
|
||||||
|
|
||||||
|
*encoding* determines the encoding used to interpret any
|
||||||
|
:class:`str` objects decoded by this instance (``'utf-8'`` by
|
||||||
|
default). It has no effect when decoding :class:`unicode` objects.
|
||||||
|
|
||||||
|
Note that currently only encodings that are a superset of ASCII work,
|
||||||
|
strings of other encodings should be passed in as :class:`unicode`.
|
||||||
|
|
||||||
|
*object_hook*, if specified, will be called with the result of every
|
||||||
|
JSON object decoded and its return value will be used in place of the
|
||||||
|
given :class:`dict`. This can be used to provide custom
|
||||||
|
deserializations (e.g. to support JSON-RPC class hinting).
|
||||||
|
|
||||||
|
*object_pairs_hook* is an optional function that will be called with
|
||||||
|
the result of any object literal decode with an ordered list of pairs.
|
||||||
|
The return value of *object_pairs_hook* will be used instead of the
|
||||||
|
:class:`dict`. This feature can be used to implement custom decoders
|
||||||
|
that rely on the order that the key and value pairs are decoded (for
|
||||||
|
example, :func:`collections.OrderedDict` will remember the order of
|
||||||
|
insertion). If *object_hook* is also defined, the *object_pairs_hook*
|
||||||
|
takes priority.
|
||||||
|
|
||||||
|
*parse_float*, if specified, will be called with the string of every
|
||||||
|
JSON float to be decoded. By default, this is equivalent to
|
||||||
|
``float(num_str)``. This can be used to use another datatype or parser
|
||||||
|
for JSON floats (e.g. :class:`decimal.Decimal`).
|
||||||
|
|
||||||
|
*parse_int*, if specified, will be called with the string of every
|
||||||
|
JSON int to be decoded. By default, this is equivalent to
|
||||||
|
``int(num_str)``. This can be used to use another datatype or parser
|
||||||
|
for JSON integers (e.g. :class:`float`).
|
||||||
|
|
||||||
|
*parse_constant*, if specified, will be called with one of the
|
||||||
|
following strings: ``'-Infinity'``, ``'Infinity'``, ``'NaN'``. This
|
||||||
|
can be used to raise an exception if invalid JSON numbers are
|
||||||
|
encountered.
|
||||||
|
|
||||||
|
If *use_decimal* is true (default: ``False``) then it implies
|
||||||
|
parse_float=decimal.Decimal for parity with ``dump``.
|
||||||
|
|
||||||
|
To use a custom ``JSONDecoder`` subclass, specify it with the ``cls``
|
||||||
|
kwarg. NOTE: You should use *object_hook* or *object_pairs_hook* instead
|
||||||
|
of subclassing whenever possible.
|
||||||
|
|
||||||
|
"""
|
||||||
|
return loads(fp.read(),
|
||||||
|
encoding=encoding, cls=cls, object_hook=object_hook,
|
||||||
|
parse_float=parse_float, parse_int=parse_int,
|
||||||
|
parse_constant=parse_constant, object_pairs_hook=object_pairs_hook,
|
||||||
|
use_decimal=use_decimal, **kw)
|
||||||
|
|
||||||
|
|
||||||
|
def loads(s, encoding=None, cls=None, object_hook=None, parse_float=None,
|
||||||
|
parse_int=None, parse_constant=None, object_pairs_hook=None,
|
||||||
|
use_decimal=False, **kw):
|
||||||
|
"""Deserialize ``s`` (a ``str`` or ``unicode`` instance containing a JSON
|
||||||
|
document) to a Python object.
|
||||||
|
|
||||||
|
*encoding* determines the encoding used to interpret any
|
||||||
|
:class:`str` objects decoded by this instance (``'utf-8'`` by
|
||||||
|
default). It has no effect when decoding :class:`unicode` objects.
|
||||||
|
|
||||||
|
Note that currently only encodings that are a superset of ASCII work,
|
||||||
|
strings of other encodings should be passed in as :class:`unicode`.
|
||||||
|
|
||||||
|
*object_hook*, if specified, will be called with the result of every
|
||||||
|
JSON object decoded and its return value will be used in place of the
|
||||||
|
given :class:`dict`. This can be used to provide custom
|
||||||
|
deserializations (e.g. to support JSON-RPC class hinting).
|
||||||
|
|
||||||
|
*object_pairs_hook* is an optional function that will be called with
|
||||||
|
the result of any object literal decode with an ordered list of pairs.
|
||||||
|
The return value of *object_pairs_hook* will be used instead of the
|
||||||
|
:class:`dict`. This feature can be used to implement custom decoders
|
||||||
|
that rely on the order that the key and value pairs are decoded (for
|
||||||
|
example, :func:`collections.OrderedDict` will remember the order of
|
||||||
|
insertion). If *object_hook* is also defined, the *object_pairs_hook*
|
||||||
|
takes priority.
|
||||||
|
|
||||||
|
*parse_float*, if specified, will be called with the string of every
|
||||||
|
JSON float to be decoded. By default, this is equivalent to
|
||||||
|
``float(num_str)``. This can be used to use another datatype or parser
|
||||||
|
for JSON floats (e.g. :class:`decimal.Decimal`).
|
||||||
|
|
||||||
|
*parse_int*, if specified, will be called with the string of every
|
||||||
|
JSON int to be decoded. By default, this is equivalent to
|
||||||
|
``int(num_str)``. This can be used to use another datatype or parser
|
||||||
|
for JSON integers (e.g. :class:`float`).
|
||||||
|
|
||||||
|
*parse_constant*, if specified, will be called with one of the
|
||||||
|
following strings: ``'-Infinity'``, ``'Infinity'``, ``'NaN'``. This
|
||||||
|
can be used to raise an exception if invalid JSON numbers are
|
||||||
|
encountered.
|
||||||
|
|
||||||
|
If *use_decimal* is true (default: ``False``) then it implies
|
||||||
|
parse_float=decimal.Decimal for parity with ``dump``.
|
||||||
|
|
||||||
|
To use a custom ``JSONDecoder`` subclass, specify it with the ``cls``
|
||||||
|
kwarg. NOTE: You should use *object_hook* or *object_pairs_hook* instead
|
||||||
|
of subclassing whenever possible.
|
||||||
|
|
||||||
|
"""
|
||||||
|
if (cls is None and encoding is None and object_hook is None and
|
||||||
|
parse_int is None and parse_float is None and
|
||||||
|
parse_constant is None and object_pairs_hook is None
|
||||||
|
and not use_decimal and not kw):
|
||||||
|
return _default_decoder.decode(s)
|
||||||
|
if cls is None:
|
||||||
|
cls = JSONDecoder
|
||||||
|
if object_hook is not None:
|
||||||
|
kw['object_hook'] = object_hook
|
||||||
|
if object_pairs_hook is not None:
|
||||||
|
kw['object_pairs_hook'] = object_pairs_hook
|
||||||
|
if parse_float is not None:
|
||||||
|
kw['parse_float'] = parse_float
|
||||||
|
if parse_int is not None:
|
||||||
|
kw['parse_int'] = parse_int
|
||||||
|
if parse_constant is not None:
|
||||||
|
kw['parse_constant'] = parse_constant
|
||||||
|
if use_decimal:
|
||||||
|
if parse_float is not None:
|
||||||
|
raise TypeError("use_decimal=True implies parse_float=Decimal")
|
||||||
|
kw['parse_float'] = Decimal
|
||||||
|
return cls(encoding=encoding, **kw).decode(s)
|
||||||
|
|
||||||
|
|
||||||
|
def _toggle_speedups(enabled):
|
||||||
|
from . import decoder as dec
|
||||||
|
from . import encoder as enc
|
||||||
|
from . import scanner as scan
|
||||||
|
c_make_encoder = _import_c_make_encoder()
|
||||||
|
if enabled:
|
||||||
|
dec.scanstring = dec.c_scanstring or dec.py_scanstring
|
||||||
|
enc.c_make_encoder = c_make_encoder
|
||||||
|
enc.encode_basestring_ascii = (enc.c_encode_basestring_ascii or
|
||||||
|
enc.py_encode_basestring_ascii)
|
||||||
|
scan.make_scanner = scan.c_make_scanner or scan.py_make_scanner
|
||||||
|
else:
|
||||||
|
dec.scanstring = dec.py_scanstring
|
||||||
|
enc.c_make_encoder = None
|
||||||
|
enc.encode_basestring_ascii = enc.py_encode_basestring_ascii
|
||||||
|
scan.make_scanner = scan.py_make_scanner
|
||||||
|
dec.make_scanner = scan.make_scanner
|
||||||
|
global _default_decoder
|
||||||
|
_default_decoder = JSONDecoder(
|
||||||
|
encoding=None,
|
||||||
|
object_hook=None,
|
||||||
|
object_pairs_hook=None,
|
||||||
|
)
|
||||||
|
global _default_encoder
|
||||||
|
_default_encoder = JSONEncoder(
|
||||||
|
skipkeys=False,
|
||||||
|
ensure_ascii=True,
|
||||||
|
check_circular=True,
|
||||||
|
allow_nan=True,
|
||||||
|
indent=None,
|
||||||
|
separators=None,
|
||||||
|
encoding='utf-8',
|
||||||
|
default=None,
|
||||||
|
)
|
||||||
|
|
||||||
|
def simple_first(kv):
|
||||||
|
"""Helper function to pass to item_sort_key to sort simple
|
||||||
|
elements to the top, then container elements.
|
||||||
|
"""
|
||||||
|
return (isinstance(kv[1], (list, dict, tuple)), kv[0])
|
File diff suppressed because it is too large
Load Diff
|
@ -0,0 +1,34 @@
|
||||||
|
"""Python 3 compatibility shims
|
||||||
|
"""
|
||||||
|
import sys
|
||||||
|
if sys.version_info[0] < 3:
|
||||||
|
PY3 = False
|
||||||
|
def b(s):
|
||||||
|
return s
|
||||||
|
try:
|
||||||
|
from cStringIO import StringIO
|
||||||
|
except ImportError:
|
||||||
|
from StringIO import StringIO
|
||||||
|
BytesIO = StringIO
|
||||||
|
text_type = unicode
|
||||||
|
binary_type = str
|
||||||
|
string_types = (basestring,)
|
||||||
|
integer_types = (int, long)
|
||||||
|
unichr = unichr
|
||||||
|
reload_module = reload
|
||||||
|
else:
|
||||||
|
PY3 = True
|
||||||
|
if sys.version_info[:2] >= (3, 4):
|
||||||
|
from importlib import reload as reload_module
|
||||||
|
else:
|
||||||
|
from imp import reload as reload_module
|
||||||
|
def b(s):
|
||||||
|
return bytes(s, 'latin1')
|
||||||
|
from io import StringIO, BytesIO
|
||||||
|
text_type = str
|
||||||
|
binary_type = bytes
|
||||||
|
string_types = (str,)
|
||||||
|
integer_types = (int,)
|
||||||
|
unichr = chr
|
||||||
|
|
||||||
|
long_type = integer_types[-1]
|
|
@ -0,0 +1,400 @@
|
||||||
|
"""Implementation of JSONDecoder
|
||||||
|
"""
|
||||||
|
from __future__ import absolute_import
|
||||||
|
import re
|
||||||
|
import sys
|
||||||
|
import struct
|
||||||
|
from .compat import PY3, unichr
|
||||||
|
from .scanner import make_scanner, JSONDecodeError
|
||||||
|
|
||||||
|
def _import_c_scanstring():
|
||||||
|
try:
|
||||||
|
from ._speedups import scanstring
|
||||||
|
return scanstring
|
||||||
|
except ImportError:
|
||||||
|
return None
|
||||||
|
c_scanstring = _import_c_scanstring()
|
||||||
|
|
||||||
|
# NOTE (3.1.0): JSONDecodeError may still be imported from this module for
|
||||||
|
# compatibility, but it was never in the __all__
|
||||||
|
__all__ = ['JSONDecoder']
|
||||||
|
|
||||||
|
FLAGS = re.VERBOSE | re.MULTILINE | re.DOTALL
|
||||||
|
|
||||||
|
def _floatconstants():
|
||||||
|
if sys.version_info < (2, 6):
|
||||||
|
_BYTES = '7FF80000000000007FF0000000000000'.decode('hex')
|
||||||
|
nan, inf = struct.unpack('>dd', _BYTES)
|
||||||
|
else:
|
||||||
|
nan = float('nan')
|
||||||
|
inf = float('inf')
|
||||||
|
return nan, inf, -inf
|
||||||
|
|
||||||
|
NaN, PosInf, NegInf = _floatconstants()
|
||||||
|
|
||||||
|
_CONSTANTS = {
|
||||||
|
'-Infinity': NegInf,
|
||||||
|
'Infinity': PosInf,
|
||||||
|
'NaN': NaN,
|
||||||
|
}
|
||||||
|
|
||||||
|
STRINGCHUNK = re.compile(r'(.*?)(["\\\x00-\x1f])', FLAGS)
|
||||||
|
BACKSLASH = {
|
||||||
|
'"': u'"', '\\': u'\\', '/': u'/',
|
||||||
|
'b': u'\b', 'f': u'\f', 'n': u'\n', 'r': u'\r', 't': u'\t',
|
||||||
|
}
|
||||||
|
|
||||||
|
DEFAULT_ENCODING = "utf-8"
|
||||||
|
|
||||||
|
def py_scanstring(s, end, encoding=None, strict=True,
|
||||||
|
_b=BACKSLASH, _m=STRINGCHUNK.match, _join=u''.join,
|
||||||
|
_PY3=PY3, _maxunicode=sys.maxunicode):
|
||||||
|
"""Scan the string s for a JSON string. End is the index of the
|
||||||
|
character in s after the quote that started the JSON string.
|
||||||
|
Unescapes all valid JSON string escape sequences and raises ValueError
|
||||||
|
on attempt to decode an invalid string. If strict is False then literal
|
||||||
|
control characters are allowed in the string.
|
||||||
|
|
||||||
|
Returns a tuple of the decoded string and the index of the character in s
|
||||||
|
after the end quote."""
|
||||||
|
if encoding is None:
|
||||||
|
encoding = DEFAULT_ENCODING
|
||||||
|
chunks = []
|
||||||
|
_append = chunks.append
|
||||||
|
begin = end - 1
|
||||||
|
while 1:
|
||||||
|
chunk = _m(s, end)
|
||||||
|
if chunk is None:
|
||||||
|
raise JSONDecodeError(
|
||||||
|
"Unterminated string starting at", s, begin)
|
||||||
|
end = chunk.end()
|
||||||
|
content, terminator = chunk.groups()
|
||||||
|
# Content is contains zero or more unescaped string characters
|
||||||
|
if content:
|
||||||
|
if not _PY3 and not isinstance(content, unicode):
|
||||||
|
content = unicode(content, encoding)
|
||||||
|
_append(content)
|
||||||
|
# Terminator is the end of string, a literal control character,
|
||||||
|
# or a backslash denoting that an escape sequence follows
|
||||||
|
if terminator == '"':
|
||||||
|
break
|
||||||
|
elif terminator != '\\':
|
||||||
|
if strict:
|
||||||
|
msg = "Invalid control character %r at"
|
||||||
|
raise JSONDecodeError(msg, s, end)
|
||||||
|
else:
|
||||||
|
_append(terminator)
|
||||||
|
continue
|
||||||
|
try:
|
||||||
|
esc = s[end]
|
||||||
|
except IndexError:
|
||||||
|
raise JSONDecodeError(
|
||||||
|
"Unterminated string starting at", s, begin)
|
||||||
|
# If not a unicode escape sequence, must be in the lookup table
|
||||||
|
if esc != 'u':
|
||||||
|
try:
|
||||||
|
char = _b[esc]
|
||||||
|
except KeyError:
|
||||||
|
msg = "Invalid \\X escape sequence %r"
|
||||||
|
raise JSONDecodeError(msg, s, end)
|
||||||
|
end += 1
|
||||||
|
else:
|
||||||
|
# Unicode escape sequence
|
||||||
|
msg = "Invalid \\uXXXX escape sequence"
|
||||||
|
esc = s[end + 1:end + 5]
|
||||||
|
escX = esc[1:2]
|
||||||
|
if len(esc) != 4 or escX == 'x' or escX == 'X':
|
||||||
|
raise JSONDecodeError(msg, s, end - 1)
|
||||||
|
try:
|
||||||
|
uni = int(esc, 16)
|
||||||
|
except ValueError:
|
||||||
|
raise JSONDecodeError(msg, s, end - 1)
|
||||||
|
end += 5
|
||||||
|
# Check for surrogate pair on UCS-4 systems
|
||||||
|
# Note that this will join high/low surrogate pairs
|
||||||
|
# but will also pass unpaired surrogates through
|
||||||
|
if (_maxunicode > 65535 and
|
||||||
|
uni & 0xfc00 == 0xd800 and
|
||||||
|
s[end:end + 2] == '\\u'):
|
||||||
|
esc2 = s[end + 2:end + 6]
|
||||||
|
escX = esc2[1:2]
|
||||||
|
if len(esc2) == 4 and not (escX == 'x' or escX == 'X'):
|
||||||
|
try:
|
||||||
|
uni2 = int(esc2, 16)
|
||||||
|
except ValueError:
|
||||||
|
raise JSONDecodeError(msg, s, end)
|
||||||
|
if uni2 & 0xfc00 == 0xdc00:
|
||||||
|
uni = 0x10000 + (((uni - 0xd800) << 10) |
|
||||||
|
(uni2 - 0xdc00))
|
||||||
|
end += 6
|
||||||
|
char = unichr(uni)
|
||||||
|
# Append the unescaped character
|
||||||
|
_append(char)
|
||||||
|
return _join(chunks), end
|
||||||
|
|
||||||
|
|
||||||
|
# Use speedup if available
|
||||||
|
scanstring = c_scanstring or py_scanstring
|
||||||
|
|
||||||
|
WHITESPACE = re.compile(r'[ \t\n\r]*', FLAGS)
|
||||||
|
WHITESPACE_STR = ' \t\n\r'
|
||||||
|
|
||||||
|
def JSONObject(state, encoding, strict, scan_once, object_hook,
|
||||||
|
object_pairs_hook, memo=None,
|
||||||
|
_w=WHITESPACE.match, _ws=WHITESPACE_STR):
|
||||||
|
(s, end) = state
|
||||||
|
# Backwards compatibility
|
||||||
|
if memo is None:
|
||||||
|
memo = {}
|
||||||
|
memo_get = memo.setdefault
|
||||||
|
pairs = []
|
||||||
|
# Use a slice to prevent IndexError from being raised, the following
|
||||||
|
# check will raise a more specific ValueError if the string is empty
|
||||||
|
nextchar = s[end:end + 1]
|
||||||
|
# Normally we expect nextchar == '"'
|
||||||
|
if nextchar != '"':
|
||||||
|
if nextchar in _ws:
|
||||||
|
end = _w(s, end).end()
|
||||||
|
nextchar = s[end:end + 1]
|
||||||
|
# Trivial empty object
|
||||||
|
if nextchar == '}':
|
||||||
|
if object_pairs_hook is not None:
|
||||||
|
result = object_pairs_hook(pairs)
|
||||||
|
return result, end + 1
|
||||||
|
pairs = {}
|
||||||
|
if object_hook is not None:
|
||||||
|
pairs = object_hook(pairs)
|
||||||
|
return pairs, end + 1
|
||||||
|
elif nextchar != '"':
|
||||||
|
raise JSONDecodeError(
|
||||||
|
"Expecting property name enclosed in double quotes",
|
||||||
|
s, end)
|
||||||
|
end += 1
|
||||||
|
while True:
|
||||||
|
key, end = scanstring(s, end, encoding, strict)
|
||||||
|
key = memo_get(key, key)
|
||||||
|
|
||||||
|
# To skip some function call overhead we optimize the fast paths where
|
||||||
|
# the JSON key separator is ": " or just ":".
|
||||||
|
if s[end:end + 1] != ':':
|
||||||
|
end = _w(s, end).end()
|
||||||
|
if s[end:end + 1] != ':':
|
||||||
|
raise JSONDecodeError("Expecting ':' delimiter", s, end)
|
||||||
|
|
||||||
|
end += 1
|
||||||
|
|
||||||
|
try:
|
||||||
|
if s[end] in _ws:
|
||||||
|
end += 1
|
||||||
|
if s[end] in _ws:
|
||||||
|
end = _w(s, end + 1).end()
|
||||||
|
except IndexError:
|
||||||
|
pass
|
||||||
|
|
||||||
|
value, end = scan_once(s, end)
|
||||||
|
pairs.append((key, value))
|
||||||
|
|
||||||
|
try:
|
||||||
|
nextchar = s[end]
|
||||||
|
if nextchar in _ws:
|
||||||
|
end = _w(s, end + 1).end()
|
||||||
|
nextchar = s[end]
|
||||||
|
except IndexError:
|
||||||
|
nextchar = ''
|
||||||
|
end += 1
|
||||||
|
|
||||||
|
if nextchar == '}':
|
||||||
|
break
|
||||||
|
elif nextchar != ',':
|
||||||
|
raise JSONDecodeError("Expecting ',' delimiter or '}'", s, end - 1)
|
||||||
|
|
||||||
|
try:
|
||||||
|
nextchar = s[end]
|
||||||
|
if nextchar in _ws:
|
||||||
|
end += 1
|
||||||
|
nextchar = s[end]
|
||||||
|
if nextchar in _ws:
|
||||||
|
end = _w(s, end + 1).end()
|
||||||
|
nextchar = s[end]
|
||||||
|
except IndexError:
|
||||||
|
nextchar = ''
|
||||||
|
|
||||||
|
end += 1
|
||||||
|
if nextchar != '"':
|
||||||
|
raise JSONDecodeError(
|
||||||
|
"Expecting property name enclosed in double quotes",
|
||||||
|
s, end - 1)
|
||||||
|
|
||||||
|
if object_pairs_hook is not None:
|
||||||
|
result = object_pairs_hook(pairs)
|
||||||
|
return result, end
|
||||||
|
pairs = dict(pairs)
|
||||||
|
if object_hook is not None:
|
||||||
|
pairs = object_hook(pairs)
|
||||||
|
return pairs, end
|
||||||
|
|
||||||
|
def JSONArray(state, scan_once, _w=WHITESPACE.match, _ws=WHITESPACE_STR):
|
||||||
|
(s, end) = state
|
||||||
|
values = []
|
||||||
|
nextchar = s[end:end + 1]
|
||||||
|
if nextchar in _ws:
|
||||||
|
end = _w(s, end + 1).end()
|
||||||
|
nextchar = s[end:end + 1]
|
||||||
|
# Look-ahead for trivial empty array
|
||||||
|
if nextchar == ']':
|
||||||
|
return values, end + 1
|
||||||
|
elif nextchar == '':
|
||||||
|
raise JSONDecodeError("Expecting value or ']'", s, end)
|
||||||
|
_append = values.append
|
||||||
|
while True:
|
||||||
|
value, end = scan_once(s, end)
|
||||||
|
_append(value)
|
||||||
|
nextchar = s[end:end + 1]
|
||||||
|
if nextchar in _ws:
|
||||||
|
end = _w(s, end + 1).end()
|
||||||
|
nextchar = s[end:end + 1]
|
||||||
|
end += 1
|
||||||
|
if nextchar == ']':
|
||||||
|
break
|
||||||
|
elif nextchar != ',':
|
||||||
|
raise JSONDecodeError("Expecting ',' delimiter or ']'", s, end - 1)
|
||||||
|
|
||||||
|
try:
|
||||||
|
if s[end] in _ws:
|
||||||
|
end += 1
|
||||||
|
if s[end] in _ws:
|
||||||
|
end = _w(s, end + 1).end()
|
||||||
|
except IndexError:
|
||||||
|
pass
|
||||||
|
|
||||||
|
return values, end
|
||||||
|
|
||||||
|
class JSONDecoder(object):
|
||||||
|
"""Simple JSON <http://json.org> decoder
|
||||||
|
|
||||||
|
Performs the following translations in decoding by default:
|
||||||
|
|
||||||
|
+---------------+-------------------+
|
||||||
|
| JSON | Python |
|
||||||
|
+===============+===================+
|
||||||
|
| object | dict |
|
||||||
|
+---------------+-------------------+
|
||||||
|
| array | list |
|
||||||
|
+---------------+-------------------+
|
||||||
|
| string | str, unicode |
|
||||||
|
+---------------+-------------------+
|
||||||
|
| number (int) | int, long |
|
||||||
|
+---------------+-------------------+
|
||||||
|
| number (real) | float |
|
||||||
|
+---------------+-------------------+
|
||||||
|
| true | True |
|
||||||
|
+---------------+-------------------+
|
||||||
|
| false | False |
|
||||||
|
+---------------+-------------------+
|
||||||
|
| null | None |
|
||||||
|
+---------------+-------------------+
|
||||||
|
|
||||||
|
It also understands ``NaN``, ``Infinity``, and ``-Infinity`` as
|
||||||
|
their corresponding ``float`` values, which is outside the JSON spec.
|
||||||
|
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __init__(self, encoding=None, object_hook=None, parse_float=None,
|
||||||
|
parse_int=None, parse_constant=None, strict=True,
|
||||||
|
object_pairs_hook=None):
|
||||||
|
"""
|
||||||
|
*encoding* determines the encoding used to interpret any
|
||||||
|
:class:`str` objects decoded by this instance (``'utf-8'`` by
|
||||||
|
default). It has no effect when decoding :class:`unicode` objects.
|
||||||
|
|
||||||
|
Note that currently only encodings that are a superset of ASCII work,
|
||||||
|
strings of other encodings should be passed in as :class:`unicode`.
|
||||||
|
|
||||||
|
*object_hook*, if specified, will be called with the result of every
|
||||||
|
JSON object decoded and its return value will be used in place of the
|
||||||
|
given :class:`dict`. This can be used to provide custom
|
||||||
|
deserializations (e.g. to support JSON-RPC class hinting).
|
||||||
|
|
||||||
|
*object_pairs_hook* is an optional function that will be called with
|
||||||
|
the result of any object literal decode with an ordered list of pairs.
|
||||||
|
The return value of *object_pairs_hook* will be used instead of the
|
||||||
|
:class:`dict`. This feature can be used to implement custom decoders
|
||||||
|
that rely on the order that the key and value pairs are decoded (for
|
||||||
|
example, :func:`collections.OrderedDict` will remember the order of
|
||||||
|
insertion). If *object_hook* is also defined, the *object_pairs_hook*
|
||||||
|
takes priority.
|
||||||
|
|
||||||
|
*parse_float*, if specified, will be called with the string of every
|
||||||
|
JSON float to be decoded. By default, this is equivalent to
|
||||||
|
``float(num_str)``. This can be used to use another datatype or parser
|
||||||
|
for JSON floats (e.g. :class:`decimal.Decimal`).
|
||||||
|
|
||||||
|
*parse_int*, if specified, will be called with the string of every
|
||||||
|
JSON int to be decoded. By default, this is equivalent to
|
||||||
|
``int(num_str)``. This can be used to use another datatype or parser
|
||||||
|
for JSON integers (e.g. :class:`float`).
|
||||||
|
|
||||||
|
*parse_constant*, if specified, will be called with one of the
|
||||||
|
following strings: ``'-Infinity'``, ``'Infinity'``, ``'NaN'``. This
|
||||||
|
can be used to raise an exception if invalid JSON numbers are
|
||||||
|
encountered.
|
||||||
|
|
||||||
|
*strict* controls the parser's behavior when it encounters an
|
||||||
|
invalid control character in a string. The default setting of
|
||||||
|
``True`` means that unescaped control characters are parse errors, if
|
||||||
|
``False`` then control characters will be allowed in strings.
|
||||||
|
|
||||||
|
"""
|
||||||
|
if encoding is None:
|
||||||
|
encoding = DEFAULT_ENCODING
|
||||||
|
self.encoding = encoding
|
||||||
|
self.object_hook = object_hook
|
||||||
|
self.object_pairs_hook = object_pairs_hook
|
||||||
|
self.parse_float = parse_float or float
|
||||||
|
self.parse_int = parse_int or int
|
||||||
|
self.parse_constant = parse_constant or _CONSTANTS.__getitem__
|
||||||
|
self.strict = strict
|
||||||
|
self.parse_object = JSONObject
|
||||||
|
self.parse_array = JSONArray
|
||||||
|
self.parse_string = scanstring
|
||||||
|
self.memo = {}
|
||||||
|
self.scan_once = make_scanner(self)
|
||||||
|
|
||||||
|
def decode(self, s, _w=WHITESPACE.match, _PY3=PY3):
|
||||||
|
"""Return the Python representation of ``s`` (a ``str`` or ``unicode``
|
||||||
|
instance containing a JSON document)
|
||||||
|
|
||||||
|
"""
|
||||||
|
if _PY3 and isinstance(s, bytes):
|
||||||
|
s = str(s, self.encoding)
|
||||||
|
obj, end = self.raw_decode(s)
|
||||||
|
end = _w(s, end).end()
|
||||||
|
if end != len(s):
|
||||||
|
raise JSONDecodeError("Extra data", s, end, len(s))
|
||||||
|
return obj
|
||||||
|
|
||||||
|
def raw_decode(self, s, idx=0, _w=WHITESPACE.match, _PY3=PY3):
|
||||||
|
"""Decode a JSON document from ``s`` (a ``str`` or ``unicode``
|
||||||
|
beginning with a JSON document) and return a 2-tuple of the Python
|
||||||
|
representation and the index in ``s`` where the document ended.
|
||||||
|
Optionally, ``idx`` can be used to specify an offset in ``s`` where
|
||||||
|
the JSON document begins.
|
||||||
|
|
||||||
|
This can be used to decode a JSON document from a string that may
|
||||||
|
have extraneous data at the end.
|
||||||
|
|
||||||
|
"""
|
||||||
|
if idx < 0:
|
||||||
|
# Ensure that raw_decode bails on negative indexes, the regex
|
||||||
|
# would otherwise mask this behavior. #98
|
||||||
|
raise JSONDecodeError('Expecting value', s, idx)
|
||||||
|
if _PY3 and not isinstance(s, str):
|
||||||
|
raise TypeError("Input string must be text, not bytes")
|
||||||
|
# strip UTF-8 bom
|
||||||
|
if len(s) > idx:
|
||||||
|
ord0 = ord(s[idx])
|
||||||
|
if ord0 == 0xfeff:
|
||||||
|
idx += 1
|
||||||
|
elif ord0 == 0xef and s[idx:idx + 3] == '\xef\xbb\xbf':
|
||||||
|
idx += 3
|
||||||
|
return self.scan_once(s, idx=_w(s, idx).end())
|
|
@ -0,0 +1,722 @@
|
||||||
|
"""Implementation of JSONEncoder
|
||||||
|
"""
|
||||||
|
from __future__ import absolute_import
|
||||||
|
import re
|
||||||
|
from operator import itemgetter
|
||||||
|
# Do not import Decimal directly to avoid reload issues
|
||||||
|
import decimal
|
||||||
|
from .compat import unichr, binary_type, text_type, string_types, integer_types, PY3
|
||||||
|
def _import_speedups():
|
||||||
|
try:
|
||||||
|
from . import _speedups
|
||||||
|
return _speedups.encode_basestring_ascii, _speedups.make_encoder
|
||||||
|
except ImportError:
|
||||||
|
return None, None
|
||||||
|
c_encode_basestring_ascii, c_make_encoder = _import_speedups()
|
||||||
|
|
||||||
|
from .decoder import PosInf
|
||||||
|
from .raw_json import RawJSON
|
||||||
|
|
||||||
|
ESCAPE = re.compile(r'[\x00-\x1f\\"]')
|
||||||
|
ESCAPE_ASCII = re.compile(r'([\\"]|[^\ -~])')
|
||||||
|
HAS_UTF8 = re.compile(r'[\x80-\xff]')
|
||||||
|
ESCAPE_DCT = {
|
||||||
|
'\\': '\\\\',
|
||||||
|
'"': '\\"',
|
||||||
|
'\b': '\\b',
|
||||||
|
'\f': '\\f',
|
||||||
|
'\n': '\\n',
|
||||||
|
'\r': '\\r',
|
||||||
|
'\t': '\\t',
|
||||||
|
}
|
||||||
|
for i in range(0x20):
|
||||||
|
#ESCAPE_DCT.setdefault(chr(i), '\\u{0:04x}'.format(i))
|
||||||
|
ESCAPE_DCT.setdefault(chr(i), '\\u%04x' % (i,))
|
||||||
|
|
||||||
|
FLOAT_REPR = repr
|
||||||
|
|
||||||
|
def encode_basestring(s, _PY3=PY3, _q=u'"'):
|
||||||
|
"""Return a JSON representation of a Python string
|
||||||
|
|
||||||
|
"""
|
||||||
|
if _PY3:
|
||||||
|
if isinstance(s, bytes):
|
||||||
|
s = str(s, 'utf-8')
|
||||||
|
elif type(s) is not str:
|
||||||
|
# convert an str subclass instance to exact str
|
||||||
|
# raise a TypeError otherwise
|
||||||
|
s = str.__str__(s)
|
||||||
|
else:
|
||||||
|
if isinstance(s, str) and HAS_UTF8.search(s) is not None:
|
||||||
|
s = unicode(s, 'utf-8')
|
||||||
|
elif type(s) not in (str, unicode):
|
||||||
|
# convert an str subclass instance to exact str
|
||||||
|
# convert a unicode subclass instance to exact unicode
|
||||||
|
# raise a TypeError otherwise
|
||||||
|
if isinstance(s, str):
|
||||||
|
s = str.__str__(s)
|
||||||
|
else:
|
||||||
|
s = unicode.__getnewargs__(s)[0]
|
||||||
|
def replace(match):
|
||||||
|
return ESCAPE_DCT[match.group(0)]
|
||||||
|
return _q + ESCAPE.sub(replace, s) + _q
|
||||||
|
|
||||||
|
|
||||||
|
def py_encode_basestring_ascii(s, _PY3=PY3):
|
||||||
|
"""Return an ASCII-only JSON representation of a Python string
|
||||||
|
|
||||||
|
"""
|
||||||
|
if _PY3:
|
||||||
|
if isinstance(s, bytes):
|
||||||
|
s = str(s, 'utf-8')
|
||||||
|
elif type(s) is not str:
|
||||||
|
# convert an str subclass instance to exact str
|
||||||
|
# raise a TypeError otherwise
|
||||||
|
s = str.__str__(s)
|
||||||
|
else:
|
||||||
|
if isinstance(s, str) and HAS_UTF8.search(s) is not None:
|
||||||
|
s = unicode(s, 'utf-8')
|
||||||
|
elif type(s) not in (str, unicode):
|
||||||
|
# convert an str subclass instance to exact str
|
||||||
|
# convert a unicode subclass instance to exact unicode
|
||||||
|
# raise a TypeError otherwise
|
||||||
|
if isinstance(s, str):
|
||||||
|
s = str.__str__(s)
|
||||||
|
else:
|
||||||
|
s = unicode.__getnewargs__(s)[0]
|
||||||
|
def replace(match):
|
||||||
|
s = match.group(0)
|
||||||
|
try:
|
||||||
|
return ESCAPE_DCT[s]
|
||||||
|
except KeyError:
|
||||||
|
n = ord(s)
|
||||||
|
if n < 0x10000:
|
||||||
|
#return '\\u{0:04x}'.format(n)
|
||||||
|
return '\\u%04x' % (n,)
|
||||||
|
else:
|
||||||
|
# surrogate pair
|
||||||
|
n -= 0x10000
|
||||||
|
s1 = 0xd800 | ((n >> 10) & 0x3ff)
|
||||||
|
s2 = 0xdc00 | (n & 0x3ff)
|
||||||
|
#return '\\u{0:04x}\\u{1:04x}'.format(s1, s2)
|
||||||
|
return '\\u%04x\\u%04x' % (s1, s2)
|
||||||
|
return '"' + str(ESCAPE_ASCII.sub(replace, s)) + '"'
|
||||||
|
|
||||||
|
|
||||||
|
encode_basestring_ascii = (
|
||||||
|
c_encode_basestring_ascii or py_encode_basestring_ascii)
|
||||||
|
|
||||||
|
class JSONEncoder(object):
|
||||||
|
"""Extensible JSON <http://json.org> encoder for Python data structures.
|
||||||
|
|
||||||
|
Supports the following objects and types by default:
|
||||||
|
|
||||||
|
+-------------------+---------------+
|
||||||
|
| Python | JSON |
|
||||||
|
+===================+===============+
|
||||||
|
| dict, namedtuple | object |
|
||||||
|
+-------------------+---------------+
|
||||||
|
| list, tuple | array |
|
||||||
|
+-------------------+---------------+
|
||||||
|
| str, unicode | string |
|
||||||
|
+-------------------+---------------+
|
||||||
|
| int, long, float | number |
|
||||||
|
+-------------------+---------------+
|
||||||
|
| True | true |
|
||||||
|
+-------------------+---------------+
|
||||||
|
| False | false |
|
||||||
|
+-------------------+---------------+
|
||||||
|
| None | null |
|
||||||
|
+-------------------+---------------+
|
||||||
|
|
||||||
|
To extend this to recognize other objects, subclass and implement a
|
||||||
|
``.default()`` method with another method that returns a serializable
|
||||||
|
object for ``o`` if possible, otherwise it should call the superclass
|
||||||
|
implementation (to raise ``TypeError``).
|
||||||
|
|
||||||
|
"""
|
||||||
|
item_separator = ', '
|
||||||
|
key_separator = ': '
|
||||||
|
|
||||||
|
def __init__(self, skipkeys=False, ensure_ascii=True,
|
||||||
|
check_circular=True, allow_nan=True, sort_keys=False,
|
||||||
|
indent=None, separators=None, encoding='utf-8', default=None,
|
||||||
|
use_decimal=True, namedtuple_as_object=True,
|
||||||
|
tuple_as_array=True, bigint_as_string=False,
|
||||||
|
item_sort_key=None, for_json=False, ignore_nan=False,
|
||||||
|
int_as_string_bitcount=None, iterable_as_array=False):
|
||||||
|
"""Constructor for JSONEncoder, with sensible defaults.
|
||||||
|
|
||||||
|
If skipkeys is false, then it is a TypeError to attempt
|
||||||
|
encoding of keys that are not str, int, long, float or None. If
|
||||||
|
skipkeys is True, such items are simply skipped.
|
||||||
|
|
||||||
|
If ensure_ascii is true, the output is guaranteed to be str
|
||||||
|
objects with all incoming unicode characters escaped. If
|
||||||
|
ensure_ascii is false, the output will be unicode object.
|
||||||
|
|
||||||
|
If check_circular is true, then lists, dicts, and custom encoded
|
||||||
|
objects will be checked for circular references during encoding to
|
||||||
|
prevent an infinite recursion (which would cause an OverflowError).
|
||||||
|
Otherwise, no such check takes place.
|
||||||
|
|
||||||
|
If allow_nan is true, then NaN, Infinity, and -Infinity will be
|
||||||
|
encoded as such. This behavior is not JSON specification compliant,
|
||||||
|
but is consistent with most JavaScript based encoders and decoders.
|
||||||
|
Otherwise, it will be a ValueError to encode such floats.
|
||||||
|
|
||||||
|
If sort_keys is true, then the output of dictionaries will be
|
||||||
|
sorted by key; this is useful for regression tests to ensure
|
||||||
|
that JSON serializations can be compared on a day-to-day basis.
|
||||||
|
|
||||||
|
If indent is a string, then JSON array elements and object members
|
||||||
|
will be pretty-printed with a newline followed by that string repeated
|
||||||
|
for each level of nesting. ``None`` (the default) selects the most compact
|
||||||
|
representation without any newlines. For backwards compatibility with
|
||||||
|
versions of simplejson earlier than 2.1.0, an integer is also accepted
|
||||||
|
and is converted to a string with that many spaces.
|
||||||
|
|
||||||
|
If specified, separators should be an (item_separator, key_separator)
|
||||||
|
tuple. The default is (', ', ': ') if *indent* is ``None`` and
|
||||||
|
(',', ': ') otherwise. To get the most compact JSON representation,
|
||||||
|
you should specify (',', ':') to eliminate whitespace.
|
||||||
|
|
||||||
|
If specified, default is a function that gets called for objects
|
||||||
|
that can't otherwise be serialized. It should return a JSON encodable
|
||||||
|
version of the object or raise a ``TypeError``.
|
||||||
|
|
||||||
|
If encoding is not None, then all input strings will be
|
||||||
|
transformed into unicode using that encoding prior to JSON-encoding.
|
||||||
|
The default is UTF-8.
|
||||||
|
|
||||||
|
If use_decimal is true (default: ``True``), ``decimal.Decimal`` will
|
||||||
|
be supported directly by the encoder. For the inverse, decode JSON
|
||||||
|
with ``parse_float=decimal.Decimal``.
|
||||||
|
|
||||||
|
If namedtuple_as_object is true (the default), objects with
|
||||||
|
``_asdict()`` methods will be encoded as JSON objects.
|
||||||
|
|
||||||
|
If tuple_as_array is true (the default), tuple (and subclasses) will
|
||||||
|
be encoded as JSON arrays.
|
||||||
|
|
||||||
|
If *iterable_as_array* is true (default: ``False``),
|
||||||
|
any object not in the above table that implements ``__iter__()``
|
||||||
|
will be encoded as a JSON array.
|
||||||
|
|
||||||
|
If bigint_as_string is true (not the default), ints 2**53 and higher
|
||||||
|
or lower than -2**53 will be encoded as strings. This is to avoid the
|
||||||
|
rounding that happens in Javascript otherwise.
|
||||||
|
|
||||||
|
If int_as_string_bitcount is a positive number (n), then int of size
|
||||||
|
greater than or equal to 2**n or lower than or equal to -2**n will be
|
||||||
|
encoded as strings.
|
||||||
|
|
||||||
|
If specified, item_sort_key is a callable used to sort the items in
|
||||||
|
each dictionary. This is useful if you want to sort items other than
|
||||||
|
in alphabetical order by key.
|
||||||
|
|
||||||
|
If for_json is true (not the default), objects with a ``for_json()``
|
||||||
|
method will use the return value of that method for encoding as JSON
|
||||||
|
instead of the object.
|
||||||
|
|
||||||
|
If *ignore_nan* is true (default: ``False``), then out of range
|
||||||
|
:class:`float` values (``nan``, ``inf``, ``-inf``) will be serialized
|
||||||
|
as ``null`` in compliance with the ECMA-262 specification. If true,
|
||||||
|
this will override *allow_nan*.
|
||||||
|
|
||||||
|
"""
|
||||||
|
|
||||||
|
self.skipkeys = skipkeys
|
||||||
|
self.ensure_ascii = ensure_ascii
|
||||||
|
self.check_circular = check_circular
|
||||||
|
self.allow_nan = allow_nan
|
||||||
|
self.sort_keys = sort_keys
|
||||||
|
self.use_decimal = use_decimal
|
||||||
|
self.namedtuple_as_object = namedtuple_as_object
|
||||||
|
self.tuple_as_array = tuple_as_array
|
||||||
|
self.iterable_as_array = iterable_as_array
|
||||||
|
self.bigint_as_string = bigint_as_string
|
||||||
|
self.item_sort_key = item_sort_key
|
||||||
|
self.for_json = for_json
|
||||||
|
self.ignore_nan = ignore_nan
|
||||||
|
self.int_as_string_bitcount = int_as_string_bitcount
|
||||||
|
if indent is not None and not isinstance(indent, string_types):
|
||||||
|
indent = indent * ' '
|
||||||
|
self.indent = indent
|
||||||
|
if separators is not None:
|
||||||
|
self.item_separator, self.key_separator = separators
|
||||||
|
elif indent is not None:
|
||||||
|
self.item_separator = ','
|
||||||
|
if default is not None:
|
||||||
|
self.default = default
|
||||||
|
self.encoding = encoding
|
||||||
|
|
||||||
|
def default(self, o):
|
||||||
|
"""Implement this method in a subclass such that it returns
|
||||||
|
a serializable object for ``o``, or calls the base implementation
|
||||||
|
(to raise a ``TypeError``).
|
||||||
|
|
||||||
|
For example, to support arbitrary iterators, you could
|
||||||
|
implement default like this::
|
||||||
|
|
||||||
|
def default(self, o):
|
||||||
|
try:
|
||||||
|
iterable = iter(o)
|
||||||
|
except TypeError:
|
||||||
|
pass
|
||||||
|
else:
|
||||||
|
return list(iterable)
|
||||||
|
return JSONEncoder.default(self, o)
|
||||||
|
|
||||||
|
"""
|
||||||
|
raise TypeError('Object of type %s is not JSON serializable' %
|
||||||
|
o.__class__.__name__)
|
||||||
|
|
||||||
|
def encode(self, o):
|
||||||
|
"""Return a JSON string representation of a Python data structure.
|
||||||
|
|
||||||
|
>>> from simplejson import JSONEncoder
|
||||||
|
>>> JSONEncoder().encode({"foo": ["bar", "baz"]})
|
||||||
|
'{"foo": ["bar", "baz"]}'
|
||||||
|
|
||||||
|
"""
|
||||||
|
# This is for extremely simple cases and benchmarks.
|
||||||
|
if isinstance(o, binary_type):
|
||||||
|
_encoding = self.encoding
|
||||||
|
if (_encoding is not None and not (_encoding == 'utf-8')):
|
||||||
|
o = text_type(o, _encoding)
|
||||||
|
if isinstance(o, string_types):
|
||||||
|
if self.ensure_ascii:
|
||||||
|
return encode_basestring_ascii(o)
|
||||||
|
else:
|
||||||
|
return encode_basestring(o)
|
||||||
|
# This doesn't pass the iterator directly to ''.join() because the
|
||||||
|
# exceptions aren't as detailed. The list call should be roughly
|
||||||
|
# equivalent to the PySequence_Fast that ''.join() would do.
|
||||||
|
chunks = self.iterencode(o, _one_shot=True)
|
||||||
|
if not isinstance(chunks, (list, tuple)):
|
||||||
|
chunks = list(chunks)
|
||||||
|
if self.ensure_ascii:
|
||||||
|
return ''.join(chunks)
|
||||||
|
else:
|
||||||
|
return u''.join(chunks)
|
||||||
|
|
||||||
|
def iterencode(self, o, _one_shot=False):
|
||||||
|
"""Encode the given object and yield each string
|
||||||
|
representation as available.
|
||||||
|
|
||||||
|
For example::
|
||||||
|
|
||||||
|
for chunk in JSONEncoder().iterencode(bigobject):
|
||||||
|
mysocket.write(chunk)
|
||||||
|
|
||||||
|
"""
|
||||||
|
if self.check_circular:
|
||||||
|
markers = {}
|
||||||
|
else:
|
||||||
|
markers = None
|
||||||
|
if self.ensure_ascii:
|
||||||
|
_encoder = encode_basestring_ascii
|
||||||
|
else:
|
||||||
|
_encoder = encode_basestring
|
||||||
|
if self.encoding != 'utf-8' and self.encoding is not None:
|
||||||
|
def _encoder(o, _orig_encoder=_encoder, _encoding=self.encoding):
|
||||||
|
if isinstance(o, binary_type):
|
||||||
|
o = text_type(o, _encoding)
|
||||||
|
return _orig_encoder(o)
|
||||||
|
|
||||||
|
def floatstr(o, allow_nan=self.allow_nan, ignore_nan=self.ignore_nan,
|
||||||
|
_repr=FLOAT_REPR, _inf=PosInf, _neginf=-PosInf):
|
||||||
|
# Check for specials. Note that this type of test is processor
|
||||||
|
# and/or platform-specific, so do tests which don't depend on
|
||||||
|
# the internals.
|
||||||
|
|
||||||
|
if o != o:
|
||||||
|
text = 'NaN'
|
||||||
|
elif o == _inf:
|
||||||
|
text = 'Infinity'
|
||||||
|
elif o == _neginf:
|
||||||
|
text = '-Infinity'
|
||||||
|
else:
|
||||||
|
if type(o) != float:
|
||||||
|
# See #118, do not trust custom str/repr
|
||||||
|
o = float(o)
|
||||||
|
return _repr(o)
|
||||||
|
|
||||||
|
if ignore_nan:
|
||||||
|
text = 'null'
|
||||||
|
elif not allow_nan:
|
||||||
|
raise ValueError(
|
||||||
|
"Out of range float values are not JSON compliant: " +
|
||||||
|
repr(o))
|
||||||
|
|
||||||
|
return text
|
||||||
|
|
||||||
|
key_memo = {}
|
||||||
|
int_as_string_bitcount = (
|
||||||
|
53 if self.bigint_as_string else self.int_as_string_bitcount)
|
||||||
|
if (_one_shot and c_make_encoder is not None
|
||||||
|
and self.indent is None):
|
||||||
|
_iterencode = c_make_encoder(
|
||||||
|
markers, self.default, _encoder, self.indent,
|
||||||
|
self.key_separator, self.item_separator, self.sort_keys,
|
||||||
|
self.skipkeys, self.allow_nan, key_memo, self.use_decimal,
|
||||||
|
self.namedtuple_as_object, self.tuple_as_array,
|
||||||
|
int_as_string_bitcount,
|
||||||
|
self.item_sort_key, self.encoding, self.for_json,
|
||||||
|
self.ignore_nan, decimal.Decimal, self.iterable_as_array)
|
||||||
|
else:
|
||||||
|
_iterencode = _make_iterencode(
|
||||||
|
markers, self.default, _encoder, self.indent, floatstr,
|
||||||
|
self.key_separator, self.item_separator, self.sort_keys,
|
||||||
|
self.skipkeys, _one_shot, self.use_decimal,
|
||||||
|
self.namedtuple_as_object, self.tuple_as_array,
|
||||||
|
int_as_string_bitcount,
|
||||||
|
self.item_sort_key, self.encoding, self.for_json,
|
||||||
|
self.iterable_as_array, Decimal=decimal.Decimal)
|
||||||
|
try:
|
||||||
|
return _iterencode(o, 0)
|
||||||
|
finally:
|
||||||
|
key_memo.clear()
|
||||||
|
|
||||||
|
|
||||||
|
class JSONEncoderForHTML(JSONEncoder):
|
||||||
|
"""An encoder that produces JSON safe to embed in HTML.
|
||||||
|
|
||||||
|
To embed JSON content in, say, a script tag on a web page, the
|
||||||
|
characters &, < and > should be escaped. They cannot be escaped
|
||||||
|
with the usual entities (e.g. &) because they are not expanded
|
||||||
|
within <script> tags.
|
||||||
|
|
||||||
|
This class also escapes the line separator and paragraph separator
|
||||||
|
characters U+2028 and U+2029, irrespective of the ensure_ascii setting,
|
||||||
|
as these characters are not valid in JavaScript strings (see
|
||||||
|
http://timelessrepo.com/json-isnt-a-javascript-subset).
|
||||||
|
"""
|
||||||
|
|
||||||
|
def encode(self, o):
|
||||||
|
# Override JSONEncoder.encode because it has hacks for
|
||||||
|
# performance that make things more complicated.
|
||||||
|
chunks = self.iterencode(o, True)
|
||||||
|
if self.ensure_ascii:
|
||||||
|
return ''.join(chunks)
|
||||||
|
else:
|
||||||
|
return u''.join(chunks)
|
||||||
|
|
||||||
|
def iterencode(self, o, _one_shot=False):
|
||||||
|
chunks = super(JSONEncoderForHTML, self).iterencode(o, _one_shot)
|
||||||
|
for chunk in chunks:
|
||||||
|
chunk = chunk.replace('&', '\\u0026')
|
||||||
|
chunk = chunk.replace('<', '\\u003c')
|
||||||
|
chunk = chunk.replace('>', '\\u003e')
|
||||||
|
|
||||||
|
if not self.ensure_ascii:
|
||||||
|
chunk = chunk.replace(u'\u2028', '\\u2028')
|
||||||
|
chunk = chunk.replace(u'\u2029', '\\u2029')
|
||||||
|
|
||||||
|
yield chunk
|
||||||
|
|
||||||
|
|
||||||
|
def _make_iterencode(markers, _default, _encoder, _indent, _floatstr,
|
||||||
|
_key_separator, _item_separator, _sort_keys, _skipkeys, _one_shot,
|
||||||
|
_use_decimal, _namedtuple_as_object, _tuple_as_array,
|
||||||
|
_int_as_string_bitcount, _item_sort_key,
|
||||||
|
_encoding,_for_json,
|
||||||
|
_iterable_as_array,
|
||||||
|
## HACK: hand-optimized bytecode; turn globals into locals
|
||||||
|
_PY3=PY3,
|
||||||
|
ValueError=ValueError,
|
||||||
|
string_types=string_types,
|
||||||
|
Decimal=None,
|
||||||
|
dict=dict,
|
||||||
|
float=float,
|
||||||
|
id=id,
|
||||||
|
integer_types=integer_types,
|
||||||
|
isinstance=isinstance,
|
||||||
|
list=list,
|
||||||
|
str=str,
|
||||||
|
tuple=tuple,
|
||||||
|
iter=iter,
|
||||||
|
):
|
||||||
|
if _use_decimal and Decimal is None:
|
||||||
|
Decimal = decimal.Decimal
|
||||||
|
if _item_sort_key and not callable(_item_sort_key):
|
||||||
|
raise TypeError("item_sort_key must be None or callable")
|
||||||
|
elif _sort_keys and not _item_sort_key:
|
||||||
|
_item_sort_key = itemgetter(0)
|
||||||
|
|
||||||
|
if (_int_as_string_bitcount is not None and
|
||||||
|
(_int_as_string_bitcount <= 0 or
|
||||||
|
not isinstance(_int_as_string_bitcount, integer_types))):
|
||||||
|
raise TypeError("int_as_string_bitcount must be a positive integer")
|
||||||
|
|
||||||
|
def _encode_int(value):
|
||||||
|
skip_quoting = (
|
||||||
|
_int_as_string_bitcount is None
|
||||||
|
or
|
||||||
|
_int_as_string_bitcount < 1
|
||||||
|
)
|
||||||
|
if type(value) not in integer_types:
|
||||||
|
# See #118, do not trust custom str/repr
|
||||||
|
value = int(value)
|
||||||
|
if (
|
||||||
|
skip_quoting or
|
||||||
|
(-1 << _int_as_string_bitcount)
|
||||||
|
< value <
|
||||||
|
(1 << _int_as_string_bitcount)
|
||||||
|
):
|
||||||
|
return str(value)
|
||||||
|
return '"' + str(value) + '"'
|
||||||
|
|
||||||
|
def _iterencode_list(lst, _current_indent_level):
|
||||||
|
if not lst:
|
||||||
|
yield '[]'
|
||||||
|
return
|
||||||
|
if markers is not None:
|
||||||
|
markerid = id(lst)
|
||||||
|
if markerid in markers:
|
||||||
|
raise ValueError("Circular reference detected")
|
||||||
|
markers[markerid] = lst
|
||||||
|
buf = '['
|
||||||
|
if _indent is not None:
|
||||||
|
_current_indent_level += 1
|
||||||
|
newline_indent = '\n' + (_indent * _current_indent_level)
|
||||||
|
separator = _item_separator + newline_indent
|
||||||
|
buf += newline_indent
|
||||||
|
else:
|
||||||
|
newline_indent = None
|
||||||
|
separator = _item_separator
|
||||||
|
first = True
|
||||||
|
for value in lst:
|
||||||
|
if first:
|
||||||
|
first = False
|
||||||
|
else:
|
||||||
|
buf = separator
|
||||||
|
if isinstance(value, string_types):
|
||||||
|
yield buf + _encoder(value)
|
||||||
|
elif _PY3 and isinstance(value, bytes) and _encoding is not None:
|
||||||
|
yield buf + _encoder(value)
|
||||||
|
elif isinstance(value, RawJSON):
|
||||||
|
yield buf + value.encoded_json
|
||||||
|
elif value is None:
|
||||||
|
yield buf + 'null'
|
||||||
|
elif value is True:
|
||||||
|
yield buf + 'true'
|
||||||
|
elif value is False:
|
||||||
|
yield buf + 'false'
|
||||||
|
elif isinstance(value, integer_types):
|
||||||
|
yield buf + _encode_int(value)
|
||||||
|
elif isinstance(value, float):
|
||||||
|
yield buf + _floatstr(value)
|
||||||
|
elif _use_decimal and isinstance(value, Decimal):
|
||||||
|
yield buf + str(value)
|
||||||
|
else:
|
||||||
|
yield buf
|
||||||
|
for_json = _for_json and getattr(value, 'for_json', None)
|
||||||
|
if for_json and callable(for_json):
|
||||||
|
chunks = _iterencode(for_json(), _current_indent_level)
|
||||||
|
elif isinstance(value, list):
|
||||||
|
chunks = _iterencode_list(value, _current_indent_level)
|
||||||
|
else:
|
||||||
|
_asdict = _namedtuple_as_object and getattr(value, '_asdict', None)
|
||||||
|
if _asdict and callable(_asdict):
|
||||||
|
chunks = _iterencode_dict(_asdict(),
|
||||||
|
_current_indent_level)
|
||||||
|
elif _tuple_as_array and isinstance(value, tuple):
|
||||||
|
chunks = _iterencode_list(value, _current_indent_level)
|
||||||
|
elif isinstance(value, dict):
|
||||||
|
chunks = _iterencode_dict(value, _current_indent_level)
|
||||||
|
else:
|
||||||
|
chunks = _iterencode(value, _current_indent_level)
|
||||||
|
for chunk in chunks:
|
||||||
|
yield chunk
|
||||||
|
if first:
|
||||||
|
# iterable_as_array misses the fast path at the top
|
||||||
|
yield '[]'
|
||||||
|
else:
|
||||||
|
if newline_indent is not None:
|
||||||
|
_current_indent_level -= 1
|
||||||
|
yield '\n' + (_indent * _current_indent_level)
|
||||||
|
yield ']'
|
||||||
|
if markers is not None:
|
||||||
|
del markers[markerid]
|
||||||
|
|
||||||
|
def _stringify_key(key):
|
||||||
|
if isinstance(key, string_types): # pragma: no cover
|
||||||
|
pass
|
||||||
|
elif _PY3 and isinstance(key, bytes) and _encoding is not None:
|
||||||
|
key = str(key, _encoding)
|
||||||
|
elif isinstance(key, float):
|
||||||
|
key = _floatstr(key)
|
||||||
|
elif key is True:
|
||||||
|
key = 'true'
|
||||||
|
elif key is False:
|
||||||
|
key = 'false'
|
||||||
|
elif key is None:
|
||||||
|
key = 'null'
|
||||||
|
elif isinstance(key, integer_types):
|
||||||
|
if type(key) not in integer_types:
|
||||||
|
# See #118, do not trust custom str/repr
|
||||||
|
key = int(key)
|
||||||
|
key = str(key)
|
||||||
|
elif _use_decimal and isinstance(key, Decimal):
|
||||||
|
key = str(key)
|
||||||
|
elif _skipkeys:
|
||||||
|
key = None
|
||||||
|
else:
|
||||||
|
raise TypeError('keys must be str, int, float, bool or None, '
|
||||||
|
'not %s' % key.__class__.__name__)
|
||||||
|
return key
|
||||||
|
|
||||||
|
def _iterencode_dict(dct, _current_indent_level):
|
||||||
|
if not dct:
|
||||||
|
yield '{}'
|
||||||
|
return
|
||||||
|
if markers is not None:
|
||||||
|
markerid = id(dct)
|
||||||
|
if markerid in markers:
|
||||||
|
raise ValueError("Circular reference detected")
|
||||||
|
markers[markerid] = dct
|
||||||
|
yield '{'
|
||||||
|
if _indent is not None:
|
||||||
|
_current_indent_level += 1
|
||||||
|
newline_indent = '\n' + (_indent * _current_indent_level)
|
||||||
|
item_separator = _item_separator + newline_indent
|
||||||
|
yield newline_indent
|
||||||
|
else:
|
||||||
|
newline_indent = None
|
||||||
|
item_separator = _item_separator
|
||||||
|
first = True
|
||||||
|
if _PY3:
|
||||||
|
iteritems = dct.items()
|
||||||
|
else:
|
||||||
|
iteritems = dct.iteritems()
|
||||||
|
if _item_sort_key:
|
||||||
|
items = []
|
||||||
|
for k, v in dct.items():
|
||||||
|
if not isinstance(k, string_types):
|
||||||
|
k = _stringify_key(k)
|
||||||
|
if k is None:
|
||||||
|
continue
|
||||||
|
items.append((k, v))
|
||||||
|
items.sort(key=_item_sort_key)
|
||||||
|
else:
|
||||||
|
items = iteritems
|
||||||
|
for key, value in items:
|
||||||
|
if not (_item_sort_key or isinstance(key, string_types)):
|
||||||
|
key = _stringify_key(key)
|
||||||
|
if key is None:
|
||||||
|
# _skipkeys must be True
|
||||||
|
continue
|
||||||
|
if first:
|
||||||
|
first = False
|
||||||
|
else:
|
||||||
|
yield item_separator
|
||||||
|
yield _encoder(key)
|
||||||
|
yield _key_separator
|
||||||
|
if isinstance(value, string_types):
|
||||||
|
yield _encoder(value)
|
||||||
|
elif _PY3 and isinstance(value, bytes) and _encoding is not None:
|
||||||
|
yield _encoder(value)
|
||||||
|
elif isinstance(value, RawJSON):
|
||||||
|
yield value.encoded_json
|
||||||
|
elif value is None:
|
||||||
|
yield 'null'
|
||||||
|
elif value is True:
|
||||||
|
yield 'true'
|
||||||
|
elif value is False:
|
||||||
|
yield 'false'
|
||||||
|
elif isinstance(value, integer_types):
|
||||||
|
yield _encode_int(value)
|
||||||
|
elif isinstance(value, float):
|
||||||
|
yield _floatstr(value)
|
||||||
|
elif _use_decimal and isinstance(value, Decimal):
|
||||||
|
yield str(value)
|
||||||
|
else:
|
||||||
|
for_json = _for_json and getattr(value, 'for_json', None)
|
||||||
|
if for_json and callable(for_json):
|
||||||
|
chunks = _iterencode(for_json(), _current_indent_level)
|
||||||
|
elif isinstance(value, list):
|
||||||
|
chunks = _iterencode_list(value, _current_indent_level)
|
||||||
|
else:
|
||||||
|
_asdict = _namedtuple_as_object and getattr(value, '_asdict', None)
|
||||||
|
if _asdict and callable(_asdict):
|
||||||
|
chunks = _iterencode_dict(_asdict(),
|
||||||
|
_current_indent_level)
|
||||||
|
elif _tuple_as_array and isinstance(value, tuple):
|
||||||
|
chunks = _iterencode_list(value, _current_indent_level)
|
||||||
|
elif isinstance(value, dict):
|
||||||
|
chunks = _iterencode_dict(value, _current_indent_level)
|
||||||
|
else:
|
||||||
|
chunks = _iterencode(value, _current_indent_level)
|
||||||
|
for chunk in chunks:
|
||||||
|
yield chunk
|
||||||
|
if newline_indent is not None:
|
||||||
|
_current_indent_level -= 1
|
||||||
|
yield '\n' + (_indent * _current_indent_level)
|
||||||
|
yield '}'
|
||||||
|
if markers is not None:
|
||||||
|
del markers[markerid]
|
||||||
|
|
||||||
|
def _iterencode(o, _current_indent_level):
|
||||||
|
if isinstance(o, string_types):
|
||||||
|
yield _encoder(o)
|
||||||
|
elif _PY3 and isinstance(o, bytes) and _encoding is not None:
|
||||||
|
yield _encoder(o)
|
||||||
|
elif isinstance(o, RawJSON):
|
||||||
|
yield o.encoded_json
|
||||||
|
elif o is None:
|
||||||
|
yield 'null'
|
||||||
|
elif o is True:
|
||||||
|
yield 'true'
|
||||||
|
elif o is False:
|
||||||
|
yield 'false'
|
||||||
|
elif isinstance(o, integer_types):
|
||||||
|
yield _encode_int(o)
|
||||||
|
elif isinstance(o, float):
|
||||||
|
yield _floatstr(o)
|
||||||
|
else:
|
||||||
|
for_json = _for_json and getattr(o, 'for_json', None)
|
||||||
|
if for_json and callable(for_json):
|
||||||
|
for chunk in _iterencode(for_json(), _current_indent_level):
|
||||||
|
yield chunk
|
||||||
|
elif isinstance(o, list):
|
||||||
|
for chunk in _iterencode_list(o, _current_indent_level):
|
||||||
|
yield chunk
|
||||||
|
else:
|
||||||
|
_asdict = _namedtuple_as_object and getattr(o, '_asdict', None)
|
||||||
|
if _asdict and callable(_asdict):
|
||||||
|
for chunk in _iterencode_dict(_asdict(),
|
||||||
|
_current_indent_level):
|
||||||
|
yield chunk
|
||||||
|
elif (_tuple_as_array and isinstance(o, tuple)):
|
||||||
|
for chunk in _iterencode_list(o, _current_indent_level):
|
||||||
|
yield chunk
|
||||||
|
elif isinstance(o, dict):
|
||||||
|
for chunk in _iterencode_dict(o, _current_indent_level):
|
||||||
|
yield chunk
|
||||||
|
elif _use_decimal and isinstance(o, Decimal):
|
||||||
|
yield str(o)
|
||||||
|
else:
|
||||||
|
while _iterable_as_array:
|
||||||
|
# Markers are not checked here because it is valid for
|
||||||
|
# an iterable to return self.
|
||||||
|
try:
|
||||||
|
o = iter(o)
|
||||||
|
except TypeError:
|
||||||
|
break
|
||||||
|
for chunk in _iterencode_list(o, _current_indent_level):
|
||||||
|
yield chunk
|
||||||
|
return
|
||||||
|
if markers is not None:
|
||||||
|
markerid = id(o)
|
||||||
|
if markerid in markers:
|
||||||
|
raise ValueError("Circular reference detected")
|
||||||
|
markers[markerid] = o
|
||||||
|
o = _default(o)
|
||||||
|
for chunk in _iterencode(o, _current_indent_level):
|
||||||
|
yield chunk
|
||||||
|
if markers is not None:
|
||||||
|
del markers[markerid]
|
||||||
|
|
||||||
|
return _iterencode
|
|
@ -0,0 +1,53 @@
|
||||||
|
"""Error classes used by simplejson
|
||||||
|
"""
|
||||||
|
__all__ = ['JSONDecodeError']
|
||||||
|
|
||||||
|
|
||||||
|
def linecol(doc, pos):
|
||||||
|
lineno = doc.count('\n', 0, pos) + 1
|
||||||
|
if lineno == 1:
|
||||||
|
colno = pos + 1
|
||||||
|
else:
|
||||||
|
colno = pos - doc.rindex('\n', 0, pos)
|
||||||
|
return lineno, colno
|
||||||
|
|
||||||
|
|
||||||
|
def errmsg(msg, doc, pos, end=None):
|
||||||
|
lineno, colno = linecol(doc, pos)
|
||||||
|
msg = msg.replace('%r', repr(doc[pos:pos + 1]))
|
||||||
|
if end is None:
|
||||||
|
fmt = '%s: line %d column %d (char %d)'
|
||||||
|
return fmt % (msg, lineno, colno, pos)
|
||||||
|
endlineno, endcolno = linecol(doc, end)
|
||||||
|
fmt = '%s: line %d column %d - line %d column %d (char %d - %d)'
|
||||||
|
return fmt % (msg, lineno, colno, endlineno, endcolno, pos, end)
|
||||||
|
|
||||||
|
|
||||||
|
class JSONDecodeError(ValueError):
|
||||||
|
"""Subclass of ValueError with the following additional properties:
|
||||||
|
|
||||||
|
msg: The unformatted error message
|
||||||
|
doc: The JSON document being parsed
|
||||||
|
pos: The start index of doc where parsing failed
|
||||||
|
end: The end index of doc where parsing failed (may be None)
|
||||||
|
lineno: The line corresponding to pos
|
||||||
|
colno: The column corresponding to pos
|
||||||
|
endlineno: The line corresponding to end (may be None)
|
||||||
|
endcolno: The column corresponding to end (may be None)
|
||||||
|
|
||||||
|
"""
|
||||||
|
# Note that this exception is used from _speedups
|
||||||
|
def __init__(self, msg, doc, pos, end=None):
|
||||||
|
ValueError.__init__(self, errmsg(msg, doc, pos, end=end))
|
||||||
|
self.msg = msg
|
||||||
|
self.doc = doc
|
||||||
|
self.pos = pos
|
||||||
|
self.end = end
|
||||||
|
self.lineno, self.colno = linecol(doc, pos)
|
||||||
|
if end is not None:
|
||||||
|
self.endlineno, self.endcolno = linecol(doc, end)
|
||||||
|
else:
|
||||||
|
self.endlineno, self.endcolno = None, None
|
||||||
|
|
||||||
|
def __reduce__(self):
|
||||||
|
return self.__class__, (self.msg, self.doc, self.pos, self.end)
|
|
@ -0,0 +1,103 @@
|
||||||
|
"""Drop-in replacement for collections.OrderedDict by Raymond Hettinger
|
||||||
|
|
||||||
|
http://code.activestate.com/recipes/576693/
|
||||||
|
|
||||||
|
"""
|
||||||
|
from UserDict import DictMixin
|
||||||
|
|
||||||
|
class OrderedDict(dict, DictMixin):
|
||||||
|
|
||||||
|
def __init__(self, *args, **kwds):
|
||||||
|
if len(args) > 1:
|
||||||
|
raise TypeError('expected at most 1 arguments, got %d' % len(args))
|
||||||
|
try:
|
||||||
|
self.__end
|
||||||
|
except AttributeError:
|
||||||
|
self.clear()
|
||||||
|
self.update(*args, **kwds)
|
||||||
|
|
||||||
|
def clear(self):
|
||||||
|
self.__end = end = []
|
||||||
|
end += [None, end, end] # sentinel node for doubly linked list
|
||||||
|
self.__map = {} # key --> [key, prev, next]
|
||||||
|
dict.clear(self)
|
||||||
|
|
||||||
|
def __setitem__(self, key, value):
|
||||||
|
if key not in self:
|
||||||
|
end = self.__end
|
||||||
|
curr = end[1]
|
||||||
|
curr[2] = end[1] = self.__map[key] = [key, curr, end]
|
||||||
|
dict.__setitem__(self, key, value)
|
||||||
|
|
||||||
|
def __delitem__(self, key):
|
||||||
|
dict.__delitem__(self, key)
|
||||||
|
key, prev, next = self.__map.pop(key)
|
||||||
|
prev[2] = next
|
||||||
|
next[1] = prev
|
||||||
|
|
||||||
|
def __iter__(self):
|
||||||
|
end = self.__end
|
||||||
|
curr = end[2]
|
||||||
|
while curr is not end:
|
||||||
|
yield curr[0]
|
||||||
|
curr = curr[2]
|
||||||
|
|
||||||
|
def __reversed__(self):
|
||||||
|
end = self.__end
|
||||||
|
curr = end[1]
|
||||||
|
while curr is not end:
|
||||||
|
yield curr[0]
|
||||||
|
curr = curr[1]
|
||||||
|
|
||||||
|
def popitem(self, last=True):
|
||||||
|
if not self:
|
||||||
|
raise KeyError('dictionary is empty')
|
||||||
|
key = reversed(self).next() if last else iter(self).next()
|
||||||
|
value = self.pop(key)
|
||||||
|
return key, value
|
||||||
|
|
||||||
|
def __reduce__(self):
|
||||||
|
items = [[k, self[k]] for k in self]
|
||||||
|
tmp = self.__map, self.__end
|
||||||
|
del self.__map, self.__end
|
||||||
|
inst_dict = vars(self).copy()
|
||||||
|
self.__map, self.__end = tmp
|
||||||
|
if inst_dict:
|
||||||
|
return (self.__class__, (items,), inst_dict)
|
||||||
|
return self.__class__, (items,)
|
||||||
|
|
||||||
|
def keys(self):
|
||||||
|
return list(self)
|
||||||
|
|
||||||
|
setdefault = DictMixin.setdefault
|
||||||
|
update = DictMixin.update
|
||||||
|
pop = DictMixin.pop
|
||||||
|
values = DictMixin.values
|
||||||
|
items = DictMixin.items
|
||||||
|
iterkeys = DictMixin.iterkeys
|
||||||
|
itervalues = DictMixin.itervalues
|
||||||
|
iteritems = DictMixin.iteritems
|
||||||
|
|
||||||
|
def __repr__(self):
|
||||||
|
if not self:
|
||||||
|
return '%s()' % (self.__class__.__name__,)
|
||||||
|
return '%s(%r)' % (self.__class__.__name__, self.items())
|
||||||
|
|
||||||
|
def copy(self):
|
||||||
|
return self.__class__(self)
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def fromkeys(cls, iterable, value=None):
|
||||||
|
d = cls()
|
||||||
|
for key in iterable:
|
||||||
|
d[key] = value
|
||||||
|
return d
|
||||||
|
|
||||||
|
def __eq__(self, other):
|
||||||
|
if isinstance(other, OrderedDict):
|
||||||
|
return len(self)==len(other) and \
|
||||||
|
all(p==q for p, q in zip(self.items(), other.items()))
|
||||||
|
return dict.__eq__(self, other)
|
||||||
|
|
||||||
|
def __ne__(self, other):
|
||||||
|
return not self == other
|
|
@ -0,0 +1,9 @@
|
||||||
|
"""Implementation of RawJSON
|
||||||
|
"""
|
||||||
|
|
||||||
|
class RawJSON(object):
|
||||||
|
"""Wrap an encoded JSON document for direct embedding in the output
|
||||||
|
|
||||||
|
"""
|
||||||
|
def __init__(self, encoded_json):
|
||||||
|
self.encoded_json = encoded_json
|
|
@ -0,0 +1,85 @@
|
||||||
|
"""JSON token scanner
|
||||||
|
"""
|
||||||
|
import re
|
||||||
|
from .errors import JSONDecodeError
|
||||||
|
def _import_c_make_scanner():
|
||||||
|
try:
|
||||||
|
from ._speedups import make_scanner
|
||||||
|
return make_scanner
|
||||||
|
except ImportError:
|
||||||
|
return None
|
||||||
|
c_make_scanner = _import_c_make_scanner()
|
||||||
|
|
||||||
|
__all__ = ['make_scanner', 'JSONDecodeError']
|
||||||
|
|
||||||
|
NUMBER_RE = re.compile(
|
||||||
|
r'(-?(?:0|[1-9]\d*))(\.\d+)?([eE][-+]?\d+)?',
|
||||||
|
(re.VERBOSE | re.MULTILINE | re.DOTALL))
|
||||||
|
|
||||||
|
|
||||||
|
def py_make_scanner(context):
|
||||||
|
parse_object = context.parse_object
|
||||||
|
parse_array = context.parse_array
|
||||||
|
parse_string = context.parse_string
|
||||||
|
match_number = NUMBER_RE.match
|
||||||
|
encoding = context.encoding
|
||||||
|
strict = context.strict
|
||||||
|
parse_float = context.parse_float
|
||||||
|
parse_int = context.parse_int
|
||||||
|
parse_constant = context.parse_constant
|
||||||
|
object_hook = context.object_hook
|
||||||
|
object_pairs_hook = context.object_pairs_hook
|
||||||
|
memo = context.memo
|
||||||
|
|
||||||
|
def _scan_once(string, idx):
|
||||||
|
errmsg = 'Expecting value'
|
||||||
|
try:
|
||||||
|
nextchar = string[idx]
|
||||||
|
except IndexError:
|
||||||
|
raise JSONDecodeError(errmsg, string, idx)
|
||||||
|
|
||||||
|
if nextchar == '"':
|
||||||
|
return parse_string(string, idx + 1, encoding, strict)
|
||||||
|
elif nextchar == '{':
|
||||||
|
return parse_object((string, idx + 1), encoding, strict,
|
||||||
|
_scan_once, object_hook, object_pairs_hook, memo)
|
||||||
|
elif nextchar == '[':
|
||||||
|
return parse_array((string, idx + 1), _scan_once)
|
||||||
|
elif nextchar == 'n' and string[idx:idx + 4] == 'null':
|
||||||
|
return None, idx + 4
|
||||||
|
elif nextchar == 't' and string[idx:idx + 4] == 'true':
|
||||||
|
return True, idx + 4
|
||||||
|
elif nextchar == 'f' and string[idx:idx + 5] == 'false':
|
||||||
|
return False, idx + 5
|
||||||
|
|
||||||
|
m = match_number(string, idx)
|
||||||
|
if m is not None:
|
||||||
|
integer, frac, exp = m.groups()
|
||||||
|
if frac or exp:
|
||||||
|
res = parse_float(integer + (frac or '') + (exp or ''))
|
||||||
|
else:
|
||||||
|
res = parse_int(integer)
|
||||||
|
return res, m.end()
|
||||||
|
elif nextchar == 'N' and string[idx:idx + 3] == 'NaN':
|
||||||
|
return parse_constant('NaN'), idx + 3
|
||||||
|
elif nextchar == 'I' and string[idx:idx + 8] == 'Infinity':
|
||||||
|
return parse_constant('Infinity'), idx + 8
|
||||||
|
elif nextchar == '-' and string[idx:idx + 9] == '-Infinity':
|
||||||
|
return parse_constant('-Infinity'), idx + 9
|
||||||
|
else:
|
||||||
|
raise JSONDecodeError(errmsg, string, idx)
|
||||||
|
|
||||||
|
def scan_once(string, idx):
|
||||||
|
if idx < 0:
|
||||||
|
# Ensure the same behavior as the C speedup, otherwise
|
||||||
|
# this would work for *some* negative string indices due
|
||||||
|
# to the behavior of __getitem__ for strings. #98
|
||||||
|
raise JSONDecodeError('Expecting value', string, idx)
|
||||||
|
try:
|
||||||
|
return _scan_once(string, idx)
|
||||||
|
finally:
|
||||||
|
memo.clear()
|
||||||
|
|
||||||
|
return scan_once
|
||||||
|
|
||||||
|
make_scanner = c_make_scanner or py_make_scanner
|
|
@ -0,0 +1,74 @@
|
||||||
|
from __future__ import absolute_import
|
||||||
|
import unittest
|
||||||
|
import sys
|
||||||
|
import os
|
||||||
|
|
||||||
|
|
||||||
|
class NoExtensionTestSuite(unittest.TestSuite):
|
||||||
|
def run(self, result):
|
||||||
|
import simplejson
|
||||||
|
simplejson._toggle_speedups(False)
|
||||||
|
result = unittest.TestSuite.run(self, result)
|
||||||
|
simplejson._toggle_speedups(True)
|
||||||
|
return result
|
||||||
|
|
||||||
|
|
||||||
|
class TestMissingSpeedups(unittest.TestCase):
|
||||||
|
def runTest(self):
|
||||||
|
if hasattr(sys, 'pypy_translation_info'):
|
||||||
|
"PyPy doesn't need speedups! :)"
|
||||||
|
elif hasattr(self, 'skipTest'):
|
||||||
|
self.skipTest('_speedups.so is missing!')
|
||||||
|
|
||||||
|
|
||||||
|
def additional_tests(suite=None):
|
||||||
|
import simplejson
|
||||||
|
import simplejson.encoder
|
||||||
|
import simplejson.decoder
|
||||||
|
if suite is None:
|
||||||
|
suite = unittest.TestSuite()
|
||||||
|
try:
|
||||||
|
import doctest
|
||||||
|
except ImportError:
|
||||||
|
if sys.version_info < (2, 7):
|
||||||
|
# doctests in 2.6 depends on cStringIO
|
||||||
|
return suite
|
||||||
|
raise
|
||||||
|
for mod in (simplejson, simplejson.encoder, simplejson.decoder):
|
||||||
|
suite.addTest(doctest.DocTestSuite(mod))
|
||||||
|
suite.addTest(doctest.DocFileSuite('../../index.rst'))
|
||||||
|
return suite
|
||||||
|
|
||||||
|
|
||||||
|
def all_tests_suite():
|
||||||
|
def get_suite():
|
||||||
|
suite_names = [
|
||||||
|
'simplejson.tests.%s' % (os.path.splitext(f)[0],)
|
||||||
|
for f in os.listdir(os.path.dirname(__file__))
|
||||||
|
if f.startswith('test_') and f.endswith('.py')
|
||||||
|
]
|
||||||
|
return additional_tests(
|
||||||
|
unittest.TestLoader().loadTestsFromNames(suite_names))
|
||||||
|
suite = get_suite()
|
||||||
|
import simplejson
|
||||||
|
if simplejson._import_c_make_encoder() is None:
|
||||||
|
suite.addTest(TestMissingSpeedups())
|
||||||
|
else:
|
||||||
|
suite = unittest.TestSuite([
|
||||||
|
suite,
|
||||||
|
NoExtensionTestSuite([get_suite()]),
|
||||||
|
])
|
||||||
|
return suite
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
runner = unittest.TextTestRunner(verbosity=1 + sys.argv.count('-v'))
|
||||||
|
suite = all_tests_suite()
|
||||||
|
raise SystemExit(not runner.run(suite).wasSuccessful())
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
import os
|
||||||
|
import sys
|
||||||
|
sys.path.insert(0, os.path.dirname(os.path.dirname(os.path.dirname(os.path.abspath(__file__)))))
|
||||||
|
main()
|
|
@ -0,0 +1,67 @@
|
||||||
|
from unittest import TestCase
|
||||||
|
|
||||||
|
import simplejson as json
|
||||||
|
|
||||||
|
|
||||||
|
class TestBigintAsString(TestCase):
|
||||||
|
# Python 2.5, at least the one that ships on Mac OS X, calculates
|
||||||
|
# 2 ** 53 as 0! It manages to calculate 1 << 53 correctly.
|
||||||
|
values = [(200, 200),
|
||||||
|
((1 << 53) - 1, 9007199254740991),
|
||||||
|
((1 << 53), '9007199254740992'),
|
||||||
|
((1 << 53) + 1, '9007199254740993'),
|
||||||
|
(-100, -100),
|
||||||
|
((-1 << 53), '-9007199254740992'),
|
||||||
|
((-1 << 53) - 1, '-9007199254740993'),
|
||||||
|
((-1 << 53) + 1, -9007199254740991)]
|
||||||
|
|
||||||
|
options = (
|
||||||
|
{"bigint_as_string": True},
|
||||||
|
{"int_as_string_bitcount": 53}
|
||||||
|
)
|
||||||
|
|
||||||
|
def test_ints(self):
|
||||||
|
for opts in self.options:
|
||||||
|
for val, expect in self.values:
|
||||||
|
self.assertEqual(
|
||||||
|
val,
|
||||||
|
json.loads(json.dumps(val)))
|
||||||
|
self.assertEqual(
|
||||||
|
expect,
|
||||||
|
json.loads(json.dumps(val, **opts)))
|
||||||
|
|
||||||
|
def test_lists(self):
|
||||||
|
for opts in self.options:
|
||||||
|
for val, expect in self.values:
|
||||||
|
val = [val, val]
|
||||||
|
expect = [expect, expect]
|
||||||
|
self.assertEqual(
|
||||||
|
val,
|
||||||
|
json.loads(json.dumps(val)))
|
||||||
|
self.assertEqual(
|
||||||
|
expect,
|
||||||
|
json.loads(json.dumps(val, **opts)))
|
||||||
|
|
||||||
|
def test_dicts(self):
|
||||||
|
for opts in self.options:
|
||||||
|
for val, expect in self.values:
|
||||||
|
val = {'k': val}
|
||||||
|
expect = {'k': expect}
|
||||||
|
self.assertEqual(
|
||||||
|
val,
|
||||||
|
json.loads(json.dumps(val)))
|
||||||
|
self.assertEqual(
|
||||||
|
expect,
|
||||||
|
json.loads(json.dumps(val, **opts)))
|
||||||
|
|
||||||
|
def test_dict_keys(self):
|
||||||
|
for opts in self.options:
|
||||||
|
for val, _ in self.values:
|
||||||
|
expect = {str(val): 'value'}
|
||||||
|
val = {val: 'value'}
|
||||||
|
self.assertEqual(
|
||||||
|
expect,
|
||||||
|
json.loads(json.dumps(val)))
|
||||||
|
self.assertEqual(
|
||||||
|
expect,
|
||||||
|
json.loads(json.dumps(val, **opts)))
|
|
@ -0,0 +1,73 @@
|
||||||
|
from unittest import TestCase
|
||||||
|
|
||||||
|
import simplejson as json
|
||||||
|
|
||||||
|
|
||||||
|
class TestBitSizeIntAsString(TestCase):
|
||||||
|
# Python 2.5, at least the one that ships on Mac OS X, calculates
|
||||||
|
# 2 ** 31 as 0! It manages to calculate 1 << 31 correctly.
|
||||||
|
values = [
|
||||||
|
(200, 200),
|
||||||
|
((1 << 31) - 1, (1 << 31) - 1),
|
||||||
|
((1 << 31), str(1 << 31)),
|
||||||
|
((1 << 31) + 1, str((1 << 31) + 1)),
|
||||||
|
(-100, -100),
|
||||||
|
((-1 << 31), str(-1 << 31)),
|
||||||
|
((-1 << 31) - 1, str((-1 << 31) - 1)),
|
||||||
|
((-1 << 31) + 1, (-1 << 31) + 1),
|
||||||
|
]
|
||||||
|
|
||||||
|
def test_invalid_counts(self):
|
||||||
|
for n in ['foo', -1, 0, 1.0]:
|
||||||
|
self.assertRaises(
|
||||||
|
TypeError,
|
||||||
|
json.dumps, 0, int_as_string_bitcount=n)
|
||||||
|
|
||||||
|
def test_ints_outside_range_fails(self):
|
||||||
|
self.assertNotEqual(
|
||||||
|
str(1 << 15),
|
||||||
|
json.loads(json.dumps(1 << 15, int_as_string_bitcount=16)),
|
||||||
|
)
|
||||||
|
|
||||||
|
def test_ints(self):
|
||||||
|
for val, expect in self.values:
|
||||||
|
self.assertEqual(
|
||||||
|
val,
|
||||||
|
json.loads(json.dumps(val)))
|
||||||
|
self.assertEqual(
|
||||||
|
expect,
|
||||||
|
json.loads(json.dumps(val, int_as_string_bitcount=31)),
|
||||||
|
)
|
||||||
|
|
||||||
|
def test_lists(self):
|
||||||
|
for val, expect in self.values:
|
||||||
|
val = [val, val]
|
||||||
|
expect = [expect, expect]
|
||||||
|
self.assertEqual(
|
||||||
|
val,
|
||||||
|
json.loads(json.dumps(val)))
|
||||||
|
self.assertEqual(
|
||||||
|
expect,
|
||||||
|
json.loads(json.dumps(val, int_as_string_bitcount=31)))
|
||||||
|
|
||||||
|
def test_dicts(self):
|
||||||
|
for val, expect in self.values:
|
||||||
|
val = {'k': val}
|
||||||
|
expect = {'k': expect}
|
||||||
|
self.assertEqual(
|
||||||
|
val,
|
||||||
|
json.loads(json.dumps(val)))
|
||||||
|
self.assertEqual(
|
||||||
|
expect,
|
||||||
|
json.loads(json.dumps(val, int_as_string_bitcount=31)))
|
||||||
|
|
||||||
|
def test_dict_keys(self):
|
||||||
|
for val, _ in self.values:
|
||||||
|
expect = {str(val): 'value'}
|
||||||
|
val = {val: 'value'}
|
||||||
|
self.assertEqual(
|
||||||
|
expect,
|
||||||
|
json.loads(json.dumps(val)))
|
||||||
|
self.assertEqual(
|
||||||
|
expect,
|
||||||
|
json.loads(json.dumps(val, int_as_string_bitcount=31)))
|
|
@ -0,0 +1,30 @@
|
||||||
|
from unittest import TestCase
|
||||||
|
import simplejson as json
|
||||||
|
|
||||||
|
def default_iterable(obj):
|
||||||
|
return list(obj)
|
||||||
|
|
||||||
|
class TestCheckCircular(TestCase):
|
||||||
|
def test_circular_dict(self):
|
||||||
|
dct = {}
|
||||||
|
dct['a'] = dct
|
||||||
|
self.assertRaises(ValueError, json.dumps, dct)
|
||||||
|
|
||||||
|
def test_circular_list(self):
|
||||||
|
lst = []
|
||||||
|
lst.append(lst)
|
||||||
|
self.assertRaises(ValueError, json.dumps, lst)
|
||||||
|
|
||||||
|
def test_circular_composite(self):
|
||||||
|
dct2 = {}
|
||||||
|
dct2['a'] = []
|
||||||
|
dct2['a'].append(dct2)
|
||||||
|
self.assertRaises(ValueError, json.dumps, dct2)
|
||||||
|
|
||||||
|
def test_circular_default(self):
|
||||||
|
json.dumps([set()], default=default_iterable)
|
||||||
|
self.assertRaises(TypeError, json.dumps, [set()])
|
||||||
|
|
||||||
|
def test_circular_off_default(self):
|
||||||
|
json.dumps([set()], default=default_iterable, check_circular=False)
|
||||||
|
self.assertRaises(TypeError, json.dumps, [set()], check_circular=False)
|
|
@ -0,0 +1,71 @@
|
||||||
|
import decimal
|
||||||
|
from decimal import Decimal
|
||||||
|
from unittest import TestCase
|
||||||
|
from simplejson.compat import StringIO, reload_module
|
||||||
|
|
||||||
|
import simplejson as json
|
||||||
|
|
||||||
|
class TestDecimal(TestCase):
|
||||||
|
NUMS = "1.0", "10.00", "1.1", "1234567890.1234567890", "500"
|
||||||
|
def dumps(self, obj, **kw):
|
||||||
|
sio = StringIO()
|
||||||
|
json.dump(obj, sio, **kw)
|
||||||
|
res = json.dumps(obj, **kw)
|
||||||
|
self.assertEqual(res, sio.getvalue())
|
||||||
|
return res
|
||||||
|
|
||||||
|
def loads(self, s, **kw):
|
||||||
|
sio = StringIO(s)
|
||||||
|
res = json.loads(s, **kw)
|
||||||
|
self.assertEqual(res, json.load(sio, **kw))
|
||||||
|
return res
|
||||||
|
|
||||||
|
def test_decimal_encode(self):
|
||||||
|
for d in map(Decimal, self.NUMS):
|
||||||
|
self.assertEqual(self.dumps(d, use_decimal=True), str(d))
|
||||||
|
|
||||||
|
def test_decimal_decode(self):
|
||||||
|
for s in self.NUMS:
|
||||||
|
self.assertEqual(self.loads(s, parse_float=Decimal), Decimal(s))
|
||||||
|
|
||||||
|
def test_stringify_key(self):
|
||||||
|
for d in map(Decimal, self.NUMS):
|
||||||
|
v = {d: d}
|
||||||
|
self.assertEqual(
|
||||||
|
self.loads(
|
||||||
|
self.dumps(v, use_decimal=True), parse_float=Decimal),
|
||||||
|
{str(d): d})
|
||||||
|
|
||||||
|
def test_decimal_roundtrip(self):
|
||||||
|
for d in map(Decimal, self.NUMS):
|
||||||
|
# The type might not be the same (int and Decimal) but they
|
||||||
|
# should still compare equal.
|
||||||
|
for v in [d, [d], {'': d}]:
|
||||||
|
self.assertEqual(
|
||||||
|
self.loads(
|
||||||
|
self.dumps(v, use_decimal=True), parse_float=Decimal),
|
||||||
|
v)
|
||||||
|
|
||||||
|
def test_decimal_defaults(self):
|
||||||
|
d = Decimal('1.1')
|
||||||
|
# use_decimal=True is the default
|
||||||
|
self.assertRaises(TypeError, json.dumps, d, use_decimal=False)
|
||||||
|
self.assertEqual('1.1', json.dumps(d))
|
||||||
|
self.assertEqual('1.1', json.dumps(d, use_decimal=True))
|
||||||
|
self.assertRaises(TypeError, json.dump, d, StringIO(),
|
||||||
|
use_decimal=False)
|
||||||
|
sio = StringIO()
|
||||||
|
json.dump(d, sio)
|
||||||
|
self.assertEqual('1.1', sio.getvalue())
|
||||||
|
sio = StringIO()
|
||||||
|
json.dump(d, sio, use_decimal=True)
|
||||||
|
self.assertEqual('1.1', sio.getvalue())
|
||||||
|
|
||||||
|
def test_decimal_reload(self):
|
||||||
|
# Simulate a subinterpreter that reloads the Python modules but not
|
||||||
|
# the C code https://github.com/simplejson/simplejson/issues/34
|
||||||
|
global Decimal
|
||||||
|
Decimal = reload_module(decimal).Decimal
|
||||||
|
import simplejson.encoder
|
||||||
|
simplejson.encoder.Decimal = Decimal
|
||||||
|
self.test_decimal_roundtrip()
|
|
@ -0,0 +1,119 @@
|
||||||
|
from __future__ import absolute_import
|
||||||
|
import decimal
|
||||||
|
from unittest import TestCase
|
||||||
|
|
||||||
|
import simplejson as json
|
||||||
|
from simplejson.compat import StringIO, b, binary_type
|
||||||
|
from simplejson import OrderedDict
|
||||||
|
|
||||||
|
class MisbehavingBytesSubtype(binary_type):
|
||||||
|
def decode(self, encoding=None):
|
||||||
|
return "bad decode"
|
||||||
|
def __str__(self):
|
||||||
|
return "bad __str__"
|
||||||
|
def __bytes__(self):
|
||||||
|
return b("bad __bytes__")
|
||||||
|
|
||||||
|
class TestDecode(TestCase):
|
||||||
|
if not hasattr(TestCase, 'assertIs'):
|
||||||
|
def assertIs(self, a, b):
|
||||||
|
self.assertTrue(a is b, '%r is %r' % (a, b))
|
||||||
|
|
||||||
|
def test_decimal(self):
|
||||||
|
rval = json.loads('1.1', parse_float=decimal.Decimal)
|
||||||
|
self.assertTrue(isinstance(rval, decimal.Decimal))
|
||||||
|
self.assertEqual(rval, decimal.Decimal('1.1'))
|
||||||
|
|
||||||
|
def test_float(self):
|
||||||
|
rval = json.loads('1', parse_int=float)
|
||||||
|
self.assertTrue(isinstance(rval, float))
|
||||||
|
self.assertEqual(rval, 1.0)
|
||||||
|
|
||||||
|
def test_decoder_optimizations(self):
|
||||||
|
# Several optimizations were made that skip over calls to
|
||||||
|
# the whitespace regex, so this test is designed to try and
|
||||||
|
# exercise the uncommon cases. The array cases are already covered.
|
||||||
|
rval = json.loads('{ "key" : "value" , "k":"v" }')
|
||||||
|
self.assertEqual(rval, {"key":"value", "k":"v"})
|
||||||
|
|
||||||
|
def test_empty_objects(self):
|
||||||
|
s = '{}'
|
||||||
|
self.assertEqual(json.loads(s), eval(s))
|
||||||
|
s = '[]'
|
||||||
|
self.assertEqual(json.loads(s), eval(s))
|
||||||
|
s = '""'
|
||||||
|
self.assertEqual(json.loads(s), eval(s))
|
||||||
|
|
||||||
|
def test_object_pairs_hook(self):
|
||||||
|
s = '{"xkd":1, "kcw":2, "art":3, "hxm":4, "qrt":5, "pad":6, "hoy":7}'
|
||||||
|
p = [("xkd", 1), ("kcw", 2), ("art", 3), ("hxm", 4),
|
||||||
|
("qrt", 5), ("pad", 6), ("hoy", 7)]
|
||||||
|
self.assertEqual(json.loads(s), eval(s))
|
||||||
|
self.assertEqual(json.loads(s, object_pairs_hook=lambda x: x), p)
|
||||||
|
self.assertEqual(json.load(StringIO(s),
|
||||||
|
object_pairs_hook=lambda x: x), p)
|
||||||
|
od = json.loads(s, object_pairs_hook=OrderedDict)
|
||||||
|
self.assertEqual(od, OrderedDict(p))
|
||||||
|
self.assertEqual(type(od), OrderedDict)
|
||||||
|
# the object_pairs_hook takes priority over the object_hook
|
||||||
|
self.assertEqual(json.loads(s,
|
||||||
|
object_pairs_hook=OrderedDict,
|
||||||
|
object_hook=lambda x: None),
|
||||||
|
OrderedDict(p))
|
||||||
|
|
||||||
|
def check_keys_reuse(self, source, loads):
|
||||||
|
rval = loads(source)
|
||||||
|
(a, b), (c, d) = sorted(rval[0]), sorted(rval[1])
|
||||||
|
self.assertIs(a, c)
|
||||||
|
self.assertIs(b, d)
|
||||||
|
|
||||||
|
def test_keys_reuse_str(self):
|
||||||
|
s = u'[{"a_key": 1, "b_\xe9": 2}, {"a_key": 3, "b_\xe9": 4}]'.encode('utf8')
|
||||||
|
self.check_keys_reuse(s, json.loads)
|
||||||
|
|
||||||
|
def test_keys_reuse_unicode(self):
|
||||||
|
s = u'[{"a_key": 1, "b_\xe9": 2}, {"a_key": 3, "b_\xe9": 4}]'
|
||||||
|
self.check_keys_reuse(s, json.loads)
|
||||||
|
|
||||||
|
def test_empty_strings(self):
|
||||||
|
self.assertEqual(json.loads('""'), "")
|
||||||
|
self.assertEqual(json.loads(u'""'), u"")
|
||||||
|
self.assertEqual(json.loads('[""]'), [""])
|
||||||
|
self.assertEqual(json.loads(u'[""]'), [u""])
|
||||||
|
|
||||||
|
def test_raw_decode(self):
|
||||||
|
cls = json.decoder.JSONDecoder
|
||||||
|
self.assertEqual(
|
||||||
|
({'a': {}}, 9),
|
||||||
|
cls().raw_decode("{\"a\": {}}"))
|
||||||
|
# http://code.google.com/p/simplejson/issues/detail?id=85
|
||||||
|
self.assertEqual(
|
||||||
|
({'a': {}}, 9),
|
||||||
|
cls(object_pairs_hook=dict).raw_decode("{\"a\": {}}"))
|
||||||
|
# https://github.com/simplejson/simplejson/pull/38
|
||||||
|
self.assertEqual(
|
||||||
|
({'a': {}}, 11),
|
||||||
|
cls().raw_decode(" \n{\"a\": {}}"))
|
||||||
|
|
||||||
|
def test_bytes_decode(self):
|
||||||
|
cls = json.decoder.JSONDecoder
|
||||||
|
data = b('"\xe2\x82\xac"')
|
||||||
|
self.assertEqual(cls().decode(data), u'\u20ac')
|
||||||
|
self.assertEqual(cls(encoding='latin1').decode(data), u'\xe2\x82\xac')
|
||||||
|
self.assertEqual(cls(encoding=None).decode(data), u'\u20ac')
|
||||||
|
|
||||||
|
data = MisbehavingBytesSubtype(b('"\xe2\x82\xac"'))
|
||||||
|
self.assertEqual(cls().decode(data), u'\u20ac')
|
||||||
|
self.assertEqual(cls(encoding='latin1').decode(data), u'\xe2\x82\xac')
|
||||||
|
self.assertEqual(cls(encoding=None).decode(data), u'\u20ac')
|
||||||
|
|
||||||
|
def test_bounds_checking(self):
|
||||||
|
# https://github.com/simplejson/simplejson/issues/98
|
||||||
|
j = json.decoder.JSONDecoder()
|
||||||
|
for i in [4, 5, 6, -1, -2, -3, -4, -5, -6]:
|
||||||
|
self.assertRaises(ValueError, j.scan_once, '1234', i)
|
||||||
|
self.assertRaises(ValueError, j.raw_decode, '1234', i)
|
||||||
|
x, y = sorted(['128931233', '472389423'], key=id)
|
||||||
|
diff = id(x) - id(y)
|
||||||
|
self.assertRaises(ValueError, j.scan_once, y, diff)
|
||||||
|
self.assertRaises(ValueError, j.raw_decode, y, i)
|
|
@ -0,0 +1,9 @@
|
||||||
|
from unittest import TestCase
|
||||||
|
|
||||||
|
import simplejson as json
|
||||||
|
|
||||||
|
class TestDefault(TestCase):
|
||||||
|
def test_default(self):
|
||||||
|
self.assertEqual(
|
||||||
|
json.dumps(type, default=repr),
|
||||||
|
json.dumps(repr(type)))
|
|
@ -0,0 +1,249 @@
|
||||||
|
from unittest import TestCase
|
||||||
|
from simplejson.compat import StringIO, long_type, b, binary_type, text_type, PY3
|
||||||
|
import simplejson as json
|
||||||
|
|
||||||
|
class MisbehavingTextSubtype(text_type):
|
||||||
|
def __str__(self):
|
||||||
|
return "FAIL!"
|
||||||
|
|
||||||
|
class MisbehavingBytesSubtype(binary_type):
|
||||||
|
def decode(self, encoding=None):
|
||||||
|
return "bad decode"
|
||||||
|
def __str__(self):
|
||||||
|
return "bad __str__"
|
||||||
|
def __bytes__(self):
|
||||||
|
return b("bad __bytes__")
|
||||||
|
|
||||||
|
def as_text_type(s):
|
||||||
|
if PY3 and isinstance(s, bytes):
|
||||||
|
return s.decode('ascii')
|
||||||
|
return s
|
||||||
|
|
||||||
|
def decode_iso_8859_15(b):
|
||||||
|
return b.decode('iso-8859-15')
|
||||||
|
|
||||||
|
class TestDump(TestCase):
|
||||||
|
def test_dump(self):
|
||||||
|
sio = StringIO()
|
||||||
|
json.dump({}, sio)
|
||||||
|
self.assertEqual(sio.getvalue(), '{}')
|
||||||
|
|
||||||
|
def test_constants(self):
|
||||||
|
for c in [None, True, False]:
|
||||||
|
self.assertTrue(json.loads(json.dumps(c)) is c)
|
||||||
|
self.assertTrue(json.loads(json.dumps([c]))[0] is c)
|
||||||
|
self.assertTrue(json.loads(json.dumps({'a': c}))['a'] is c)
|
||||||
|
|
||||||
|
def test_stringify_key(self):
|
||||||
|
items = [(b('bytes'), 'bytes'),
|
||||||
|
(1.0, '1.0'),
|
||||||
|
(10, '10'),
|
||||||
|
(True, 'true'),
|
||||||
|
(False, 'false'),
|
||||||
|
(None, 'null'),
|
||||||
|
(long_type(100), '100')]
|
||||||
|
for k, expect in items:
|
||||||
|
self.assertEqual(
|
||||||
|
json.loads(json.dumps({k: expect})),
|
||||||
|
{expect: expect})
|
||||||
|
self.assertEqual(
|
||||||
|
json.loads(json.dumps({k: expect}, sort_keys=True)),
|
||||||
|
{expect: expect})
|
||||||
|
self.assertRaises(TypeError, json.dumps, {json: 1})
|
||||||
|
for v in [{}, {'other': 1}, {b('derp'): 1, 'herp': 2}]:
|
||||||
|
for sort_keys in [False, True]:
|
||||||
|
v0 = dict(v)
|
||||||
|
v0[json] = 1
|
||||||
|
v1 = dict((as_text_type(key), val) for (key, val) in v.items())
|
||||||
|
self.assertEqual(
|
||||||
|
json.loads(json.dumps(v0, skipkeys=True, sort_keys=sort_keys)),
|
||||||
|
v1)
|
||||||
|
self.assertEqual(
|
||||||
|
json.loads(json.dumps({'': v0}, skipkeys=True, sort_keys=sort_keys)),
|
||||||
|
{'': v1})
|
||||||
|
self.assertEqual(
|
||||||
|
json.loads(json.dumps([v0], skipkeys=True, sort_keys=sort_keys)),
|
||||||
|
[v1])
|
||||||
|
|
||||||
|
def test_dumps(self):
|
||||||
|
self.assertEqual(json.dumps({}), '{}')
|
||||||
|
|
||||||
|
def test_encode_truefalse(self):
|
||||||
|
self.assertEqual(json.dumps(
|
||||||
|
{True: False, False: True}, sort_keys=True),
|
||||||
|
'{"false": true, "true": false}')
|
||||||
|
self.assertEqual(
|
||||||
|
json.dumps(
|
||||||
|
{2: 3.0,
|
||||||
|
4.0: long_type(5),
|
||||||
|
False: 1,
|
||||||
|
long_type(6): True,
|
||||||
|
"7": 0},
|
||||||
|
sort_keys=True),
|
||||||
|
'{"2": 3.0, "4.0": 5, "6": true, "7": 0, "false": 1}')
|
||||||
|
|
||||||
|
def test_ordered_dict(self):
|
||||||
|
# http://bugs.python.org/issue6105
|
||||||
|
items = [('one', 1), ('two', 2), ('three', 3), ('four', 4), ('five', 5)]
|
||||||
|
s = json.dumps(json.OrderedDict(items))
|
||||||
|
self.assertEqual(
|
||||||
|
s,
|
||||||
|
'{"one": 1, "two": 2, "three": 3, "four": 4, "five": 5}')
|
||||||
|
|
||||||
|
def test_indent_unknown_type_acceptance(self):
|
||||||
|
"""
|
||||||
|
A test against the regression mentioned at `github issue 29`_.
|
||||||
|
|
||||||
|
The indent parameter should accept any type which pretends to be
|
||||||
|
an instance of int or long when it comes to being multiplied by
|
||||||
|
strings, even if it is not actually an int or long, for
|
||||||
|
backwards compatibility.
|
||||||
|
|
||||||
|
.. _github issue 29:
|
||||||
|
http://github.com/simplejson/simplejson/issue/29
|
||||||
|
"""
|
||||||
|
|
||||||
|
class AwesomeInt(object):
|
||||||
|
"""An awesome reimplementation of integers"""
|
||||||
|
|
||||||
|
def __init__(self, *args, **kwargs):
|
||||||
|
if len(args) > 0:
|
||||||
|
# [construct from literals, objects, etc.]
|
||||||
|
# ...
|
||||||
|
|
||||||
|
# Finally, if args[0] is an integer, store it
|
||||||
|
if isinstance(args[0], int):
|
||||||
|
self._int = args[0]
|
||||||
|
|
||||||
|
# [various methods]
|
||||||
|
|
||||||
|
def __mul__(self, other):
|
||||||
|
# [various ways to multiply AwesomeInt objects]
|
||||||
|
# ... finally, if the right-hand operand is not awesome enough,
|
||||||
|
# try to do a normal integer multiplication
|
||||||
|
if hasattr(self, '_int'):
|
||||||
|
return self._int * other
|
||||||
|
else:
|
||||||
|
raise NotImplementedError("To do non-awesome things with"
|
||||||
|
" this object, please construct it from an integer!")
|
||||||
|
|
||||||
|
s = json.dumps([0, 1, 2], indent=AwesomeInt(3))
|
||||||
|
self.assertEqual(s, '[\n 0,\n 1,\n 2\n]')
|
||||||
|
|
||||||
|
def test_accumulator(self):
|
||||||
|
# the C API uses an accumulator that collects after 100,000 appends
|
||||||
|
lst = [0] * 100000
|
||||||
|
self.assertEqual(json.loads(json.dumps(lst)), lst)
|
||||||
|
|
||||||
|
def test_sort_keys(self):
|
||||||
|
# https://github.com/simplejson/simplejson/issues/106
|
||||||
|
for num_keys in range(2, 32):
|
||||||
|
p = dict((str(x), x) for x in range(num_keys))
|
||||||
|
sio = StringIO()
|
||||||
|
json.dump(p, sio, sort_keys=True)
|
||||||
|
self.assertEqual(sio.getvalue(), json.dumps(p, sort_keys=True))
|
||||||
|
self.assertEqual(json.loads(sio.getvalue()), p)
|
||||||
|
|
||||||
|
def test_misbehaving_text_subtype(self):
|
||||||
|
# https://github.com/simplejson/simplejson/issues/185
|
||||||
|
text = "this is some text"
|
||||||
|
self.assertEqual(
|
||||||
|
json.dumps(MisbehavingTextSubtype(text)),
|
||||||
|
json.dumps(text)
|
||||||
|
)
|
||||||
|
self.assertEqual(
|
||||||
|
json.dumps([MisbehavingTextSubtype(text)]),
|
||||||
|
json.dumps([text])
|
||||||
|
)
|
||||||
|
self.assertEqual(
|
||||||
|
json.dumps({MisbehavingTextSubtype(text): 42}),
|
||||||
|
json.dumps({text: 42})
|
||||||
|
)
|
||||||
|
|
||||||
|
def test_misbehaving_bytes_subtype(self):
|
||||||
|
data = b("this is some data \xe2\x82\xac")
|
||||||
|
self.assertEqual(
|
||||||
|
json.dumps(MisbehavingBytesSubtype(data)),
|
||||||
|
json.dumps(data)
|
||||||
|
)
|
||||||
|
self.assertEqual(
|
||||||
|
json.dumps([MisbehavingBytesSubtype(data)]),
|
||||||
|
json.dumps([data])
|
||||||
|
)
|
||||||
|
self.assertEqual(
|
||||||
|
json.dumps({MisbehavingBytesSubtype(data): 42}),
|
||||||
|
json.dumps({data: 42})
|
||||||
|
)
|
||||||
|
|
||||||
|
def test_bytes_toplevel(self):
|
||||||
|
self.assertEqual(json.dumps(b('\xe2\x82\xac')), r'"\u20ac"')
|
||||||
|
self.assertRaises(UnicodeDecodeError, json.dumps, b('\xa4'))
|
||||||
|
self.assertEqual(json.dumps(b('\xa4'), encoding='iso-8859-1'),
|
||||||
|
r'"\u00a4"')
|
||||||
|
self.assertEqual(json.dumps(b('\xa4'), encoding='iso-8859-15'),
|
||||||
|
r'"\u20ac"')
|
||||||
|
if PY3:
|
||||||
|
self.assertRaises(TypeError, json.dumps, b('\xe2\x82\xac'),
|
||||||
|
encoding=None)
|
||||||
|
self.assertRaises(TypeError, json.dumps, b('\xa4'),
|
||||||
|
encoding=None)
|
||||||
|
self.assertEqual(json.dumps(b('\xa4'), encoding=None,
|
||||||
|
default=decode_iso_8859_15),
|
||||||
|
r'"\u20ac"')
|
||||||
|
else:
|
||||||
|
self.assertEqual(json.dumps(b('\xe2\x82\xac'), encoding=None),
|
||||||
|
r'"\u20ac"')
|
||||||
|
self.assertRaises(UnicodeDecodeError, json.dumps, b('\xa4'),
|
||||||
|
encoding=None)
|
||||||
|
self.assertRaises(UnicodeDecodeError, json.dumps, b('\xa4'),
|
||||||
|
encoding=None, default=decode_iso_8859_15)
|
||||||
|
|
||||||
|
def test_bytes_nested(self):
|
||||||
|
self.assertEqual(json.dumps([b('\xe2\x82\xac')]), r'["\u20ac"]')
|
||||||
|
self.assertRaises(UnicodeDecodeError, json.dumps, [b('\xa4')])
|
||||||
|
self.assertEqual(json.dumps([b('\xa4')], encoding='iso-8859-1'),
|
||||||
|
r'["\u00a4"]')
|
||||||
|
self.assertEqual(json.dumps([b('\xa4')], encoding='iso-8859-15'),
|
||||||
|
r'["\u20ac"]')
|
||||||
|
if PY3:
|
||||||
|
self.assertRaises(TypeError, json.dumps, [b('\xe2\x82\xac')],
|
||||||
|
encoding=None)
|
||||||
|
self.assertRaises(TypeError, json.dumps, [b('\xa4')],
|
||||||
|
encoding=None)
|
||||||
|
self.assertEqual(json.dumps([b('\xa4')], encoding=None,
|
||||||
|
default=decode_iso_8859_15),
|
||||||
|
r'["\u20ac"]')
|
||||||
|
else:
|
||||||
|
self.assertEqual(json.dumps([b('\xe2\x82\xac')], encoding=None),
|
||||||
|
r'["\u20ac"]')
|
||||||
|
self.assertRaises(UnicodeDecodeError, json.dumps, [b('\xa4')],
|
||||||
|
encoding=None)
|
||||||
|
self.assertRaises(UnicodeDecodeError, json.dumps, [b('\xa4')],
|
||||||
|
encoding=None, default=decode_iso_8859_15)
|
||||||
|
|
||||||
|
def test_bytes_key(self):
|
||||||
|
self.assertEqual(json.dumps({b('\xe2\x82\xac'): 42}), r'{"\u20ac": 42}')
|
||||||
|
self.assertRaises(UnicodeDecodeError, json.dumps, {b('\xa4'): 42})
|
||||||
|
self.assertEqual(json.dumps({b('\xa4'): 42}, encoding='iso-8859-1'),
|
||||||
|
r'{"\u00a4": 42}')
|
||||||
|
self.assertEqual(json.dumps({b('\xa4'): 42}, encoding='iso-8859-15'),
|
||||||
|
r'{"\u20ac": 42}')
|
||||||
|
if PY3:
|
||||||
|
self.assertRaises(TypeError, json.dumps, {b('\xe2\x82\xac'): 42},
|
||||||
|
encoding=None)
|
||||||
|
self.assertRaises(TypeError, json.dumps, {b('\xa4'): 42},
|
||||||
|
encoding=None)
|
||||||
|
self.assertRaises(TypeError, json.dumps, {b('\xa4'): 42},
|
||||||
|
encoding=None, default=decode_iso_8859_15)
|
||||||
|
self.assertEqual(json.dumps({b('\xa4'): 42}, encoding=None,
|
||||||
|
skipkeys=True),
|
||||||
|
r'{}')
|
||||||
|
else:
|
||||||
|
self.assertEqual(json.dumps({b('\xe2\x82\xac'): 42}, encoding=None),
|
||||||
|
r'{"\u20ac": 42}')
|
||||||
|
self.assertRaises(UnicodeDecodeError, json.dumps, {b('\xa4'): 42},
|
||||||
|
encoding=None)
|
||||||
|
self.assertRaises(UnicodeDecodeError, json.dumps, {b('\xa4'): 42},
|
||||||
|
encoding=None, default=decode_iso_8859_15)
|
||||||
|
self.assertRaises(UnicodeDecodeError, json.dumps, {b('\xa4'): 42},
|
||||||
|
encoding=None, skipkeys=True)
|
|
@ -0,0 +1,47 @@
|
||||||
|
from unittest import TestCase
|
||||||
|
|
||||||
|
import simplejson.encoder
|
||||||
|
from simplejson.compat import b
|
||||||
|
|
||||||
|
CASES = [
|
||||||
|
(u'/\\"\ucafe\ubabe\uab98\ufcde\ubcda\uef4a\x08\x0c\n\r\t`1~!@#$%^&*()_+-=[]{}|;:\',./<>?', '"/\\\\\\"\\ucafe\\ubabe\\uab98\\ufcde\\ubcda\\uef4a\\b\\f\\n\\r\\t`1~!@#$%^&*()_+-=[]{}|;:\',./<>?"'),
|
||||||
|
(u'\u0123\u4567\u89ab\ucdef\uabcd\uef4a', '"\\u0123\\u4567\\u89ab\\ucdef\\uabcd\\uef4a"'),
|
||||||
|
(u'controls', '"controls"'),
|
||||||
|
(u'\x08\x0c\n\r\t', '"\\b\\f\\n\\r\\t"'),
|
||||||
|
(u'{"object with 1 member":["array with 1 element"]}', '"{\\"object with 1 member\\":[\\"array with 1 element\\"]}"'),
|
||||||
|
(u' s p a c e d ', '" s p a c e d "'),
|
||||||
|
(u'\U0001d120', '"\\ud834\\udd20"'),
|
||||||
|
(u'\u03b1\u03a9', '"\\u03b1\\u03a9"'),
|
||||||
|
(b('\xce\xb1\xce\xa9'), '"\\u03b1\\u03a9"'),
|
||||||
|
(u'\u03b1\u03a9', '"\\u03b1\\u03a9"'),
|
||||||
|
(b('\xce\xb1\xce\xa9'), '"\\u03b1\\u03a9"'),
|
||||||
|
(u'\u03b1\u03a9', '"\\u03b1\\u03a9"'),
|
||||||
|
(u'\u03b1\u03a9', '"\\u03b1\\u03a9"'),
|
||||||
|
(u"`1~!@#$%^&*()_+-={':[,]}|;.</>?", '"`1~!@#$%^&*()_+-={\':[,]}|;.</>?"'),
|
||||||
|
(u'\x08\x0c\n\r\t', '"\\b\\f\\n\\r\\t"'),
|
||||||
|
(u'\u0123\u4567\u89ab\ucdef\uabcd\uef4a', '"\\u0123\\u4567\\u89ab\\ucdef\\uabcd\\uef4a"'),
|
||||||
|
]
|
||||||
|
|
||||||
|
class TestEncodeBaseStringAscii(TestCase):
|
||||||
|
def test_py_encode_basestring_ascii(self):
|
||||||
|
self._test_encode_basestring_ascii(simplejson.encoder.py_encode_basestring_ascii)
|
||||||
|
|
||||||
|
def test_c_encode_basestring_ascii(self):
|
||||||
|
if not simplejson.encoder.c_encode_basestring_ascii:
|
||||||
|
return
|
||||||
|
self._test_encode_basestring_ascii(simplejson.encoder.c_encode_basestring_ascii)
|
||||||
|
|
||||||
|
def _test_encode_basestring_ascii(self, encode_basestring_ascii):
|
||||||
|
fname = encode_basestring_ascii.__name__
|
||||||
|
for input_string, expect in CASES:
|
||||||
|
result = encode_basestring_ascii(input_string)
|
||||||
|
#self.assertEqual(result, expect,
|
||||||
|
# '{0!r} != {1!r} for {2}({3!r})'.format(
|
||||||
|
# result, expect, fname, input_string))
|
||||||
|
self.assertEqual(result, expect,
|
||||||
|
'%r != %r for %s(%r)' % (result, expect, fname, input_string))
|
||||||
|
|
||||||
|
def test_sorted_dict(self):
|
||||||
|
items = [('one', 1), ('two', 2), ('three', 3), ('four', 4), ('five', 5)]
|
||||||
|
s = simplejson.dumps(dict(items), sort_keys=True)
|
||||||
|
self.assertEqual(s, '{"five": 5, "four": 4, "one": 1, "three": 3, "two": 2}')
|
|
@ -0,0 +1,38 @@
|
||||||
|
import unittest
|
||||||
|
|
||||||
|
import simplejson as json
|
||||||
|
|
||||||
|
class TestEncodeForHTML(unittest.TestCase):
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
self.decoder = json.JSONDecoder()
|
||||||
|
self.encoder = json.JSONEncoderForHTML()
|
||||||
|
self.non_ascii_encoder = json.JSONEncoderForHTML(ensure_ascii=False)
|
||||||
|
|
||||||
|
def test_basic_encode(self):
|
||||||
|
self.assertEqual(r'"\u0026"', self.encoder.encode('&'))
|
||||||
|
self.assertEqual(r'"\u003c"', self.encoder.encode('<'))
|
||||||
|
self.assertEqual(r'"\u003e"', self.encoder.encode('>'))
|
||||||
|
self.assertEqual(r'"\u2028"', self.encoder.encode(u'\u2028'))
|
||||||
|
|
||||||
|
def test_non_ascii_basic_encode(self):
|
||||||
|
self.assertEqual(r'"\u0026"', self.non_ascii_encoder.encode('&'))
|
||||||
|
self.assertEqual(r'"\u003c"', self.non_ascii_encoder.encode('<'))
|
||||||
|
self.assertEqual(r'"\u003e"', self.non_ascii_encoder.encode('>'))
|
||||||
|
self.assertEqual(r'"\u2028"', self.non_ascii_encoder.encode(u'\u2028'))
|
||||||
|
|
||||||
|
def test_basic_roundtrip(self):
|
||||||
|
for char in '&<>':
|
||||||
|
self.assertEqual(
|
||||||
|
char, self.decoder.decode(
|
||||||
|
self.encoder.encode(char)))
|
||||||
|
|
||||||
|
def test_prevent_script_breakout(self):
|
||||||
|
bad_string = '</script><script>alert("gotcha")</script>'
|
||||||
|
self.assertEqual(
|
||||||
|
r'"\u003c/script\u003e\u003cscript\u003e'
|
||||||
|
r'alert(\"gotcha\")\u003c/script\u003e"',
|
||||||
|
self.encoder.encode(bad_string))
|
||||||
|
self.assertEqual(
|
||||||
|
bad_string, self.decoder.decode(
|
||||||
|
self.encoder.encode(bad_string)))
|
|
@ -0,0 +1,68 @@
|
||||||
|
import sys, pickle
|
||||||
|
from unittest import TestCase
|
||||||
|
|
||||||
|
import simplejson as json
|
||||||
|
from simplejson.compat import text_type, b
|
||||||
|
|
||||||
|
class TestErrors(TestCase):
|
||||||
|
def test_string_keys_error(self):
|
||||||
|
data = [{'a': 'A', 'b': (2, 4), 'c': 3.0, ('d',): 'D tuple'}]
|
||||||
|
try:
|
||||||
|
json.dumps(data)
|
||||||
|
except TypeError:
|
||||||
|
err = sys.exc_info()[1]
|
||||||
|
else:
|
||||||
|
self.fail('Expected TypeError')
|
||||||
|
self.assertEqual(str(err),
|
||||||
|
'keys must be str, int, float, bool or None, not tuple')
|
||||||
|
|
||||||
|
def test_not_serializable(self):
|
||||||
|
try:
|
||||||
|
json.dumps(json)
|
||||||
|
except TypeError:
|
||||||
|
err = sys.exc_info()[1]
|
||||||
|
else:
|
||||||
|
self.fail('Expected TypeError')
|
||||||
|
self.assertEqual(str(err),
|
||||||
|
'Object of type module is not JSON serializable')
|
||||||
|
|
||||||
|
def test_decode_error(self):
|
||||||
|
err = None
|
||||||
|
try:
|
||||||
|
json.loads('{}\na\nb')
|
||||||
|
except json.JSONDecodeError:
|
||||||
|
err = sys.exc_info()[1]
|
||||||
|
else:
|
||||||
|
self.fail('Expected JSONDecodeError')
|
||||||
|
self.assertEqual(err.lineno, 2)
|
||||||
|
self.assertEqual(err.colno, 1)
|
||||||
|
self.assertEqual(err.endlineno, 3)
|
||||||
|
self.assertEqual(err.endcolno, 2)
|
||||||
|
|
||||||
|
def test_scan_error(self):
|
||||||
|
err = None
|
||||||
|
for t in (text_type, b):
|
||||||
|
try:
|
||||||
|
json.loads(t('{"asdf": "'))
|
||||||
|
except json.JSONDecodeError:
|
||||||
|
err = sys.exc_info()[1]
|
||||||
|
else:
|
||||||
|
self.fail('Expected JSONDecodeError')
|
||||||
|
self.assertEqual(err.lineno, 1)
|
||||||
|
self.assertEqual(err.colno, 10)
|
||||||
|
|
||||||
|
def test_error_is_pickable(self):
|
||||||
|
err = None
|
||||||
|
try:
|
||||||
|
json.loads('{}\na\nb')
|
||||||
|
except json.JSONDecodeError:
|
||||||
|
err = sys.exc_info()[1]
|
||||||
|
else:
|
||||||
|
self.fail('Expected JSONDecodeError')
|
||||||
|
s = pickle.dumps(err)
|
||||||
|
e = pickle.loads(s)
|
||||||
|
|
||||||
|
self.assertEqual(err.msg, e.msg)
|
||||||
|
self.assertEqual(err.doc, e.doc)
|
||||||
|
self.assertEqual(err.pos, e.pos)
|
||||||
|
self.assertEqual(err.end, e.end)
|
|
@ -0,0 +1,176 @@
|
||||||
|
import sys
|
||||||
|
from unittest import TestCase
|
||||||
|
|
||||||
|
import simplejson as json
|
||||||
|
|
||||||
|
# 2007-10-05
|
||||||
|
JSONDOCS = [
|
||||||
|
# http://json.org/JSON_checker/test/fail1.json
|
||||||
|
'"A JSON payload should be an object or array, not a string."',
|
||||||
|
# http://json.org/JSON_checker/test/fail2.json
|
||||||
|
'["Unclosed array"',
|
||||||
|
# http://json.org/JSON_checker/test/fail3.json
|
||||||
|
'{unquoted_key: "keys must be quoted"}',
|
||||||
|
# http://json.org/JSON_checker/test/fail4.json
|
||||||
|
'["extra comma",]',
|
||||||
|
# http://json.org/JSON_checker/test/fail5.json
|
||||||
|
'["double extra comma",,]',
|
||||||
|
# http://json.org/JSON_checker/test/fail6.json
|
||||||
|
'[ , "<-- missing value"]',
|
||||||
|
# http://json.org/JSON_checker/test/fail7.json
|
||||||
|
'["Comma after the close"],',
|
||||||
|
# http://json.org/JSON_checker/test/fail8.json
|
||||||
|
'["Extra close"]]',
|
||||||
|
# http://json.org/JSON_checker/test/fail9.json
|
||||||
|
'{"Extra comma": true,}',
|
||||||
|
# http://json.org/JSON_checker/test/fail10.json
|
||||||
|
'{"Extra value after close": true} "misplaced quoted value"',
|
||||||
|
# http://json.org/JSON_checker/test/fail11.json
|
||||||
|
'{"Illegal expression": 1 + 2}',
|
||||||
|
# http://json.org/JSON_checker/test/fail12.json
|
||||||
|
'{"Illegal invocation": alert()}',
|
||||||
|
# http://json.org/JSON_checker/test/fail13.json
|
||||||
|
'{"Numbers cannot have leading zeroes": 013}',
|
||||||
|
# http://json.org/JSON_checker/test/fail14.json
|
||||||
|
'{"Numbers cannot be hex": 0x14}',
|
||||||
|
# http://json.org/JSON_checker/test/fail15.json
|
||||||
|
'["Illegal backslash escape: \\x15"]',
|
||||||
|
# http://json.org/JSON_checker/test/fail16.json
|
||||||
|
'[\\naked]',
|
||||||
|
# http://json.org/JSON_checker/test/fail17.json
|
||||||
|
'["Illegal backslash escape: \\017"]',
|
||||||
|
# http://json.org/JSON_checker/test/fail18.json
|
||||||
|
'[[[[[[[[[[[[[[[[[[[["Too deep"]]]]]]]]]]]]]]]]]]]]',
|
||||||
|
# http://json.org/JSON_checker/test/fail19.json
|
||||||
|
'{"Missing colon" null}',
|
||||||
|
# http://json.org/JSON_checker/test/fail20.json
|
||||||
|
'{"Double colon":: null}',
|
||||||
|
# http://json.org/JSON_checker/test/fail21.json
|
||||||
|
'{"Comma instead of colon", null}',
|
||||||
|
# http://json.org/JSON_checker/test/fail22.json
|
||||||
|
'["Colon instead of comma": false]',
|
||||||
|
# http://json.org/JSON_checker/test/fail23.json
|
||||||
|
'["Bad value", truth]',
|
||||||
|
# http://json.org/JSON_checker/test/fail24.json
|
||||||
|
"['single quote']",
|
||||||
|
# http://json.org/JSON_checker/test/fail25.json
|
||||||
|
'["\ttab\tcharacter\tin\tstring\t"]',
|
||||||
|
# http://json.org/JSON_checker/test/fail26.json
|
||||||
|
'["tab\\ character\\ in\\ string\\ "]',
|
||||||
|
# http://json.org/JSON_checker/test/fail27.json
|
||||||
|
'["line\nbreak"]',
|
||||||
|
# http://json.org/JSON_checker/test/fail28.json
|
||||||
|
'["line\\\nbreak"]',
|
||||||
|
# http://json.org/JSON_checker/test/fail29.json
|
||||||
|
'[0e]',
|
||||||
|
# http://json.org/JSON_checker/test/fail30.json
|
||||||
|
'[0e+]',
|
||||||
|
# http://json.org/JSON_checker/test/fail31.json
|
||||||
|
'[0e+-1]',
|
||||||
|
# http://json.org/JSON_checker/test/fail32.json
|
||||||
|
'{"Comma instead if closing brace": true,',
|
||||||
|
# http://json.org/JSON_checker/test/fail33.json
|
||||||
|
'["mismatch"}',
|
||||||
|
# http://code.google.com/p/simplejson/issues/detail?id=3
|
||||||
|
u'["A\u001FZ control characters in string"]',
|
||||||
|
# misc based on coverage
|
||||||
|
'{',
|
||||||
|
'{]',
|
||||||
|
'{"foo": "bar"]',
|
||||||
|
'{"foo": "bar"',
|
||||||
|
'nul',
|
||||||
|
'nulx',
|
||||||
|
'-',
|
||||||
|
'-x',
|
||||||
|
'-e',
|
||||||
|
'-e0',
|
||||||
|
'-Infinite',
|
||||||
|
'-Inf',
|
||||||
|
'Infinit',
|
||||||
|
'Infinite',
|
||||||
|
'NaM',
|
||||||
|
'NuN',
|
||||||
|
'falsy',
|
||||||
|
'fal',
|
||||||
|
'trug',
|
||||||
|
'tru',
|
||||||
|
'1e',
|
||||||
|
'1ex',
|
||||||
|
'1e-',
|
||||||
|
'1e-x',
|
||||||
|
]
|
||||||
|
|
||||||
|
SKIPS = {
|
||||||
|
1: "why not have a string payload?",
|
||||||
|
18: "spec doesn't specify any nesting limitations",
|
||||||
|
}
|
||||||
|
|
||||||
|
class TestFail(TestCase):
|
||||||
|
def test_failures(self):
|
||||||
|
for idx, doc in enumerate(JSONDOCS):
|
||||||
|
idx = idx + 1
|
||||||
|
if idx in SKIPS:
|
||||||
|
json.loads(doc)
|
||||||
|
continue
|
||||||
|
try:
|
||||||
|
json.loads(doc)
|
||||||
|
except json.JSONDecodeError:
|
||||||
|
pass
|
||||||
|
else:
|
||||||
|
self.fail("Expected failure for fail%d.json: %r" % (idx, doc))
|
||||||
|
|
||||||
|
def test_array_decoder_issue46(self):
|
||||||
|
# http://code.google.com/p/simplejson/issues/detail?id=46
|
||||||
|
for doc in [u'[,]', '[,]']:
|
||||||
|
try:
|
||||||
|
json.loads(doc)
|
||||||
|
except json.JSONDecodeError:
|
||||||
|
e = sys.exc_info()[1]
|
||||||
|
self.assertEqual(e.pos, 1)
|
||||||
|
self.assertEqual(e.lineno, 1)
|
||||||
|
self.assertEqual(e.colno, 2)
|
||||||
|
except Exception:
|
||||||
|
e = sys.exc_info()[1]
|
||||||
|
self.fail("Unexpected exception raised %r %s" % (e, e))
|
||||||
|
else:
|
||||||
|
self.fail("Unexpected success parsing '[,]'")
|
||||||
|
|
||||||
|
def test_truncated_input(self):
|
||||||
|
test_cases = [
|
||||||
|
('', 'Expecting value', 0),
|
||||||
|
('[', "Expecting value or ']'", 1),
|
||||||
|
('[42', "Expecting ',' delimiter", 3),
|
||||||
|
('[42,', 'Expecting value', 4),
|
||||||
|
('["', 'Unterminated string starting at', 1),
|
||||||
|
('["spam', 'Unterminated string starting at', 1),
|
||||||
|
('["spam"', "Expecting ',' delimiter", 7),
|
||||||
|
('["spam",', 'Expecting value', 8),
|
||||||
|
('{', 'Expecting property name enclosed in double quotes', 1),
|
||||||
|
('{"', 'Unterminated string starting at', 1),
|
||||||
|
('{"spam', 'Unterminated string starting at', 1),
|
||||||
|
('{"spam"', "Expecting ':' delimiter", 7),
|
||||||
|
('{"spam":', 'Expecting value', 8),
|
||||||
|
('{"spam":42', "Expecting ',' delimiter", 10),
|
||||||
|
('{"spam":42,', 'Expecting property name enclosed in double quotes',
|
||||||
|
11),
|
||||||
|
('"', 'Unterminated string starting at', 0),
|
||||||
|
('"spam', 'Unterminated string starting at', 0),
|
||||||
|
('[,', "Expecting value", 1),
|
||||||
|
]
|
||||||
|
for data, msg, idx in test_cases:
|
||||||
|
try:
|
||||||
|
json.loads(data)
|
||||||
|
except json.JSONDecodeError:
|
||||||
|
e = sys.exc_info()[1]
|
||||||
|
self.assertEqual(
|
||||||
|
e.msg[:len(msg)],
|
||||||
|
msg,
|
||||||
|
"%r doesn't start with %r for %r" % (e.msg, msg, data))
|
||||||
|
self.assertEqual(
|
||||||
|
e.pos, idx,
|
||||||
|
"pos %r != %r for %r" % (e.pos, idx, data))
|
||||||
|
except Exception:
|
||||||
|
e = sys.exc_info()[1]
|
||||||
|
self.fail("Unexpected exception raised %r %s" % (e, e))
|
||||||
|
else:
|
||||||
|
self.fail("Unexpected success parsing '%r'" % (data,))
|
|
@ -0,0 +1,35 @@
|
||||||
|
import math
|
||||||
|
from unittest import TestCase
|
||||||
|
from simplejson.compat import long_type, text_type
|
||||||
|
import simplejson as json
|
||||||
|
from simplejson.decoder import NaN, PosInf, NegInf
|
||||||
|
|
||||||
|
class TestFloat(TestCase):
|
||||||
|
def test_degenerates_allow(self):
|
||||||
|
for inf in (PosInf, NegInf):
|
||||||
|
self.assertEqual(json.loads(json.dumps(inf)), inf)
|
||||||
|
# Python 2.5 doesn't have math.isnan
|
||||||
|
nan = json.loads(json.dumps(NaN))
|
||||||
|
self.assertTrue((0 + nan) != nan)
|
||||||
|
|
||||||
|
def test_degenerates_ignore(self):
|
||||||
|
for f in (PosInf, NegInf, NaN):
|
||||||
|
self.assertEqual(json.loads(json.dumps(f, ignore_nan=True)), None)
|
||||||
|
|
||||||
|
def test_degenerates_deny(self):
|
||||||
|
for f in (PosInf, NegInf, NaN):
|
||||||
|
self.assertRaises(ValueError, json.dumps, f, allow_nan=False)
|
||||||
|
|
||||||
|
def test_floats(self):
|
||||||
|
for num in [1617161771.7650001, math.pi, math.pi**100,
|
||||||
|
math.pi**-100, 3.1]:
|
||||||
|
self.assertEqual(float(json.dumps(num)), num)
|
||||||
|
self.assertEqual(json.loads(json.dumps(num)), num)
|
||||||
|
self.assertEqual(json.loads(text_type(json.dumps(num))), num)
|
||||||
|
|
||||||
|
def test_ints(self):
|
||||||
|
for num in [1, long_type(1), 1<<32, 1<<64]:
|
||||||
|
self.assertEqual(json.dumps(num), str(num))
|
||||||
|
self.assertEqual(int(json.dumps(num)), num)
|
||||||
|
self.assertEqual(json.loads(json.dumps(num)), num)
|
||||||
|
self.assertEqual(json.loads(text_type(json.dumps(num))), num)
|
|
@ -0,0 +1,97 @@
|
||||||
|
import unittest
|
||||||
|
import simplejson as json
|
||||||
|
|
||||||
|
|
||||||
|
class ForJson(object):
|
||||||
|
def for_json(self):
|
||||||
|
return {'for_json': 1}
|
||||||
|
|
||||||
|
|
||||||
|
class NestedForJson(object):
|
||||||
|
def for_json(self):
|
||||||
|
return {'nested': ForJson()}
|
||||||
|
|
||||||
|
|
||||||
|
class ForJsonList(object):
|
||||||
|
def for_json(self):
|
||||||
|
return ['list']
|
||||||
|
|
||||||
|
|
||||||
|
class DictForJson(dict):
|
||||||
|
def for_json(self):
|
||||||
|
return {'alpha': 1}
|
||||||
|
|
||||||
|
|
||||||
|
class ListForJson(list):
|
||||||
|
def for_json(self):
|
||||||
|
return ['list']
|
||||||
|
|
||||||
|
|
||||||
|
class TestForJson(unittest.TestCase):
|
||||||
|
def assertRoundTrip(self, obj, other, for_json=True):
|
||||||
|
if for_json is None:
|
||||||
|
# None will use the default
|
||||||
|
s = json.dumps(obj)
|
||||||
|
else:
|
||||||
|
s = json.dumps(obj, for_json=for_json)
|
||||||
|
self.assertEqual(
|
||||||
|
json.loads(s),
|
||||||
|
other)
|
||||||
|
|
||||||
|
def test_for_json_encodes_stand_alone_object(self):
|
||||||
|
self.assertRoundTrip(
|
||||||
|
ForJson(),
|
||||||
|
ForJson().for_json())
|
||||||
|
|
||||||
|
def test_for_json_encodes_object_nested_in_dict(self):
|
||||||
|
self.assertRoundTrip(
|
||||||
|
{'hooray': ForJson()},
|
||||||
|
{'hooray': ForJson().for_json()})
|
||||||
|
|
||||||
|
def test_for_json_encodes_object_nested_in_list_within_dict(self):
|
||||||
|
self.assertRoundTrip(
|
||||||
|
{'list': [0, ForJson(), 2, 3]},
|
||||||
|
{'list': [0, ForJson().for_json(), 2, 3]})
|
||||||
|
|
||||||
|
def test_for_json_encodes_object_nested_within_object(self):
|
||||||
|
self.assertRoundTrip(
|
||||||
|
NestedForJson(),
|
||||||
|
{'nested': {'for_json': 1}})
|
||||||
|
|
||||||
|
def test_for_json_encodes_list(self):
|
||||||
|
self.assertRoundTrip(
|
||||||
|
ForJsonList(),
|
||||||
|
ForJsonList().for_json())
|
||||||
|
|
||||||
|
def test_for_json_encodes_list_within_object(self):
|
||||||
|
self.assertRoundTrip(
|
||||||
|
{'nested': ForJsonList()},
|
||||||
|
{'nested': ForJsonList().for_json()})
|
||||||
|
|
||||||
|
def test_for_json_encodes_dict_subclass(self):
|
||||||
|
self.assertRoundTrip(
|
||||||
|
DictForJson(a=1),
|
||||||
|
DictForJson(a=1).for_json())
|
||||||
|
|
||||||
|
def test_for_json_encodes_list_subclass(self):
|
||||||
|
self.assertRoundTrip(
|
||||||
|
ListForJson(['l']),
|
||||||
|
ListForJson(['l']).for_json())
|
||||||
|
|
||||||
|
def test_for_json_ignored_if_not_true_with_dict_subclass(self):
|
||||||
|
for for_json in (None, False):
|
||||||
|
self.assertRoundTrip(
|
||||||
|
DictForJson(a=1),
|
||||||
|
{'a': 1},
|
||||||
|
for_json=for_json)
|
||||||
|
|
||||||
|
def test_for_json_ignored_if_not_true_with_list_subclass(self):
|
||||||
|
for for_json in (None, False):
|
||||||
|
self.assertRoundTrip(
|
||||||
|
ListForJson(['l']),
|
||||||
|
['l'],
|
||||||
|
for_json=for_json)
|
||||||
|
|
||||||
|
def test_raises_typeerror_if_for_json_not_true_with_object(self):
|
||||||
|
self.assertRaises(TypeError, json.dumps, ForJson())
|
||||||
|
self.assertRaises(TypeError, json.dumps, ForJson(), for_json=False)
|
|
@ -0,0 +1,86 @@
|
||||||
|
from unittest import TestCase
|
||||||
|
import textwrap
|
||||||
|
|
||||||
|
import simplejson as json
|
||||||
|
from simplejson.compat import StringIO
|
||||||
|
|
||||||
|
class TestIndent(TestCase):
|
||||||
|
def test_indent(self):
|
||||||
|
h = [['blorpie'], ['whoops'], [], 'd-shtaeou', 'd-nthiouh',
|
||||||
|
'i-vhbjkhnth',
|
||||||
|
{'nifty': 87}, {'field': 'yes', 'morefield': False} ]
|
||||||
|
|
||||||
|
expect = textwrap.dedent("""\
|
||||||
|
[
|
||||||
|
\t[
|
||||||
|
\t\t"blorpie"
|
||||||
|
\t],
|
||||||
|
\t[
|
||||||
|
\t\t"whoops"
|
||||||
|
\t],
|
||||||
|
\t[],
|
||||||
|
\t"d-shtaeou",
|
||||||
|
\t"d-nthiouh",
|
||||||
|
\t"i-vhbjkhnth",
|
||||||
|
\t{
|
||||||
|
\t\t"nifty": 87
|
||||||
|
\t},
|
||||||
|
\t{
|
||||||
|
\t\t"field": "yes",
|
||||||
|
\t\t"morefield": false
|
||||||
|
\t}
|
||||||
|
]""")
|
||||||
|
|
||||||
|
|
||||||
|
d1 = json.dumps(h)
|
||||||
|
d2 = json.dumps(h, indent='\t', sort_keys=True, separators=(',', ': '))
|
||||||
|
d3 = json.dumps(h, indent=' ', sort_keys=True, separators=(',', ': '))
|
||||||
|
d4 = json.dumps(h, indent=2, sort_keys=True, separators=(',', ': '))
|
||||||
|
|
||||||
|
h1 = json.loads(d1)
|
||||||
|
h2 = json.loads(d2)
|
||||||
|
h3 = json.loads(d3)
|
||||||
|
h4 = json.loads(d4)
|
||||||
|
|
||||||
|
self.assertEqual(h1, h)
|
||||||
|
self.assertEqual(h2, h)
|
||||||
|
self.assertEqual(h3, h)
|
||||||
|
self.assertEqual(h4, h)
|
||||||
|
self.assertEqual(d3, expect.replace('\t', ' '))
|
||||||
|
self.assertEqual(d4, expect.replace('\t', ' '))
|
||||||
|
# NOTE: Python 2.4 textwrap.dedent converts tabs to spaces,
|
||||||
|
# so the following is expected to fail. Python 2.4 is not a
|
||||||
|
# supported platform in simplejson 2.1.0+.
|
||||||
|
self.assertEqual(d2, expect)
|
||||||
|
|
||||||
|
def test_indent0(self):
|
||||||
|
h = {3: 1}
|
||||||
|
def check(indent, expected):
|
||||||
|
d1 = json.dumps(h, indent=indent)
|
||||||
|
self.assertEqual(d1, expected)
|
||||||
|
|
||||||
|
sio = StringIO()
|
||||||
|
json.dump(h, sio, indent=indent)
|
||||||
|
self.assertEqual(sio.getvalue(), expected)
|
||||||
|
|
||||||
|
# indent=0 should emit newlines
|
||||||
|
check(0, '{\n"3": 1\n}')
|
||||||
|
# indent=None is more compact
|
||||||
|
check(None, '{"3": 1}')
|
||||||
|
|
||||||
|
def test_separators(self):
|
||||||
|
lst = [1,2,3,4]
|
||||||
|
expect = '[\n1,\n2,\n3,\n4\n]'
|
||||||
|
expect_spaces = '[\n1, \n2, \n3, \n4\n]'
|
||||||
|
# Ensure that separators still works
|
||||||
|
self.assertEqual(
|
||||||
|
expect_spaces,
|
||||||
|
json.dumps(lst, indent=0, separators=(', ', ': ')))
|
||||||
|
# Force the new defaults
|
||||||
|
self.assertEqual(
|
||||||
|
expect,
|
||||||
|
json.dumps(lst, indent=0, separators=(',', ': ')))
|
||||||
|
# Added in 2.1.4
|
||||||
|
self.assertEqual(
|
||||||
|
expect,
|
||||||
|
json.dumps(lst, indent=0))
|
|
@ -0,0 +1,27 @@
|
||||||
|
from unittest import TestCase
|
||||||
|
|
||||||
|
import simplejson as json
|
||||||
|
from operator import itemgetter
|
||||||
|
|
||||||
|
class TestItemSortKey(TestCase):
|
||||||
|
def test_simple_first(self):
|
||||||
|
a = {'a': 1, 'c': 5, 'jack': 'jill', 'pick': 'axe', 'array': [1, 5, 6, 9], 'tuple': (83, 12, 3), 'crate': 'dog', 'zeak': 'oh'}
|
||||||
|
self.assertEqual(
|
||||||
|
'{"a": 1, "c": 5, "crate": "dog", "jack": "jill", "pick": "axe", "zeak": "oh", "array": [1, 5, 6, 9], "tuple": [83, 12, 3]}',
|
||||||
|
json.dumps(a, item_sort_key=json.simple_first))
|
||||||
|
|
||||||
|
def test_case(self):
|
||||||
|
a = {'a': 1, 'c': 5, 'Jack': 'jill', 'pick': 'axe', 'Array': [1, 5, 6, 9], 'tuple': (83, 12, 3), 'crate': 'dog', 'zeak': 'oh'}
|
||||||
|
self.assertEqual(
|
||||||
|
'{"Array": [1, 5, 6, 9], "Jack": "jill", "a": 1, "c": 5, "crate": "dog", "pick": "axe", "tuple": [83, 12, 3], "zeak": "oh"}',
|
||||||
|
json.dumps(a, item_sort_key=itemgetter(0)))
|
||||||
|
self.assertEqual(
|
||||||
|
'{"a": 1, "Array": [1, 5, 6, 9], "c": 5, "crate": "dog", "Jack": "jill", "pick": "axe", "tuple": [83, 12, 3], "zeak": "oh"}',
|
||||||
|
json.dumps(a, item_sort_key=lambda kv: kv[0].lower()))
|
||||||
|
|
||||||
|
def test_item_sort_key_value(self):
|
||||||
|
# https://github.com/simplejson/simplejson/issues/173
|
||||||
|
a = {'a': 1, 'b': 0}
|
||||||
|
self.assertEqual(
|
||||||
|
'{"b": 0, "a": 1}',
|
||||||
|
json.dumps(a, item_sort_key=lambda kv: kv[1]))
|
|
@ -0,0 +1,31 @@
|
||||||
|
import unittest
|
||||||
|
from simplejson.compat import StringIO
|
||||||
|
|
||||||
|
import simplejson as json
|
||||||
|
|
||||||
|
def iter_dumps(obj, **kw):
|
||||||
|
return ''.join(json.JSONEncoder(**kw).iterencode(obj))
|
||||||
|
|
||||||
|
def sio_dump(obj, **kw):
|
||||||
|
sio = StringIO()
|
||||||
|
json.dumps(obj, **kw)
|
||||||
|
return sio.getvalue()
|
||||||
|
|
||||||
|
class TestIterable(unittest.TestCase):
|
||||||
|
def test_iterable(self):
|
||||||
|
for l in ([], [1], [1, 2], [1, 2, 3]):
|
||||||
|
for opts in [{}, {'indent': 2}]:
|
||||||
|
for dumps in (json.dumps, iter_dumps, sio_dump):
|
||||||
|
expect = dumps(l, **opts)
|
||||||
|
default_expect = dumps(sum(l), **opts)
|
||||||
|
# Default is False
|
||||||
|
self.assertRaises(TypeError, dumps, iter(l), **opts)
|
||||||
|
self.assertRaises(TypeError, dumps, iter(l), iterable_as_array=False, **opts)
|
||||||
|
self.assertEqual(expect, dumps(iter(l), iterable_as_array=True, **opts))
|
||||||
|
# Ensure that the "default" gets called
|
||||||
|
self.assertEqual(default_expect, dumps(iter(l), default=sum, **opts))
|
||||||
|
self.assertEqual(default_expect, dumps(iter(l), iterable_as_array=False, default=sum, **opts))
|
||||||
|
# Ensure that the "default" does not get called
|
||||||
|
self.assertEqual(
|
||||||
|
expect,
|
||||||
|
dumps(iter(l), iterable_as_array=True, default=sum, **opts))
|
|
@ -0,0 +1,122 @@
|
||||||
|
from __future__ import absolute_import
|
||||||
|
import unittest
|
||||||
|
import simplejson as json
|
||||||
|
from simplejson.compat import StringIO
|
||||||
|
|
||||||
|
try:
|
||||||
|
from collections import namedtuple
|
||||||
|
except ImportError:
|
||||||
|
class Value(tuple):
|
||||||
|
def __new__(cls, *args):
|
||||||
|
return tuple.__new__(cls, args)
|
||||||
|
|
||||||
|
def _asdict(self):
|
||||||
|
return {'value': self[0]}
|
||||||
|
class Point(tuple):
|
||||||
|
def __new__(cls, *args):
|
||||||
|
return tuple.__new__(cls, args)
|
||||||
|
|
||||||
|
def _asdict(self):
|
||||||
|
return {'x': self[0], 'y': self[1]}
|
||||||
|
else:
|
||||||
|
Value = namedtuple('Value', ['value'])
|
||||||
|
Point = namedtuple('Point', ['x', 'y'])
|
||||||
|
|
||||||
|
class DuckValue(object):
|
||||||
|
def __init__(self, *args):
|
||||||
|
self.value = Value(*args)
|
||||||
|
|
||||||
|
def _asdict(self):
|
||||||
|
return self.value._asdict()
|
||||||
|
|
||||||
|
class DuckPoint(object):
|
||||||
|
def __init__(self, *args):
|
||||||
|
self.point = Point(*args)
|
||||||
|
|
||||||
|
def _asdict(self):
|
||||||
|
return self.point._asdict()
|
||||||
|
|
||||||
|
class DeadDuck(object):
|
||||||
|
_asdict = None
|
||||||
|
|
||||||
|
class DeadDict(dict):
|
||||||
|
_asdict = None
|
||||||
|
|
||||||
|
CONSTRUCTORS = [
|
||||||
|
lambda v: v,
|
||||||
|
lambda v: [v],
|
||||||
|
lambda v: [{'key': v}],
|
||||||
|
]
|
||||||
|
|
||||||
|
class TestNamedTuple(unittest.TestCase):
|
||||||
|
def test_namedtuple_dumps(self):
|
||||||
|
for v in [Value(1), Point(1, 2), DuckValue(1), DuckPoint(1, 2)]:
|
||||||
|
d = v._asdict()
|
||||||
|
self.assertEqual(d, json.loads(json.dumps(v)))
|
||||||
|
self.assertEqual(
|
||||||
|
d,
|
||||||
|
json.loads(json.dumps(v, namedtuple_as_object=True)))
|
||||||
|
self.assertEqual(d, json.loads(json.dumps(v, tuple_as_array=False)))
|
||||||
|
self.assertEqual(
|
||||||
|
d,
|
||||||
|
json.loads(json.dumps(v, namedtuple_as_object=True,
|
||||||
|
tuple_as_array=False)))
|
||||||
|
|
||||||
|
def test_namedtuple_dumps_false(self):
|
||||||
|
for v in [Value(1), Point(1, 2)]:
|
||||||
|
l = list(v)
|
||||||
|
self.assertEqual(
|
||||||
|
l,
|
||||||
|
json.loads(json.dumps(v, namedtuple_as_object=False)))
|
||||||
|
self.assertRaises(TypeError, json.dumps, v,
|
||||||
|
tuple_as_array=False, namedtuple_as_object=False)
|
||||||
|
|
||||||
|
def test_namedtuple_dump(self):
|
||||||
|
for v in [Value(1), Point(1, 2), DuckValue(1), DuckPoint(1, 2)]:
|
||||||
|
d = v._asdict()
|
||||||
|
sio = StringIO()
|
||||||
|
json.dump(v, sio)
|
||||||
|
self.assertEqual(d, json.loads(sio.getvalue()))
|
||||||
|
sio = StringIO()
|
||||||
|
json.dump(v, sio, namedtuple_as_object=True)
|
||||||
|
self.assertEqual(
|
||||||
|
d,
|
||||||
|
json.loads(sio.getvalue()))
|
||||||
|
sio = StringIO()
|
||||||
|
json.dump(v, sio, tuple_as_array=False)
|
||||||
|
self.assertEqual(d, json.loads(sio.getvalue()))
|
||||||
|
sio = StringIO()
|
||||||
|
json.dump(v, sio, namedtuple_as_object=True,
|
||||||
|
tuple_as_array=False)
|
||||||
|
self.assertEqual(
|
||||||
|
d,
|
||||||
|
json.loads(sio.getvalue()))
|
||||||
|
|
||||||
|
def test_namedtuple_dump_false(self):
|
||||||
|
for v in [Value(1), Point(1, 2)]:
|
||||||
|
l = list(v)
|
||||||
|
sio = StringIO()
|
||||||
|
json.dump(v, sio, namedtuple_as_object=False)
|
||||||
|
self.assertEqual(
|
||||||
|
l,
|
||||||
|
json.loads(sio.getvalue()))
|
||||||
|
self.assertRaises(TypeError, json.dump, v, StringIO(),
|
||||||
|
tuple_as_array=False, namedtuple_as_object=False)
|
||||||
|
|
||||||
|
def test_asdict_not_callable_dump(self):
|
||||||
|
for f in CONSTRUCTORS:
|
||||||
|
self.assertRaises(TypeError,
|
||||||
|
json.dump, f(DeadDuck()), StringIO(), namedtuple_as_object=True)
|
||||||
|
sio = StringIO()
|
||||||
|
json.dump(f(DeadDict()), sio, namedtuple_as_object=True)
|
||||||
|
self.assertEqual(
|
||||||
|
json.dumps(f({})),
|
||||||
|
sio.getvalue())
|
||||||
|
|
||||||
|
def test_asdict_not_callable_dumps(self):
|
||||||
|
for f in CONSTRUCTORS:
|
||||||
|
self.assertRaises(TypeError,
|
||||||
|
json.dumps, f(DeadDuck()), namedtuple_as_object=True)
|
||||||
|
self.assertEqual(
|
||||||
|
json.dumps(f({})),
|
||||||
|
json.dumps(f(DeadDict()), namedtuple_as_object=True))
|
|
@ -0,0 +1,71 @@
|
||||||
|
from unittest import TestCase
|
||||||
|
|
||||||
|
import simplejson as json
|
||||||
|
|
||||||
|
# from http://json.org/JSON_checker/test/pass1.json
|
||||||
|
JSON = r'''
|
||||||
|
[
|
||||||
|
"JSON Test Pattern pass1",
|
||||||
|
{"object with 1 member":["array with 1 element"]},
|
||||||
|
{},
|
||||||
|
[],
|
||||||
|
-42,
|
||||||
|
true,
|
||||||
|
false,
|
||||||
|
null,
|
||||||
|
{
|
||||||
|
"integer": 1234567890,
|
||||||
|
"real": -9876.543210,
|
||||||
|
"e": 0.123456789e-12,
|
||||||
|
"E": 1.234567890E+34,
|
||||||
|
"": 23456789012E66,
|
||||||
|
"zero": 0,
|
||||||
|
"one": 1,
|
||||||
|
"space": " ",
|
||||||
|
"quote": "\"",
|
||||||
|
"backslash": "\\",
|
||||||
|
"controls": "\b\f\n\r\t",
|
||||||
|
"slash": "/ & \/",
|
||||||
|
"alpha": "abcdefghijklmnopqrstuvwyz",
|
||||||
|
"ALPHA": "ABCDEFGHIJKLMNOPQRSTUVWYZ",
|
||||||
|
"digit": "0123456789",
|
||||||
|
"special": "`1~!@#$%^&*()_+-={':[,]}|;.</>?",
|
||||||
|
"hex": "\u0123\u4567\u89AB\uCDEF\uabcd\uef4A",
|
||||||
|
"true": true,
|
||||||
|
"false": false,
|
||||||
|
"null": null,
|
||||||
|
"array":[ ],
|
||||||
|
"object":{ },
|
||||||
|
"address": "50 St. James Street",
|
||||||
|
"url": "http://www.JSON.org/",
|
||||||
|
"comment": "// /* <!-- --",
|
||||||
|
"# -- --> */": " ",
|
||||||
|
" s p a c e d " :[1,2 , 3
|
||||||
|
|
||||||
|
,
|
||||||
|
|
||||||
|
4 , 5 , 6 ,7 ],"compact": [1,2,3,4,5,6,7],
|
||||||
|
"jsontext": "{\"object with 1 member\":[\"array with 1 element\"]}",
|
||||||
|
"quotes": "" \u0022 %22 0x22 034 "",
|
||||||
|
"\/\\\"\uCAFE\uBABE\uAB98\uFCDE\ubcda\uef4A\b\f\n\r\t`1~!@#$%^&*()_+-=[]{}|;:',./<>?"
|
||||||
|
: "A key can be any string"
|
||||||
|
},
|
||||||
|
0.5 ,98.6
|
||||||
|
,
|
||||||
|
99.44
|
||||||
|
,
|
||||||
|
|
||||||
|
1066,
|
||||||
|
1e1,
|
||||||
|
0.1e1,
|
||||||
|
1e-1,
|
||||||
|
1e00,2e+00,2e-00
|
||||||
|
,"rosebud"]
|
||||||
|
'''
|
||||||
|
|
||||||
|
class TestPass1(TestCase):
|
||||||
|
def test_parse(self):
|
||||||
|
# test in/out equivalence and parsing
|
||||||
|
res = json.loads(JSON)
|
||||||
|
out = json.dumps(res)
|
||||||
|
self.assertEqual(res, json.loads(out))
|
|
@ -0,0 +1,14 @@
|
||||||
|
from unittest import TestCase
|
||||||
|
import simplejson as json
|
||||||
|
|
||||||
|
# from http://json.org/JSON_checker/test/pass2.json
|
||||||
|
JSON = r'''
|
||||||
|
[[[[[[[[[[[[[[[[[[["Not too deep"]]]]]]]]]]]]]]]]]]]
|
||||||
|
'''
|
||||||
|
|
||||||
|
class TestPass2(TestCase):
|
||||||
|
def test_parse(self):
|
||||||
|
# test in/out equivalence and parsing
|
||||||
|
res = json.loads(JSON)
|
||||||
|
out = json.dumps(res)
|
||||||
|
self.assertEqual(res, json.loads(out))
|
|
@ -0,0 +1,20 @@
|
||||||
|
from unittest import TestCase
|
||||||
|
|
||||||
|
import simplejson as json
|
||||||
|
|
||||||
|
# from http://json.org/JSON_checker/test/pass3.json
|
||||||
|
JSON = r'''
|
||||||
|
{
|
||||||
|
"JSON Test Pattern pass3": {
|
||||||
|
"The outermost value": "must be an object or array.",
|
||||||
|
"In this test": "It is an object."
|
||||||
|
}
|
||||||
|
}
|
||||||
|
'''
|
||||||
|
|
||||||
|
class TestPass3(TestCase):
|
||||||
|
def test_parse(self):
|
||||||
|
# test in/out equivalence and parsing
|
||||||
|
res = json.loads(JSON)
|
||||||
|
out = json.dumps(res)
|
||||||
|
self.assertEqual(res, json.loads(out))
|
|
@ -0,0 +1,47 @@
|
||||||
|
import unittest
|
||||||
|
import simplejson as json
|
||||||
|
|
||||||
|
dct1 = {
|
||||||
|
'key1': 'value1'
|
||||||
|
}
|
||||||
|
|
||||||
|
dct2 = {
|
||||||
|
'key2': 'value2',
|
||||||
|
'd1': dct1
|
||||||
|
}
|
||||||
|
|
||||||
|
dct3 = {
|
||||||
|
'key2': 'value2',
|
||||||
|
'd1': json.dumps(dct1)
|
||||||
|
}
|
||||||
|
|
||||||
|
dct4 = {
|
||||||
|
'key2': 'value2',
|
||||||
|
'd1': json.RawJSON(json.dumps(dct1))
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
class TestRawJson(unittest.TestCase):
|
||||||
|
|
||||||
|
def test_normal_str(self):
|
||||||
|
self.assertNotEqual(json.dumps(dct2), json.dumps(dct3))
|
||||||
|
|
||||||
|
def test_raw_json_str(self):
|
||||||
|
self.assertEqual(json.dumps(dct2), json.dumps(dct4))
|
||||||
|
self.assertEqual(dct2, json.loads(json.dumps(dct4)))
|
||||||
|
|
||||||
|
def test_list(self):
|
||||||
|
self.assertEqual(
|
||||||
|
json.dumps([dct2]),
|
||||||
|
json.dumps([json.RawJSON(json.dumps(dct2))]))
|
||||||
|
self.assertEqual(
|
||||||
|
[dct2],
|
||||||
|
json.loads(json.dumps([json.RawJSON(json.dumps(dct2))])))
|
||||||
|
|
||||||
|
def test_direct(self):
|
||||||
|
self.assertEqual(
|
||||||
|
json.dumps(dct2),
|
||||||
|
json.dumps(json.RawJSON(json.dumps(dct2))))
|
||||||
|
self.assertEqual(
|
||||||
|
dct2,
|
||||||
|
json.loads(json.dumps(json.RawJSON(json.dumps(dct2)))))
|
|
@ -0,0 +1,67 @@
|
||||||
|
from unittest import TestCase
|
||||||
|
|
||||||
|
import simplejson as json
|
||||||
|
|
||||||
|
class JSONTestObject:
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class RecursiveJSONEncoder(json.JSONEncoder):
|
||||||
|
recurse = False
|
||||||
|
def default(self, o):
|
||||||
|
if o is JSONTestObject:
|
||||||
|
if self.recurse:
|
||||||
|
return [JSONTestObject]
|
||||||
|
else:
|
||||||
|
return 'JSONTestObject'
|
||||||
|
return json.JSONEncoder.default(o)
|
||||||
|
|
||||||
|
|
||||||
|
class TestRecursion(TestCase):
|
||||||
|
def test_listrecursion(self):
|
||||||
|
x = []
|
||||||
|
x.append(x)
|
||||||
|
try:
|
||||||
|
json.dumps(x)
|
||||||
|
except ValueError:
|
||||||
|
pass
|
||||||
|
else:
|
||||||
|
self.fail("didn't raise ValueError on list recursion")
|
||||||
|
x = []
|
||||||
|
y = [x]
|
||||||
|
x.append(y)
|
||||||
|
try:
|
||||||
|
json.dumps(x)
|
||||||
|
except ValueError:
|
||||||
|
pass
|
||||||
|
else:
|
||||||
|
self.fail("didn't raise ValueError on alternating list recursion")
|
||||||
|
y = []
|
||||||
|
x = [y, y]
|
||||||
|
# ensure that the marker is cleared
|
||||||
|
json.dumps(x)
|
||||||
|
|
||||||
|
def test_dictrecursion(self):
|
||||||
|
x = {}
|
||||||
|
x["test"] = x
|
||||||
|
try:
|
||||||
|
json.dumps(x)
|
||||||
|
except ValueError:
|
||||||
|
pass
|
||||||
|
else:
|
||||||
|
self.fail("didn't raise ValueError on dict recursion")
|
||||||
|
x = {}
|
||||||
|
y = {"a": x, "b": x}
|
||||||
|
# ensure that the marker is cleared
|
||||||
|
json.dumps(y)
|
||||||
|
|
||||||
|
def test_defaultrecursion(self):
|
||||||
|
enc = RecursiveJSONEncoder()
|
||||||
|
self.assertEqual(enc.encode(JSONTestObject), '"JSONTestObject"')
|
||||||
|
enc.recurse = True
|
||||||
|
try:
|
||||||
|
enc.encode(JSONTestObject)
|
||||||
|
except ValueError:
|
||||||
|
pass
|
||||||
|
else:
|
||||||
|
self.fail("didn't raise ValueError on default recursion")
|
|
@ -0,0 +1,196 @@
|
||||||
|
import sys
|
||||||
|
from unittest import TestCase
|
||||||
|
|
||||||
|
import simplejson as json
|
||||||
|
import simplejson.decoder
|
||||||
|
from simplejson.compat import b, PY3
|
||||||
|
|
||||||
|
class TestScanString(TestCase):
|
||||||
|
# The bytes type is intentionally not used in most of these tests
|
||||||
|
# under Python 3 because the decoder immediately coerces to str before
|
||||||
|
# calling scanstring. In Python 2 we are testing the code paths
|
||||||
|
# for both unicode and str.
|
||||||
|
#
|
||||||
|
# The reason this is done is because Python 3 would require
|
||||||
|
# entirely different code paths for parsing bytes and str.
|
||||||
|
#
|
||||||
|
def test_py_scanstring(self):
|
||||||
|
self._test_scanstring(simplejson.decoder.py_scanstring)
|
||||||
|
|
||||||
|
def test_c_scanstring(self):
|
||||||
|
if not simplejson.decoder.c_scanstring:
|
||||||
|
return
|
||||||
|
self._test_scanstring(simplejson.decoder.c_scanstring)
|
||||||
|
|
||||||
|
self.assertTrue(isinstance(simplejson.decoder.c_scanstring('""', 0)[0], str))
|
||||||
|
|
||||||
|
def _test_scanstring(self, scanstring):
|
||||||
|
if sys.maxunicode == 65535:
|
||||||
|
self.assertEqual(
|
||||||
|
scanstring(u'"z\U0001d120x"', 1, None, True),
|
||||||
|
(u'z\U0001d120x', 6))
|
||||||
|
else:
|
||||||
|
self.assertEqual(
|
||||||
|
scanstring(u'"z\U0001d120x"', 1, None, True),
|
||||||
|
(u'z\U0001d120x', 5))
|
||||||
|
|
||||||
|
self.assertEqual(
|
||||||
|
scanstring('"\\u007b"', 1, None, True),
|
||||||
|
(u'{', 8))
|
||||||
|
|
||||||
|
self.assertEqual(
|
||||||
|
scanstring('"A JSON payload should be an object or array, not a string."', 1, None, True),
|
||||||
|
(u'A JSON payload should be an object or array, not a string.', 60))
|
||||||
|
|
||||||
|
self.assertEqual(
|
||||||
|
scanstring('["Unclosed array"', 2, None, True),
|
||||||
|
(u'Unclosed array', 17))
|
||||||
|
|
||||||
|
self.assertEqual(
|
||||||
|
scanstring('["extra comma",]', 2, None, True),
|
||||||
|
(u'extra comma', 14))
|
||||||
|
|
||||||
|
self.assertEqual(
|
||||||
|
scanstring('["double extra comma",,]', 2, None, True),
|
||||||
|
(u'double extra comma', 21))
|
||||||
|
|
||||||
|
self.assertEqual(
|
||||||
|
scanstring('["Comma after the close"],', 2, None, True),
|
||||||
|
(u'Comma after the close', 24))
|
||||||
|
|
||||||
|
self.assertEqual(
|
||||||
|
scanstring('["Extra close"]]', 2, None, True),
|
||||||
|
(u'Extra close', 14))
|
||||||
|
|
||||||
|
self.assertEqual(
|
||||||
|
scanstring('{"Extra comma": true,}', 2, None, True),
|
||||||
|
(u'Extra comma', 14))
|
||||||
|
|
||||||
|
self.assertEqual(
|
||||||
|
scanstring('{"Extra value after close": true} "misplaced quoted value"', 2, None, True),
|
||||||
|
(u'Extra value after close', 26))
|
||||||
|
|
||||||
|
self.assertEqual(
|
||||||
|
scanstring('{"Illegal expression": 1 + 2}', 2, None, True),
|
||||||
|
(u'Illegal expression', 21))
|
||||||
|
|
||||||
|
self.assertEqual(
|
||||||
|
scanstring('{"Illegal invocation": alert()}', 2, None, True),
|
||||||
|
(u'Illegal invocation', 21))
|
||||||
|
|
||||||
|
self.assertEqual(
|
||||||
|
scanstring('{"Numbers cannot have leading zeroes": 013}', 2, None, True),
|
||||||
|
(u'Numbers cannot have leading zeroes', 37))
|
||||||
|
|
||||||
|
self.assertEqual(
|
||||||
|
scanstring('{"Numbers cannot be hex": 0x14}', 2, None, True),
|
||||||
|
(u'Numbers cannot be hex', 24))
|
||||||
|
|
||||||
|
self.assertEqual(
|
||||||
|
scanstring('[[[[[[[[[[[[[[[[[[[["Too deep"]]]]]]]]]]]]]]]]]]]]', 21, None, True),
|
||||||
|
(u'Too deep', 30))
|
||||||
|
|
||||||
|
self.assertEqual(
|
||||||
|
scanstring('{"Missing colon" null}', 2, None, True),
|
||||||
|
(u'Missing colon', 16))
|
||||||
|
|
||||||
|
self.assertEqual(
|
||||||
|
scanstring('{"Double colon":: null}', 2, None, True),
|
||||||
|
(u'Double colon', 15))
|
||||||
|
|
||||||
|
self.assertEqual(
|
||||||
|
scanstring('{"Comma instead of colon", null}', 2, None, True),
|
||||||
|
(u'Comma instead of colon', 25))
|
||||||
|
|
||||||
|
self.assertEqual(
|
||||||
|
scanstring('["Colon instead of comma": false]', 2, None, True),
|
||||||
|
(u'Colon instead of comma', 25))
|
||||||
|
|
||||||
|
self.assertEqual(
|
||||||
|
scanstring('["Bad value", truth]', 2, None, True),
|
||||||
|
(u'Bad value', 12))
|
||||||
|
|
||||||
|
for c in map(chr, range(0x00, 0x1f)):
|
||||||
|
self.assertEqual(
|
||||||
|
scanstring(c + '"', 0, None, False),
|
||||||
|
(c, 2))
|
||||||
|
self.assertRaises(
|
||||||
|
ValueError,
|
||||||
|
scanstring, c + '"', 0, None, True)
|
||||||
|
|
||||||
|
self.assertRaises(ValueError, scanstring, '', 0, None, True)
|
||||||
|
self.assertRaises(ValueError, scanstring, 'a', 0, None, True)
|
||||||
|
self.assertRaises(ValueError, scanstring, '\\', 0, None, True)
|
||||||
|
self.assertRaises(ValueError, scanstring, '\\u', 0, None, True)
|
||||||
|
self.assertRaises(ValueError, scanstring, '\\u0', 0, None, True)
|
||||||
|
self.assertRaises(ValueError, scanstring, '\\u01', 0, None, True)
|
||||||
|
self.assertRaises(ValueError, scanstring, '\\u012', 0, None, True)
|
||||||
|
self.assertRaises(ValueError, scanstring, '\\u0123', 0, None, True)
|
||||||
|
if sys.maxunicode > 65535:
|
||||||
|
self.assertRaises(ValueError,
|
||||||
|
scanstring, '\\ud834\\u"', 0, None, True)
|
||||||
|
self.assertRaises(ValueError,
|
||||||
|
scanstring, '\\ud834\\x0123"', 0, None, True)
|
||||||
|
|
||||||
|
def test_issue3623(self):
|
||||||
|
self.assertRaises(ValueError, json.decoder.scanstring, "xxx", 1,
|
||||||
|
"xxx")
|
||||||
|
self.assertRaises(UnicodeDecodeError,
|
||||||
|
json.encoder.encode_basestring_ascii, b("xx\xff"))
|
||||||
|
|
||||||
|
def test_overflow(self):
|
||||||
|
# Python 2.5 does not have maxsize, Python 3 does not have maxint
|
||||||
|
maxsize = getattr(sys, 'maxsize', getattr(sys, 'maxint', None))
|
||||||
|
assert maxsize is not None
|
||||||
|
self.assertRaises(OverflowError, json.decoder.scanstring, "xxx",
|
||||||
|
maxsize + 1)
|
||||||
|
|
||||||
|
def test_surrogates(self):
|
||||||
|
scanstring = json.decoder.scanstring
|
||||||
|
|
||||||
|
def assertScan(given, expect, test_utf8=True):
|
||||||
|
givens = [given]
|
||||||
|
if not PY3 and test_utf8:
|
||||||
|
givens.append(given.encode('utf8'))
|
||||||
|
for given in givens:
|
||||||
|
(res, count) = scanstring(given, 1, None, True)
|
||||||
|
self.assertEqual(len(given), count)
|
||||||
|
self.assertEqual(res, expect)
|
||||||
|
|
||||||
|
assertScan(
|
||||||
|
u'"z\\ud834\\u0079x"',
|
||||||
|
u'z\ud834yx')
|
||||||
|
assertScan(
|
||||||
|
u'"z\\ud834\\udd20x"',
|
||||||
|
u'z\U0001d120x')
|
||||||
|
assertScan(
|
||||||
|
u'"z\\ud834\\ud834\\udd20x"',
|
||||||
|
u'z\ud834\U0001d120x')
|
||||||
|
assertScan(
|
||||||
|
u'"z\\ud834x"',
|
||||||
|
u'z\ud834x')
|
||||||
|
assertScan(
|
||||||
|
u'"z\\udd20x"',
|
||||||
|
u'z\udd20x')
|
||||||
|
assertScan(
|
||||||
|
u'"z\ud834x"',
|
||||||
|
u'z\ud834x')
|
||||||
|
# It may look strange to join strings together, but Python is drunk.
|
||||||
|
# https://gist.github.com/etrepum/5538443
|
||||||
|
assertScan(
|
||||||
|
u'"z\\ud834\udd20x12345"',
|
||||||
|
u''.join([u'z\ud834', u'\udd20x12345']))
|
||||||
|
assertScan(
|
||||||
|
u'"z\ud834\\udd20x"',
|
||||||
|
u''.join([u'z\ud834', u'\udd20x']))
|
||||||
|
# these have different behavior given UTF8 input, because the surrogate
|
||||||
|
# pair may be joined (in maxunicode > 65535 builds)
|
||||||
|
assertScan(
|
||||||
|
u''.join([u'"z\ud834', u'\udd20x"']),
|
||||||
|
u''.join([u'z\ud834', u'\udd20x']),
|
||||||
|
test_utf8=False)
|
||||||
|
|
||||||
|
self.assertRaises(ValueError,
|
||||||
|
scanstring, u'"z\\ud83x"', 1, None, True)
|
||||||
|
self.assertRaises(ValueError,
|
||||||
|
scanstring, u'"z\\ud834\\udd2x"', 1, None, True)
|
|
@ -0,0 +1,42 @@
|
||||||
|
import textwrap
|
||||||
|
from unittest import TestCase
|
||||||
|
|
||||||
|
import simplejson as json
|
||||||
|
|
||||||
|
|
||||||
|
class TestSeparators(TestCase):
|
||||||
|
def test_separators(self):
|
||||||
|
h = [['blorpie'], ['whoops'], [], 'd-shtaeou', 'd-nthiouh', 'i-vhbjkhnth',
|
||||||
|
{'nifty': 87}, {'field': 'yes', 'morefield': False} ]
|
||||||
|
|
||||||
|
expect = textwrap.dedent("""\
|
||||||
|
[
|
||||||
|
[
|
||||||
|
"blorpie"
|
||||||
|
] ,
|
||||||
|
[
|
||||||
|
"whoops"
|
||||||
|
] ,
|
||||||
|
[] ,
|
||||||
|
"d-shtaeou" ,
|
||||||
|
"d-nthiouh" ,
|
||||||
|
"i-vhbjkhnth" ,
|
||||||
|
{
|
||||||
|
"nifty" : 87
|
||||||
|
} ,
|
||||||
|
{
|
||||||
|
"field" : "yes" ,
|
||||||
|
"morefield" : false
|
||||||
|
}
|
||||||
|
]""")
|
||||||
|
|
||||||
|
|
||||||
|
d1 = json.dumps(h)
|
||||||
|
d2 = json.dumps(h, indent=' ', sort_keys=True, separators=(' ,', ' : '))
|
||||||
|
|
||||||
|
h1 = json.loads(d1)
|
||||||
|
h2 = json.loads(d2)
|
||||||
|
|
||||||
|
self.assertEqual(h1, h)
|
||||||
|
self.assertEqual(h2, h)
|
||||||
|
self.assertEqual(d2, expect)
|
|
@ -0,0 +1,114 @@
|
||||||
|
from __future__ import with_statement
|
||||||
|
|
||||||
|
import sys
|
||||||
|
import unittest
|
||||||
|
from unittest import TestCase
|
||||||
|
|
||||||
|
import simplejson
|
||||||
|
from simplejson import encoder, decoder, scanner
|
||||||
|
from simplejson.compat import PY3, long_type, b
|
||||||
|
|
||||||
|
|
||||||
|
def has_speedups():
|
||||||
|
return encoder.c_make_encoder is not None
|
||||||
|
|
||||||
|
|
||||||
|
def skip_if_speedups_missing(func):
|
||||||
|
def wrapper(*args, **kwargs):
|
||||||
|
if not has_speedups():
|
||||||
|
if hasattr(unittest, 'SkipTest'):
|
||||||
|
raise unittest.SkipTest("C Extension not available")
|
||||||
|
else:
|
||||||
|
sys.stdout.write("C Extension not available")
|
||||||
|
return
|
||||||
|
return func(*args, **kwargs)
|
||||||
|
|
||||||
|
return wrapper
|
||||||
|
|
||||||
|
|
||||||
|
class BadBool:
|
||||||
|
def __bool__(self):
|
||||||
|
1/0
|
||||||
|
__nonzero__ = __bool__
|
||||||
|
|
||||||
|
|
||||||
|
class TestDecode(TestCase):
|
||||||
|
@skip_if_speedups_missing
|
||||||
|
def test_make_scanner(self):
|
||||||
|
self.assertRaises(AttributeError, scanner.c_make_scanner, 1)
|
||||||
|
|
||||||
|
@skip_if_speedups_missing
|
||||||
|
def test_bad_bool_args(self):
|
||||||
|
def test(value):
|
||||||
|
decoder.JSONDecoder(strict=BadBool()).decode(value)
|
||||||
|
self.assertRaises(ZeroDivisionError, test, '""')
|
||||||
|
self.assertRaises(ZeroDivisionError, test, '{}')
|
||||||
|
if not PY3:
|
||||||
|
self.assertRaises(ZeroDivisionError, test, u'""')
|
||||||
|
self.assertRaises(ZeroDivisionError, test, u'{}')
|
||||||
|
|
||||||
|
class TestEncode(TestCase):
|
||||||
|
@skip_if_speedups_missing
|
||||||
|
def test_make_encoder(self):
|
||||||
|
self.assertRaises(
|
||||||
|
TypeError,
|
||||||
|
encoder.c_make_encoder,
|
||||||
|
None,
|
||||||
|
("\xCD\x7D\x3D\x4E\x12\x4C\xF9\x79\xD7"
|
||||||
|
"\x52\xBA\x82\xF2\x27\x4A\x7D\xA0\xCA\x75"),
|
||||||
|
None
|
||||||
|
)
|
||||||
|
|
||||||
|
@skip_if_speedups_missing
|
||||||
|
def test_bad_str_encoder(self):
|
||||||
|
# Issue #31505: There shouldn't be an assertion failure in case
|
||||||
|
# c_make_encoder() receives a bad encoder() argument.
|
||||||
|
import decimal
|
||||||
|
def bad_encoder1(*args):
|
||||||
|
return None
|
||||||
|
enc = encoder.c_make_encoder(
|
||||||
|
None, lambda obj: str(obj),
|
||||||
|
bad_encoder1, None, ': ', ', ',
|
||||||
|
False, False, False, {}, False, False, False,
|
||||||
|
None, None, 'utf-8', False, False, decimal.Decimal, False)
|
||||||
|
self.assertRaises(TypeError, enc, 'spam', 4)
|
||||||
|
self.assertRaises(TypeError, enc, {'spam': 42}, 4)
|
||||||
|
|
||||||
|
def bad_encoder2(*args):
|
||||||
|
1/0
|
||||||
|
enc = encoder.c_make_encoder(
|
||||||
|
None, lambda obj: str(obj),
|
||||||
|
bad_encoder2, None, ': ', ', ',
|
||||||
|
False, False, False, {}, False, False, False,
|
||||||
|
None, None, 'utf-8', False, False, decimal.Decimal, False)
|
||||||
|
self.assertRaises(ZeroDivisionError, enc, 'spam', 4)
|
||||||
|
|
||||||
|
@skip_if_speedups_missing
|
||||||
|
def test_bad_bool_args(self):
|
||||||
|
def test(name):
|
||||||
|
encoder.JSONEncoder(**{name: BadBool()}).encode({})
|
||||||
|
self.assertRaises(ZeroDivisionError, test, 'skipkeys')
|
||||||
|
self.assertRaises(ZeroDivisionError, test, 'ensure_ascii')
|
||||||
|
self.assertRaises(ZeroDivisionError, test, 'check_circular')
|
||||||
|
self.assertRaises(ZeroDivisionError, test, 'allow_nan')
|
||||||
|
self.assertRaises(ZeroDivisionError, test, 'sort_keys')
|
||||||
|
self.assertRaises(ZeroDivisionError, test, 'use_decimal')
|
||||||
|
self.assertRaises(ZeroDivisionError, test, 'namedtuple_as_object')
|
||||||
|
self.assertRaises(ZeroDivisionError, test, 'tuple_as_array')
|
||||||
|
self.assertRaises(ZeroDivisionError, test, 'bigint_as_string')
|
||||||
|
self.assertRaises(ZeroDivisionError, test, 'for_json')
|
||||||
|
self.assertRaises(ZeroDivisionError, test, 'ignore_nan')
|
||||||
|
self.assertRaises(ZeroDivisionError, test, 'iterable_as_array')
|
||||||
|
|
||||||
|
@skip_if_speedups_missing
|
||||||
|
def test_int_as_string_bitcount_overflow(self):
|
||||||
|
long_count = long_type(2)**32+31
|
||||||
|
def test():
|
||||||
|
encoder.JSONEncoder(int_as_string_bitcount=long_count).encode(0)
|
||||||
|
self.assertRaises((TypeError, OverflowError), test)
|
||||||
|
|
||||||
|
if PY3:
|
||||||
|
@skip_if_speedups_missing
|
||||||
|
def test_bad_encoding(self):
|
||||||
|
with self.assertRaises(UnicodeEncodeError):
|
||||||
|
encoder.JSONEncoder(encoding='\udcff').encode({b('key'): 123})
|
|
@ -0,0 +1,21 @@
|
||||||
|
from unittest import TestCase
|
||||||
|
|
||||||
|
import simplejson
|
||||||
|
from simplejson.compat import text_type
|
||||||
|
|
||||||
|
# Tests for issue demonstrated in https://github.com/simplejson/simplejson/issues/144
|
||||||
|
class WonkyTextSubclass(text_type):
|
||||||
|
def __getslice__(self, start, end):
|
||||||
|
return self.__class__('not what you wanted!')
|
||||||
|
|
||||||
|
class TestStrSubclass(TestCase):
|
||||||
|
def test_dump_load(self):
|
||||||
|
for s in ['', '"hello"', 'text', u'\u005c']:
|
||||||
|
self.assertEqual(
|
||||||
|
s,
|
||||||
|
simplejson.loads(simplejson.dumps(WonkyTextSubclass(s))))
|
||||||
|
|
||||||
|
self.assertEqual(
|
||||||
|
s,
|
||||||
|
simplejson.loads(simplejson.dumps(WonkyTextSubclass(s),
|
||||||
|
ensure_ascii=False)))
|
|
@ -0,0 +1,37 @@
|
||||||
|
from unittest import TestCase
|
||||||
|
import simplejson as json
|
||||||
|
|
||||||
|
from decimal import Decimal
|
||||||
|
|
||||||
|
class AlternateInt(int):
|
||||||
|
def __repr__(self):
|
||||||
|
return 'invalid json'
|
||||||
|
__str__ = __repr__
|
||||||
|
|
||||||
|
|
||||||
|
class AlternateFloat(float):
|
||||||
|
def __repr__(self):
|
||||||
|
return 'invalid json'
|
||||||
|
__str__ = __repr__
|
||||||
|
|
||||||
|
|
||||||
|
# class AlternateDecimal(Decimal):
|
||||||
|
# def __repr__(self):
|
||||||
|
# return 'invalid json'
|
||||||
|
|
||||||
|
|
||||||
|
class TestSubclass(TestCase):
|
||||||
|
def test_int(self):
|
||||||
|
self.assertEqual(json.dumps(AlternateInt(1)), '1')
|
||||||
|
self.assertEqual(json.dumps(AlternateInt(-1)), '-1')
|
||||||
|
self.assertEqual(json.loads(json.dumps({AlternateInt(1): 1})), {'1': 1})
|
||||||
|
|
||||||
|
def test_float(self):
|
||||||
|
self.assertEqual(json.dumps(AlternateFloat(1.0)), '1.0')
|
||||||
|
self.assertEqual(json.dumps(AlternateFloat(-1.0)), '-1.0')
|
||||||
|
self.assertEqual(json.loads(json.dumps({AlternateFloat(1.0): 1})), {'1.0': 1})
|
||||||
|
|
||||||
|
# NOTE: Decimal subclasses are not supported as-is
|
||||||
|
# def test_decimal(self):
|
||||||
|
# self.assertEqual(json.dumps(AlternateDecimal('1.0')), '1.0')
|
||||||
|
# self.assertEqual(json.dumps(AlternateDecimal('-1.0')), '-1.0')
|
|
@ -0,0 +1,114 @@
|
||||||
|
from __future__ import with_statement
|
||||||
|
import os
|
||||||
|
import sys
|
||||||
|
import textwrap
|
||||||
|
import unittest
|
||||||
|
import subprocess
|
||||||
|
import tempfile
|
||||||
|
try:
|
||||||
|
# Python 3.x
|
||||||
|
from test.support import strip_python_stderr
|
||||||
|
except ImportError:
|
||||||
|
# Python 2.6+
|
||||||
|
try:
|
||||||
|
from test.test_support import strip_python_stderr
|
||||||
|
except ImportError:
|
||||||
|
# Python 2.5
|
||||||
|
import re
|
||||||
|
def strip_python_stderr(stderr):
|
||||||
|
return re.sub(
|
||||||
|
r"\[\d+ refs\]\r?\n?$".encode(),
|
||||||
|
"".encode(),
|
||||||
|
stderr).strip()
|
||||||
|
|
||||||
|
def open_temp_file():
|
||||||
|
if sys.version_info >= (2, 6):
|
||||||
|
file = tempfile.NamedTemporaryFile(delete=False)
|
||||||
|
filename = file.name
|
||||||
|
else:
|
||||||
|
fd, filename = tempfile.mkstemp()
|
||||||
|
file = os.fdopen(fd, 'w+b')
|
||||||
|
return file, filename
|
||||||
|
|
||||||
|
class TestTool(unittest.TestCase):
|
||||||
|
data = """
|
||||||
|
|
||||||
|
[["blorpie"],[ "whoops" ] , [
|
||||||
|
],\t"d-shtaeou",\r"d-nthiouh",
|
||||||
|
"i-vhbjkhnth", {"nifty":87}, {"morefield" :\tfalse,"field"
|
||||||
|
:"yes"} ]
|
||||||
|
"""
|
||||||
|
|
||||||
|
expect = textwrap.dedent("""\
|
||||||
|
[
|
||||||
|
[
|
||||||
|
"blorpie"
|
||||||
|
],
|
||||||
|
[
|
||||||
|
"whoops"
|
||||||
|
],
|
||||||
|
[],
|
||||||
|
"d-shtaeou",
|
||||||
|
"d-nthiouh",
|
||||||
|
"i-vhbjkhnth",
|
||||||
|
{
|
||||||
|
"nifty": 87
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"field": "yes",
|
||||||
|
"morefield": false
|
||||||
|
}
|
||||||
|
]
|
||||||
|
""")
|
||||||
|
|
||||||
|
def runTool(self, args=None, data=None):
|
||||||
|
argv = [sys.executable, '-m', 'simplejson.tool']
|
||||||
|
if args:
|
||||||
|
argv.extend(args)
|
||||||
|
proc = subprocess.Popen(argv,
|
||||||
|
stdin=subprocess.PIPE,
|
||||||
|
stderr=subprocess.PIPE,
|
||||||
|
stdout=subprocess.PIPE)
|
||||||
|
out, err = proc.communicate(data)
|
||||||
|
self.assertEqual(strip_python_stderr(err), ''.encode())
|
||||||
|
self.assertEqual(proc.returncode, 0)
|
||||||
|
return out.decode('utf8').splitlines()
|
||||||
|
|
||||||
|
def test_stdin_stdout(self):
|
||||||
|
self.assertEqual(
|
||||||
|
self.runTool(data=self.data.encode()),
|
||||||
|
self.expect.splitlines())
|
||||||
|
|
||||||
|
def test_infile_stdout(self):
|
||||||
|
infile, infile_name = open_temp_file()
|
||||||
|
try:
|
||||||
|
infile.write(self.data.encode())
|
||||||
|
infile.close()
|
||||||
|
self.assertEqual(
|
||||||
|
self.runTool(args=[infile_name]),
|
||||||
|
self.expect.splitlines())
|
||||||
|
finally:
|
||||||
|
os.unlink(infile_name)
|
||||||
|
|
||||||
|
def test_infile_outfile(self):
|
||||||
|
infile, infile_name = open_temp_file()
|
||||||
|
try:
|
||||||
|
infile.write(self.data.encode())
|
||||||
|
infile.close()
|
||||||
|
# outfile will get overwritten by tool, so the delete
|
||||||
|
# may not work on some platforms. Do it manually.
|
||||||
|
outfile, outfile_name = open_temp_file()
|
||||||
|
try:
|
||||||
|
outfile.close()
|
||||||
|
self.assertEqual(
|
||||||
|
self.runTool(args=[infile_name, outfile_name]),
|
||||||
|
[])
|
||||||
|
with open(outfile_name, 'rb') as f:
|
||||||
|
self.assertEqual(
|
||||||
|
f.read().decode('utf8').splitlines(),
|
||||||
|
self.expect.splitlines()
|
||||||
|
)
|
||||||
|
finally:
|
||||||
|
os.unlink(outfile_name)
|
||||||
|
finally:
|
||||||
|
os.unlink(infile_name)
|
|
@ -0,0 +1,47 @@
|
||||||
|
import unittest
|
||||||
|
|
||||||
|
from simplejson.compat import StringIO
|
||||||
|
import simplejson as json
|
||||||
|
|
||||||
|
class TestTuples(unittest.TestCase):
|
||||||
|
def test_tuple_array_dumps(self):
|
||||||
|
t = (1, 2, 3)
|
||||||
|
expect = json.dumps(list(t))
|
||||||
|
# Default is True
|
||||||
|
self.assertEqual(expect, json.dumps(t))
|
||||||
|
self.assertEqual(expect, json.dumps(t, tuple_as_array=True))
|
||||||
|
self.assertRaises(TypeError, json.dumps, t, tuple_as_array=False)
|
||||||
|
# Ensure that the "default" does not get called
|
||||||
|
self.assertEqual(expect, json.dumps(t, default=repr))
|
||||||
|
self.assertEqual(expect, json.dumps(t, tuple_as_array=True,
|
||||||
|
default=repr))
|
||||||
|
# Ensure that the "default" gets called
|
||||||
|
self.assertEqual(
|
||||||
|
json.dumps(repr(t)),
|
||||||
|
json.dumps(t, tuple_as_array=False, default=repr))
|
||||||
|
|
||||||
|
def test_tuple_array_dump(self):
|
||||||
|
t = (1, 2, 3)
|
||||||
|
expect = json.dumps(list(t))
|
||||||
|
# Default is True
|
||||||
|
sio = StringIO()
|
||||||
|
json.dump(t, sio)
|
||||||
|
self.assertEqual(expect, sio.getvalue())
|
||||||
|
sio = StringIO()
|
||||||
|
json.dump(t, sio, tuple_as_array=True)
|
||||||
|
self.assertEqual(expect, sio.getvalue())
|
||||||
|
self.assertRaises(TypeError, json.dump, t, StringIO(),
|
||||||
|
tuple_as_array=False)
|
||||||
|
# Ensure that the "default" does not get called
|
||||||
|
sio = StringIO()
|
||||||
|
json.dump(t, sio, default=repr)
|
||||||
|
self.assertEqual(expect, sio.getvalue())
|
||||||
|
sio = StringIO()
|
||||||
|
json.dump(t, sio, tuple_as_array=True, default=repr)
|
||||||
|
self.assertEqual(expect, sio.getvalue())
|
||||||
|
# Ensure that the "default" gets called
|
||||||
|
sio = StringIO()
|
||||||
|
json.dump(t, sio, tuple_as_array=False, default=repr)
|
||||||
|
self.assertEqual(
|
||||||
|
json.dumps(repr(t)),
|
||||||
|
sio.getvalue())
|
|
@ -0,0 +1,154 @@
|
||||||
|
import sys
|
||||||
|
import codecs
|
||||||
|
from unittest import TestCase
|
||||||
|
|
||||||
|
import simplejson as json
|
||||||
|
from simplejson.compat import unichr, text_type, b, BytesIO
|
||||||
|
|
||||||
|
class TestUnicode(TestCase):
|
||||||
|
def test_encoding1(self):
|
||||||
|
encoder = json.JSONEncoder(encoding='utf-8')
|
||||||
|
u = u'\N{GREEK SMALL LETTER ALPHA}\N{GREEK CAPITAL LETTER OMEGA}'
|
||||||
|
s = u.encode('utf-8')
|
||||||
|
ju = encoder.encode(u)
|
||||||
|
js = encoder.encode(s)
|
||||||
|
self.assertEqual(ju, js)
|
||||||
|
|
||||||
|
def test_encoding2(self):
|
||||||
|
u = u'\N{GREEK SMALL LETTER ALPHA}\N{GREEK CAPITAL LETTER OMEGA}'
|
||||||
|
s = u.encode('utf-8')
|
||||||
|
ju = json.dumps(u, encoding='utf-8')
|
||||||
|
js = json.dumps(s, encoding='utf-8')
|
||||||
|
self.assertEqual(ju, js)
|
||||||
|
|
||||||
|
def test_encoding3(self):
|
||||||
|
u = u'\N{GREEK SMALL LETTER ALPHA}\N{GREEK CAPITAL LETTER OMEGA}'
|
||||||
|
j = json.dumps(u)
|
||||||
|
self.assertEqual(j, '"\\u03b1\\u03a9"')
|
||||||
|
|
||||||
|
def test_encoding4(self):
|
||||||
|
u = u'\N{GREEK SMALL LETTER ALPHA}\N{GREEK CAPITAL LETTER OMEGA}'
|
||||||
|
j = json.dumps([u])
|
||||||
|
self.assertEqual(j, '["\\u03b1\\u03a9"]')
|
||||||
|
|
||||||
|
def test_encoding5(self):
|
||||||
|
u = u'\N{GREEK SMALL LETTER ALPHA}\N{GREEK CAPITAL LETTER OMEGA}'
|
||||||
|
j = json.dumps(u, ensure_ascii=False)
|
||||||
|
self.assertEqual(j, u'"' + u + u'"')
|
||||||
|
|
||||||
|
def test_encoding6(self):
|
||||||
|
u = u'\N{GREEK SMALL LETTER ALPHA}\N{GREEK CAPITAL LETTER OMEGA}'
|
||||||
|
j = json.dumps([u], ensure_ascii=False)
|
||||||
|
self.assertEqual(j, u'["' + u + u'"]')
|
||||||
|
|
||||||
|
def test_big_unicode_encode(self):
|
||||||
|
u = u'\U0001d120'
|
||||||
|
self.assertEqual(json.dumps(u), '"\\ud834\\udd20"')
|
||||||
|
self.assertEqual(json.dumps(u, ensure_ascii=False), u'"\U0001d120"')
|
||||||
|
|
||||||
|
def test_big_unicode_decode(self):
|
||||||
|
u = u'z\U0001d120x'
|
||||||
|
self.assertEqual(json.loads('"' + u + '"'), u)
|
||||||
|
self.assertEqual(json.loads('"z\\ud834\\udd20x"'), u)
|
||||||
|
|
||||||
|
def test_unicode_decode(self):
|
||||||
|
for i in range(0, 0xd7ff):
|
||||||
|
u = unichr(i)
|
||||||
|
#s = '"\\u{0:04x}"'.format(i)
|
||||||
|
s = '"\\u%04x"' % (i,)
|
||||||
|
self.assertEqual(json.loads(s), u)
|
||||||
|
|
||||||
|
def test_object_pairs_hook_with_unicode(self):
|
||||||
|
s = u'{"xkd":1, "kcw":2, "art":3, "hxm":4, "qrt":5, "pad":6, "hoy":7}'
|
||||||
|
p = [(u"xkd", 1), (u"kcw", 2), (u"art", 3), (u"hxm", 4),
|
||||||
|
(u"qrt", 5), (u"pad", 6), (u"hoy", 7)]
|
||||||
|
self.assertEqual(json.loads(s), eval(s))
|
||||||
|
self.assertEqual(json.loads(s, object_pairs_hook=lambda x: x), p)
|
||||||
|
od = json.loads(s, object_pairs_hook=json.OrderedDict)
|
||||||
|
self.assertEqual(od, json.OrderedDict(p))
|
||||||
|
self.assertEqual(type(od), json.OrderedDict)
|
||||||
|
# the object_pairs_hook takes priority over the object_hook
|
||||||
|
self.assertEqual(json.loads(s,
|
||||||
|
object_pairs_hook=json.OrderedDict,
|
||||||
|
object_hook=lambda x: None),
|
||||||
|
json.OrderedDict(p))
|
||||||
|
|
||||||
|
|
||||||
|
def test_default_encoding(self):
|
||||||
|
self.assertEqual(json.loads(u'{"a": "\xe9"}'.encode('utf-8')),
|
||||||
|
{'a': u'\xe9'})
|
||||||
|
|
||||||
|
def test_unicode_preservation(self):
|
||||||
|
self.assertEqual(type(json.loads(u'""')), text_type)
|
||||||
|
self.assertEqual(type(json.loads(u'"a"')), text_type)
|
||||||
|
self.assertEqual(type(json.loads(u'["a"]')[0]), text_type)
|
||||||
|
|
||||||
|
def test_ensure_ascii_false_returns_unicode(self):
|
||||||
|
# http://code.google.com/p/simplejson/issues/detail?id=48
|
||||||
|
self.assertEqual(type(json.dumps([], ensure_ascii=False)), text_type)
|
||||||
|
self.assertEqual(type(json.dumps(0, ensure_ascii=False)), text_type)
|
||||||
|
self.assertEqual(type(json.dumps({}, ensure_ascii=False)), text_type)
|
||||||
|
self.assertEqual(type(json.dumps("", ensure_ascii=False)), text_type)
|
||||||
|
|
||||||
|
def test_ensure_ascii_false_bytestring_encoding(self):
|
||||||
|
# http://code.google.com/p/simplejson/issues/detail?id=48
|
||||||
|
doc1 = {u'quux': b('Arr\xc3\xaat sur images')}
|
||||||
|
doc2 = {u'quux': u'Arr\xeat sur images'}
|
||||||
|
doc_ascii = '{"quux": "Arr\\u00eat sur images"}'
|
||||||
|
doc_unicode = u'{"quux": "Arr\xeat sur images"}'
|
||||||
|
self.assertEqual(json.dumps(doc1), doc_ascii)
|
||||||
|
self.assertEqual(json.dumps(doc2), doc_ascii)
|
||||||
|
self.assertEqual(json.dumps(doc1, ensure_ascii=False), doc_unicode)
|
||||||
|
self.assertEqual(json.dumps(doc2, ensure_ascii=False), doc_unicode)
|
||||||
|
|
||||||
|
def test_ensure_ascii_linebreak_encoding(self):
|
||||||
|
# http://timelessrepo.com/json-isnt-a-javascript-subset
|
||||||
|
s1 = u'\u2029\u2028'
|
||||||
|
s2 = s1.encode('utf8')
|
||||||
|
expect = '"\\u2029\\u2028"'
|
||||||
|
expect_non_ascii = u'"\u2029\u2028"'
|
||||||
|
self.assertEqual(json.dumps(s1), expect)
|
||||||
|
self.assertEqual(json.dumps(s2), expect)
|
||||||
|
self.assertEqual(json.dumps(s1, ensure_ascii=False), expect_non_ascii)
|
||||||
|
self.assertEqual(json.dumps(s2, ensure_ascii=False), expect_non_ascii)
|
||||||
|
|
||||||
|
def test_invalid_escape_sequences(self):
|
||||||
|
# incomplete escape sequence
|
||||||
|
self.assertRaises(json.JSONDecodeError, json.loads, '"\\u')
|
||||||
|
self.assertRaises(json.JSONDecodeError, json.loads, '"\\u1')
|
||||||
|
self.assertRaises(json.JSONDecodeError, json.loads, '"\\u12')
|
||||||
|
self.assertRaises(json.JSONDecodeError, json.loads, '"\\u123')
|
||||||
|
self.assertRaises(json.JSONDecodeError, json.loads, '"\\u1234')
|
||||||
|
# invalid escape sequence
|
||||||
|
self.assertRaises(json.JSONDecodeError, json.loads, '"\\u123x"')
|
||||||
|
self.assertRaises(json.JSONDecodeError, json.loads, '"\\u12x4"')
|
||||||
|
self.assertRaises(json.JSONDecodeError, json.loads, '"\\u1x34"')
|
||||||
|
self.assertRaises(json.JSONDecodeError, json.loads, '"\\ux234"')
|
||||||
|
if sys.maxunicode > 65535:
|
||||||
|
# invalid escape sequence for low surrogate
|
||||||
|
self.assertRaises(json.JSONDecodeError, json.loads, '"\\ud800\\u"')
|
||||||
|
self.assertRaises(json.JSONDecodeError, json.loads, '"\\ud800\\u0"')
|
||||||
|
self.assertRaises(json.JSONDecodeError, json.loads, '"\\ud800\\u00"')
|
||||||
|
self.assertRaises(json.JSONDecodeError, json.loads, '"\\ud800\\u000"')
|
||||||
|
self.assertRaises(json.JSONDecodeError, json.loads, '"\\ud800\\u000x"')
|
||||||
|
self.assertRaises(json.JSONDecodeError, json.loads, '"\\ud800\\u00x0"')
|
||||||
|
self.assertRaises(json.JSONDecodeError, json.loads, '"\\ud800\\u0x00"')
|
||||||
|
self.assertRaises(json.JSONDecodeError, json.loads, '"\\ud800\\ux000"')
|
||||||
|
|
||||||
|
def test_ensure_ascii_still_works(self):
|
||||||
|
# in the ascii range, ensure that everything is the same
|
||||||
|
for c in map(unichr, range(0, 127)):
|
||||||
|
self.assertEqual(
|
||||||
|
json.dumps(c, ensure_ascii=False),
|
||||||
|
json.dumps(c))
|
||||||
|
snowman = u'\N{SNOWMAN}'
|
||||||
|
self.assertEqual(
|
||||||
|
json.dumps(c, ensure_ascii=False),
|
||||||
|
'"' + c + '"')
|
||||||
|
|
||||||
|
def test_strip_bom(self):
|
||||||
|
content = u"\u3053\u3093\u306b\u3061\u308f"
|
||||||
|
json_doc = codecs.BOM_UTF8 + b(json.dumps(content))
|
||||||
|
self.assertEqual(json.load(BytesIO(json_doc)), content)
|
||||||
|
for doc in json_doc, json_doc.decode('utf8'):
|
||||||
|
self.assertEqual(json.loads(doc), content)
|
|
@ -0,0 +1,42 @@
|
||||||
|
r"""Command-line tool to validate and pretty-print JSON
|
||||||
|
|
||||||
|
Usage::
|
||||||
|
|
||||||
|
$ echo '{"json":"obj"}' | python -m simplejson.tool
|
||||||
|
{
|
||||||
|
"json": "obj"
|
||||||
|
}
|
||||||
|
$ echo '{ 1.2:3.4}' | python -m simplejson.tool
|
||||||
|
Expecting property name: line 1 column 2 (char 2)
|
||||||
|
|
||||||
|
"""
|
||||||
|
from __future__ import with_statement
|
||||||
|
import sys
|
||||||
|
import simplejson as json
|
||||||
|
|
||||||
|
def main():
|
||||||
|
if len(sys.argv) == 1:
|
||||||
|
infile = sys.stdin
|
||||||
|
outfile = sys.stdout
|
||||||
|
elif len(sys.argv) == 2:
|
||||||
|
infile = open(sys.argv[1], 'r')
|
||||||
|
outfile = sys.stdout
|
||||||
|
elif len(sys.argv) == 3:
|
||||||
|
infile = open(sys.argv[1], 'r')
|
||||||
|
outfile = open(sys.argv[2], 'w')
|
||||||
|
else:
|
||||||
|
raise SystemExit(sys.argv[0] + " [infile [outfile]]")
|
||||||
|
with infile:
|
||||||
|
try:
|
||||||
|
obj = json.load(infile,
|
||||||
|
object_pairs_hook=json.OrderedDict,
|
||||||
|
use_decimal=True)
|
||||||
|
except ValueError:
|
||||||
|
raise SystemExit(sys.exc_info()[1])
|
||||||
|
with outfile:
|
||||||
|
json.dump(obj, outfile, sort_keys=True, indent=' ', use_decimal=True)
|
||||||
|
outfile.write('\n')
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
main()
|
|
@ -1,9 +1,9 @@
|
||||||
.TH "speedtest-cli" 1 "2014-04-23" "speedtest-cli"
|
.TH "speedtest-cli" 1 "2018-01-05" "speedtest-cli"
|
||||||
.SH NAME
|
.SH NAME
|
||||||
speedtest\-cli \- Test your bandwidth througput using speedtest.net
|
speedtest\-cli \- Command line interface for testing internet bandwidth using speedtest.net
|
||||||
.SH SYNOPSIS
|
.SH SYNOPSIS
|
||||||
.B speedtest\-cli
|
.B speedtest\-cli
|
||||||
[OPTION]...
|
[OPTION...]
|
||||||
.SH DESCRIPTION
|
.SH DESCRIPTION
|
||||||
Speedtest.net is a web service for testing your broadband connection by downloading a file
|
Speedtest.net is a web service for testing your broadband connection by downloading a file
|
||||||
from a nearby speedtest.net server on the web. This tool allows you to access the service
|
from a nearby speedtest.net server on the web. This tool allows you to access the service
|
||||||
|
@ -23,14 +23,29 @@ Displays usage for the tool.
|
||||||
|
|
||||||
.B Options
|
.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\-\-single\fR
|
||||||
|
.RS
|
||||||
|
Only use a single connection instead of multiple. This simulates a typical file transfer.
|
||||||
|
.RE
|
||||||
|
|
||||||
\fB\-\-bytes\fR
|
\fB\-\-bytes\fR
|
||||||
.RS
|
.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
|
.RE
|
||||||
|
|
||||||
\fB\-\-share\fR
|
\fB\-\-share\fR
|
||||||
.RS
|
.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
|
.RE
|
||||||
|
|
||||||
\fB\-\-simple\fR
|
\fB\-\-simple\fR
|
||||||
|
@ -38,6 +53,26 @@ Generate and provide a URL to the speedtest.net share results image
|
||||||
Suppress verbose output, only show basic information
|
Suppress verbose output, only show basic information
|
||||||
.RE
|
.RE
|
||||||
|
|
||||||
|
\fB\-\-csv\fR
|
||||||
|
.RS
|
||||||
|
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
|
||||||
|
.RS
|
||||||
|
Single character delimiter to use in CSV output. Default ","
|
||||||
|
.RE
|
||||||
|
|
||||||
|
\fB\-\-csv\-header\fR
|
||||||
|
.RS
|
||||||
|
Print CSV headers
|
||||||
|
.RE
|
||||||
|
|
||||||
|
\fB\-\-json\fR
|
||||||
|
.RS
|
||||||
|
Suppress verbose output, only show basic information in JSON format. Speeds listed in bit/s and not affected by \-\-bytes
|
||||||
|
.RE
|
||||||
|
|
||||||
\fB\-\-list\fR
|
\fB\-\-list\fR
|
||||||
.RS
|
.RS
|
||||||
Display a list of speedtest.net servers sorted by distance
|
Display a list of speedtest.net servers sorted by distance
|
||||||
|
@ -45,7 +80,12 @@ Display a list of speedtest.net servers sorted by distance
|
||||||
|
|
||||||
\fB\-\-server SERVER\fR
|
\fB\-\-server SERVER\fR
|
||||||
.RS
|
.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
|
.RE
|
||||||
|
|
||||||
\fB\-\-mini MINI\fR
|
\fB\-\-mini MINI\fR
|
||||||
|
@ -58,6 +98,21 @@ URL of the Speedtest Mini server
|
||||||
Source IP address to bind to
|
Source IP address to bind to
|
||||||
.RE
|
.RE
|
||||||
|
|
||||||
|
\fB\-\-timeout TIMEOUT\fR
|
||||||
|
.RS
|
||||||
|
HTTP timeout in seconds. Default 10
|
||||||
|
.RE
|
||||||
|
|
||||||
|
\fB\-\-secure\fR
|
||||||
|
.RS
|
||||||
|
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
|
\fB\-\-version\fR
|
||||||
.RS
|
.RS
|
||||||
Show the version number and exit
|
Show the version number and exit
|
||||||
|
|
Binary file not shown.
File diff suppressed because it is too large
Load Diff
761
speedtest_cli.py
761
speedtest_cli.py
|
@ -1,761 +0,0 @@
|
||||||
#!/usr/bin/env python
|
|
||||||
# -*- coding: utf-8 -*-
|
|
||||||
# Copyright 2012-2014 Matt Martz
|
|
||||||
# All Rights Reserved.
|
|
||||||
#
|
|
||||||
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
|
||||||
# not use this file except in compliance with the License. You may obtain
|
|
||||||
# a copy of the License at
|
|
||||||
#
|
|
||||||
# http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
#
|
|
||||||
# Unless required by applicable law or agreed to in writing, software
|
|
||||||
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
||||||
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
||||||
# License for the specific language governing permissions and limitations
|
|
||||||
# under the License.
|
|
||||||
|
|
||||||
__version__ = '0.3.1'
|
|
||||||
|
|
||||||
# Some global variables we use
|
|
||||||
user_agent = 'speedtest-cli/%s' % __version__
|
|
||||||
source = None
|
|
||||||
shutdown_event = None
|
|
||||||
|
|
||||||
import os
|
|
||||||
import re
|
|
||||||
import sys
|
|
||||||
import math
|
|
||||||
import signal
|
|
||||||
import socket
|
|
||||||
import timeit
|
|
||||||
import threading
|
|
||||||
|
|
||||||
# Used for bound_interface
|
|
||||||
socket_socket = socket.socket
|
|
||||||
|
|
||||||
try:
|
|
||||||
import xml.etree.cElementTree as ET
|
|
||||||
except ImportError:
|
|
||||||
try:
|
|
||||||
import xml.etree.ElementTree as ET
|
|
||||||
except ImportError:
|
|
||||||
from xml.dom import minidom as DOM
|
|
||||||
ET = None
|
|
||||||
|
|
||||||
# Begin import game to handle Python 2 and Python 3
|
|
||||||
try:
|
|
||||||
from urllib2 import urlopen, Request, HTTPError, URLError
|
|
||||||
except ImportError:
|
|
||||||
from urllib.request import urlopen, Request, HTTPError, URLError
|
|
||||||
|
|
||||||
try:
|
|
||||||
from httplib import HTTPConnection, HTTPSConnection
|
|
||||||
except ImportError:
|
|
||||||
from http.client import HTTPConnection, HTTPSConnection
|
|
||||||
|
|
||||||
try:
|
|
||||||
from Queue import Queue
|
|
||||||
except ImportError:
|
|
||||||
from queue import Queue
|
|
||||||
|
|
||||||
try:
|
|
||||||
from urlparse import urlparse
|
|
||||||
except ImportError:
|
|
||||||
from urllib.parse import urlparse
|
|
||||||
|
|
||||||
try:
|
|
||||||
from urlparse import parse_qs
|
|
||||||
except ImportError:
|
|
||||||
try:
|
|
||||||
from urllib.parse import parse_qs
|
|
||||||
except ImportError:
|
|
||||||
from cgi import parse_qs
|
|
||||||
|
|
||||||
try:
|
|
||||||
from hashlib import md5
|
|
||||||
except ImportError:
|
|
||||||
from md5 import md5
|
|
||||||
|
|
||||||
try:
|
|
||||||
from argparse import ArgumentParser as ArgParser
|
|
||||||
except ImportError:
|
|
||||||
from optparse import OptionParser as ArgParser
|
|
||||||
|
|
||||||
try:
|
|
||||||
import builtins
|
|
||||||
except ImportError:
|
|
||||||
def print_(*args, **kwargs):
|
|
||||||
"""The new-style print function taken from
|
|
||||||
https://pypi.python.org/pypi/six/
|
|
||||||
|
|
||||||
"""
|
|
||||||
fp = kwargs.pop("file", sys.stdout)
|
|
||||||
if fp is None:
|
|
||||||
return
|
|
||||||
|
|
||||||
def write(data):
|
|
||||||
if not isinstance(data, basestring):
|
|
||||||
data = str(data)
|
|
||||||
fp.write(data)
|
|
||||||
|
|
||||||
want_unicode = False
|
|
||||||
sep = kwargs.pop("sep", None)
|
|
||||||
if sep is not None:
|
|
||||||
if isinstance(sep, unicode):
|
|
||||||
want_unicode = True
|
|
||||||
elif not isinstance(sep, str):
|
|
||||||
raise TypeError("sep must be None or a string")
|
|
||||||
end = kwargs.pop("end", None)
|
|
||||||
if end is not None:
|
|
||||||
if isinstance(end, unicode):
|
|
||||||
want_unicode = True
|
|
||||||
elif not isinstance(end, str):
|
|
||||||
raise TypeError("end must be None or a string")
|
|
||||||
if kwargs:
|
|
||||||
raise TypeError("invalid keyword arguments to print()")
|
|
||||||
if not want_unicode:
|
|
||||||
for arg in args:
|
|
||||||
if isinstance(arg, unicode):
|
|
||||||
want_unicode = True
|
|
||||||
break
|
|
||||||
if want_unicode:
|
|
||||||
newline = unicode("\n")
|
|
||||||
space = unicode(" ")
|
|
||||||
else:
|
|
||||||
newline = "\n"
|
|
||||||
space = " "
|
|
||||||
if sep is None:
|
|
||||||
sep = space
|
|
||||||
if end is None:
|
|
||||||
end = newline
|
|
||||||
for i, arg in enumerate(args):
|
|
||||||
if i:
|
|
||||||
write(sep)
|
|
||||||
write(arg)
|
|
||||||
write(end)
|
|
||||||
else:
|
|
||||||
print_ = getattr(builtins, 'print')
|
|
||||||
del builtins
|
|
||||||
|
|
||||||
|
|
||||||
class SpeedtestCliServerListError(Exception):
|
|
||||||
"""Internal Exception class used to indicate to move on to the next
|
|
||||||
URL for retrieving speedtest.net server details
|
|
||||||
|
|
||||||
"""
|
|
||||||
|
|
||||||
|
|
||||||
def bound_socket(*args, **kwargs):
|
|
||||||
"""Bind socket to a specified source IP address"""
|
|
||||||
|
|
||||||
global source
|
|
||||||
sock = socket_socket(*args, **kwargs)
|
|
||||||
sock.bind((source, 0))
|
|
||||||
return sock
|
|
||||||
|
|
||||||
|
|
||||||
def distance(origin, destination):
|
|
||||||
"""Determine distance between 2 sets of [lat,lon] in km"""
|
|
||||||
|
|
||||||
lat1, lon1 = origin
|
|
||||||
lat2, lon2 = destination
|
|
||||||
radius = 6371 # km
|
|
||||||
|
|
||||||
dlat = math.radians(lat2 - lat1)
|
|
||||||
dlon = math.radians(lon2 - lon1)
|
|
||||||
a = (math.sin(dlat / 2) * math.sin(dlat / 2) + math.cos(math.radians(lat1))
|
|
||||||
* math.cos(math.radians(lat2)) * math.sin(dlon / 2)
|
|
||||||
* math.sin(dlon / 2))
|
|
||||||
c = 2 * math.atan2(math.sqrt(a), math.sqrt(1 - a))
|
|
||||||
d = radius * c
|
|
||||||
|
|
||||||
return d
|
|
||||||
|
|
||||||
|
|
||||||
def build_request(url, data=None, headers={}):
|
|
||||||
"""Build a urllib2 request object
|
|
||||||
|
|
||||||
This function automatically adds a User-Agent header to all requests
|
|
||||||
|
|
||||||
"""
|
|
||||||
|
|
||||||
headers['User-Agent'] = user_agent
|
|
||||||
return Request(url, data=data, headers=headers)
|
|
||||||
|
|
||||||
|
|
||||||
def catch_request(request):
|
|
||||||
"""Helper function to catch common exceptions encountered when
|
|
||||||
establishing a connection with a HTTP/HTTPS request
|
|
||||||
|
|
||||||
"""
|
|
||||||
|
|
||||||
try:
|
|
||||||
uh = urlopen(request)
|
|
||||||
return uh
|
|
||||||
except (HTTPError, URLError, socket.error):
|
|
||||||
return False
|
|
||||||
|
|
||||||
|
|
||||||
class FileGetter(threading.Thread):
|
|
||||||
"""Thread class for retrieving a URL"""
|
|
||||||
|
|
||||||
def __init__(self, url, start):
|
|
||||||
self.url = url
|
|
||||||
self.result = None
|
|
||||||
self.starttime = start
|
|
||||||
threading.Thread.__init__(self)
|
|
||||||
|
|
||||||
def run(self):
|
|
||||||
self.result = [0]
|
|
||||||
try:
|
|
||||||
if (timeit.default_timer() - self.starttime) <= 10:
|
|
||||||
request = build_request(self.url)
|
|
||||||
f = urlopen(request)
|
|
||||||
while 1 and not shutdown_event.isSet():
|
|
||||||
self.result.append(len(f.read(10240)))
|
|
||||||
if self.result[-1] == 0:
|
|
||||||
break
|
|
||||||
f.close()
|
|
||||||
except IOError:
|
|
||||||
pass
|
|
||||||
|
|
||||||
|
|
||||||
def downloadSpeed(files, quiet=False):
|
|
||||||
"""Function to launch FileGetter threads and calculate download speeds"""
|
|
||||||
|
|
||||||
start = timeit.default_timer()
|
|
||||||
|
|
||||||
def producer(q, files):
|
|
||||||
for file in files:
|
|
||||||
thread = FileGetter(file, start)
|
|
||||||
thread.start()
|
|
||||||
q.put(thread, True)
|
|
||||||
if not quiet and not shutdown_event.isSet():
|
|
||||||
sys.stdout.write('.')
|
|
||||||
sys.stdout.flush()
|
|
||||||
|
|
||||||
finished = []
|
|
||||||
|
|
||||||
def consumer(q, total_files):
|
|
||||||
while len(finished) < total_files:
|
|
||||||
thread = q.get(True)
|
|
||||||
while thread.isAlive():
|
|
||||||
thread.join(timeout=0.1)
|
|
||||||
finished.append(sum(thread.result))
|
|
||||||
del thread
|
|
||||||
|
|
||||||
q = Queue(6)
|
|
||||||
prod_thread = threading.Thread(target=producer, args=(q, files))
|
|
||||||
cons_thread = threading.Thread(target=consumer, args=(q, len(files)))
|
|
||||||
start = timeit.default_timer()
|
|
||||||
prod_thread.start()
|
|
||||||
cons_thread.start()
|
|
||||||
while prod_thread.isAlive():
|
|
||||||
prod_thread.join(timeout=0.1)
|
|
||||||
while cons_thread.isAlive():
|
|
||||||
cons_thread.join(timeout=0.1)
|
|
||||||
return (sum(finished) / (timeit.default_timer() - start))
|
|
||||||
|
|
||||||
|
|
||||||
class FilePutter(threading.Thread):
|
|
||||||
"""Thread class for putting a URL"""
|
|
||||||
|
|
||||||
def __init__(self, url, start, size):
|
|
||||||
self.url = url
|
|
||||||
chars = '0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZ'
|
|
||||||
data = chars * (int(round(int(size) / 36.0)))
|
|
||||||
self.data = ('content1=%s' % data[0:int(size) - 9]).encode()
|
|
||||||
del data
|
|
||||||
self.result = None
|
|
||||||
self.starttime = start
|
|
||||||
threading.Thread.__init__(self)
|
|
||||||
|
|
||||||
def run(self):
|
|
||||||
try:
|
|
||||||
if ((timeit.default_timer() - self.starttime) <= 10 and
|
|
||||||
not shutdown_event.isSet()):
|
|
||||||
request = build_request(self.url, data=self.data)
|
|
||||||
f = urlopen(request)
|
|
||||||
f.read(11)
|
|
||||||
f.close()
|
|
||||||
self.result = len(self.data)
|
|
||||||
else:
|
|
||||||
self.result = 0
|
|
||||||
except IOError:
|
|
||||||
self.result = 0
|
|
||||||
|
|
||||||
|
|
||||||
def uploadSpeed(url, sizes, quiet=False):
|
|
||||||
"""Function to launch FilePutter threads and calculate upload speeds"""
|
|
||||||
|
|
||||||
start = timeit.default_timer()
|
|
||||||
|
|
||||||
def producer(q, sizes):
|
|
||||||
for size in sizes:
|
|
||||||
thread = FilePutter(url, start, size)
|
|
||||||
thread.start()
|
|
||||||
q.put(thread, True)
|
|
||||||
if not quiet and not shutdown_event.isSet():
|
|
||||||
sys.stdout.write('.')
|
|
||||||
sys.stdout.flush()
|
|
||||||
|
|
||||||
finished = []
|
|
||||||
|
|
||||||
def consumer(q, total_sizes):
|
|
||||||
while len(finished) < total_sizes:
|
|
||||||
thread = q.get(True)
|
|
||||||
while thread.isAlive():
|
|
||||||
thread.join(timeout=0.1)
|
|
||||||
finished.append(thread.result)
|
|
||||||
del thread
|
|
||||||
|
|
||||||
q = Queue(6)
|
|
||||||
prod_thread = threading.Thread(target=producer, args=(q, sizes))
|
|
||||||
cons_thread = threading.Thread(target=consumer, args=(q, len(sizes)))
|
|
||||||
start = timeit.default_timer()
|
|
||||||
prod_thread.start()
|
|
||||||
cons_thread.start()
|
|
||||||
while prod_thread.isAlive():
|
|
||||||
prod_thread.join(timeout=0.1)
|
|
||||||
while cons_thread.isAlive():
|
|
||||||
cons_thread.join(timeout=0.1)
|
|
||||||
return (sum(finished) / (timeit.default_timer() - start))
|
|
||||||
|
|
||||||
|
|
||||||
def getAttributesByTagName(dom, tagName):
|
|
||||||
"""Retrieve an attribute from an XML document and return it in a
|
|
||||||
consistent format
|
|
||||||
|
|
||||||
Only used with xml.dom.minidom, which is likely only to be used
|
|
||||||
with python versions older than 2.5
|
|
||||||
"""
|
|
||||||
elem = dom.getElementsByTagName(tagName)[0]
|
|
||||||
return dict(list(elem.attributes.items()))
|
|
||||||
|
|
||||||
|
|
||||||
def getConfig():
|
|
||||||
"""Download the speedtest.net configuration and return only the data
|
|
||||||
we are interested in
|
|
||||||
"""
|
|
||||||
|
|
||||||
request = build_request('http://www.speedtest.net/speedtest-config.php')
|
|
||||||
uh = catch_request(request)
|
|
||||||
if uh is False:
|
|
||||||
print_('Could not retrieve speedtest.net configuration')
|
|
||||||
sys.exit(1)
|
|
||||||
configxml = []
|
|
||||||
while 1:
|
|
||||||
configxml.append(uh.read(10240))
|
|
||||||
if len(configxml[-1]) == 0:
|
|
||||||
break
|
|
||||||
if int(uh.code) != 200:
|
|
||||||
return None
|
|
||||||
uh.close()
|
|
||||||
try:
|
|
||||||
try:
|
|
||||||
root = ET.fromstring(''.encode().join(configxml))
|
|
||||||
config = {
|
|
||||||
'client': root.find('client').attrib,
|
|
||||||
'times': root.find('times').attrib,
|
|
||||||
'download': root.find('download').attrib,
|
|
||||||
'upload': root.find('upload').attrib}
|
|
||||||
except AttributeError: # Python3 branch
|
|
||||||
root = DOM.parseString(''.join(configxml))
|
|
||||||
config = {
|
|
||||||
'client': getAttributesByTagName(root, 'client'),
|
|
||||||
'times': getAttributesByTagName(root, 'times'),
|
|
||||||
'download': getAttributesByTagName(root, 'download'),
|
|
||||||
'upload': getAttributesByTagName(root, 'upload')}
|
|
||||||
except SyntaxError:
|
|
||||||
print_('Failed to parse speedtest.net configuration')
|
|
||||||
sys.exit(1)
|
|
||||||
del root
|
|
||||||
del configxml
|
|
||||||
return config
|
|
||||||
|
|
||||||
|
|
||||||
def closestServers(client, all=False):
|
|
||||||
"""Determine the 5 closest speedtest.net servers based on geographic
|
|
||||||
distance
|
|
||||||
"""
|
|
||||||
|
|
||||||
urls = [
|
|
||||||
'http://c.speedtest.net/speedtest-servers-static.php',
|
|
||||||
'http://www.speedtest.net/speedtest-servers-static.php',
|
|
||||||
]
|
|
||||||
servers = {}
|
|
||||||
for url in urls:
|
|
||||||
try:
|
|
||||||
request = build_request(url)
|
|
||||||
uh = catch_request(request)
|
|
||||||
if uh is False:
|
|
||||||
raise SpeedtestCliServerListError
|
|
||||||
serversxml = []
|
|
||||||
while 1:
|
|
||||||
serversxml.append(uh.read(10240))
|
|
||||||
if len(serversxml[-1]) == 0:
|
|
||||||
break
|
|
||||||
if int(uh.code) != 200:
|
|
||||||
uh.close()
|
|
||||||
raise SpeedtestCliServerListError
|
|
||||||
uh.close()
|
|
||||||
try:
|
|
||||||
try:
|
|
||||||
root = ET.fromstring(''.encode().join(serversxml))
|
|
||||||
elements = root.getiterator('server')
|
|
||||||
except AttributeError: # Python3 branch
|
|
||||||
root = DOM.parseString(''.join(serversxml))
|
|
||||||
elements = root.getElementsByTagName('server')
|
|
||||||
except SyntaxError:
|
|
||||||
raise SpeedtestCliServerListError
|
|
||||||
for server in elements:
|
|
||||||
try:
|
|
||||||
attrib = server.attrib
|
|
||||||
except AttributeError:
|
|
||||||
attrib = dict(list(server.attributes.items()))
|
|
||||||
d = distance([float(client['lat']),
|
|
||||||
float(client['lon'])],
|
|
||||||
[float(attrib.get('lat')),
|
|
||||||
float(attrib.get('lon'))])
|
|
||||||
attrib['d'] = d
|
|
||||||
if d not in servers:
|
|
||||||
servers[d] = [attrib]
|
|
||||||
else:
|
|
||||||
servers[d].append(attrib)
|
|
||||||
del root
|
|
||||||
del serversxml
|
|
||||||
del elements
|
|
||||||
except SpeedtestCliServerListError:
|
|
||||||
continue
|
|
||||||
|
|
||||||
if not servers:
|
|
||||||
print_('Failed to retrieve list of speedtest.net servers')
|
|
||||||
sys.exit(1)
|
|
||||||
|
|
||||||
closest = []
|
|
||||||
for d in sorted(servers.keys()):
|
|
||||||
for s in servers[d]:
|
|
||||||
closest.append(s)
|
|
||||||
if len(closest) == 5 and not all:
|
|
||||||
break
|
|
||||||
else:
|
|
||||||
continue
|
|
||||||
break
|
|
||||||
|
|
||||||
del servers
|
|
||||||
return closest
|
|
||||||
|
|
||||||
|
|
||||||
def getBestServer(servers):
|
|
||||||
"""Perform a speedtest.net latency request to determine which
|
|
||||||
speedtest.net server has the lowest latency
|
|
||||||
"""
|
|
||||||
|
|
||||||
results = {}
|
|
||||||
for server in servers:
|
|
||||||
cum = []
|
|
||||||
url = '%s/latency.txt' % os.path.dirname(server['url'])
|
|
||||||
urlparts = urlparse(url)
|
|
||||||
for i in range(0, 3):
|
|
||||||
try:
|
|
||||||
if urlparts[0] == 'https':
|
|
||||||
h = HTTPSConnection(urlparts[1])
|
|
||||||
else:
|
|
||||||
h = HTTPConnection(urlparts[1])
|
|
||||||
headers = {'User-Agent': user_agent}
|
|
||||||
start = timeit.default_timer()
|
|
||||||
h.request("GET", urlparts[2], headers=headers)
|
|
||||||
r = h.getresponse()
|
|
||||||
total = (timeit.default_timer() - start)
|
|
||||||
except (HTTPError, URLError, socket.error):
|
|
||||||
cum.append(3600)
|
|
||||||
continue
|
|
||||||
text = r.read(9)
|
|
||||||
if int(r.status) == 200 and text == 'test=test'.encode():
|
|
||||||
cum.append(total)
|
|
||||||
else:
|
|
||||||
cum.append(3600)
|
|
||||||
h.close()
|
|
||||||
avg = round((sum(cum) / 6) * 1000, 3)
|
|
||||||
results[avg] = server
|
|
||||||
fastest = sorted(results.keys())[0]
|
|
||||||
best = results[fastest]
|
|
||||||
best['latency'] = fastest
|
|
||||||
|
|
||||||
return best
|
|
||||||
|
|
||||||
|
|
||||||
def ctrl_c(signum, frame):
|
|
||||||
"""Catch Ctrl-C key sequence and set a shutdown_event for our threaded
|
|
||||||
operations
|
|
||||||
"""
|
|
||||||
|
|
||||||
global shutdown_event
|
|
||||||
shutdown_event.set()
|
|
||||||
raise SystemExit('\nCancelling...')
|
|
||||||
|
|
||||||
|
|
||||||
def version():
|
|
||||||
"""Print the version"""
|
|
||||||
|
|
||||||
raise SystemExit(__version__)
|
|
||||||
|
|
||||||
|
|
||||||
def speedtest():
|
|
||||||
"""Run the full speedtest.net test"""
|
|
||||||
|
|
||||||
global shutdown_event, source
|
|
||||||
shutdown_event = threading.Event()
|
|
||||||
|
|
||||||
signal.signal(signal.SIGINT, ctrl_c)
|
|
||||||
|
|
||||||
description = (
|
|
||||||
'Command line interface for testing internet bandwidth using '
|
|
||||||
'speedtest.net.\n'
|
|
||||||
'------------------------------------------------------------'
|
|
||||||
'--------------\n'
|
|
||||||
'https://github.com/sivel/speedtest-cli')
|
|
||||||
|
|
||||||
parser = ArgParser(description=description)
|
|
||||||
# Give optparse.OptionParser an `add_argument` method for
|
|
||||||
# compatibility with argparse.ArgumentParser
|
|
||||||
try:
|
|
||||||
parser.add_argument = parser.add_option
|
|
||||||
except AttributeError:
|
|
||||||
pass
|
|
||||||
parser.add_argument('--bytes', dest='units', action='store_const',
|
|
||||||
const=('byte', 1), default=('bit', 8),
|
|
||||||
help='Display values in bytes instead of bits. Does '
|
|
||||||
'not affect the image generated by --share')
|
|
||||||
parser.add_argument('--share', action='store_true',
|
|
||||||
help='Generate and provide a URL to the speedtest.net '
|
|
||||||
'share results image')
|
|
||||||
parser.add_argument('--simple', action='store_true',
|
|
||||||
help='Suppress verbose output, only show basic '
|
|
||||||
'information')
|
|
||||||
parser.add_argument('--list', action='store_true',
|
|
||||||
help='Display a list of speedtest.net servers '
|
|
||||||
'sorted by distance')
|
|
||||||
parser.add_argument('--server', help='Specify a server ID to test against')
|
|
||||||
parser.add_argument('--mini', help='URL of the Speedtest Mini server')
|
|
||||||
parser.add_argument('--source', help='Source IP address to bind to')
|
|
||||||
parser.add_argument('--timeout', default=10, type=int,
|
|
||||||
help='HTTP timeout in seconds. Default 10')
|
|
||||||
parser.add_argument('--version', action='store_true',
|
|
||||||
help='Show the version number and exit')
|
|
||||||
|
|
||||||
options = parser.parse_args()
|
|
||||||
if isinstance(options, tuple):
|
|
||||||
args = options[0]
|
|
||||||
else:
|
|
||||||
args = options
|
|
||||||
del options
|
|
||||||
|
|
||||||
# Print the version and exit
|
|
||||||
if args.version:
|
|
||||||
version()
|
|
||||||
|
|
||||||
socket.setdefaulttimeout(args.timeout)
|
|
||||||
|
|
||||||
# If specified bind to a specific IP address
|
|
||||||
if args.source:
|
|
||||||
source = args.source
|
|
||||||
socket.socket = bound_socket
|
|
||||||
|
|
||||||
if not args.simple:
|
|
||||||
print_('Retrieving speedtest.net configuration...')
|
|
||||||
try:
|
|
||||||
config = getConfig()
|
|
||||||
except URLError:
|
|
||||||
print_('Cannot retrieve speedtest configuration')
|
|
||||||
sys.exit(1)
|
|
||||||
|
|
||||||
if not args.simple:
|
|
||||||
print_('Retrieving speedtest.net server list...')
|
|
||||||
if args.list or args.server:
|
|
||||||
servers = closestServers(config['client'], True)
|
|
||||||
if args.list:
|
|
||||||
serverList = []
|
|
||||||
for server in servers:
|
|
||||||
line = ('%(id)4s) %(sponsor)s (%(name)s, %(country)s) '
|
|
||||||
'[%(d)0.2f km]' % server)
|
|
||||||
serverList.append(line)
|
|
||||||
# Python 2.7 and newer seem to be ok with the resultant encoding
|
|
||||||
# from parsing the XML, but older versions have some issues.
|
|
||||||
# This block should detect whether we need to encode or not
|
|
||||||
try:
|
|
||||||
unicode()
|
|
||||||
print_('\n'.join(serverList).encode('utf-8', 'ignore'))
|
|
||||||
except NameError:
|
|
||||||
print_('\n'.join(serverList))
|
|
||||||
except IOError:
|
|
||||||
pass
|
|
||||||
sys.exit(0)
|
|
||||||
else:
|
|
||||||
servers = closestServers(config['client'])
|
|
||||||
|
|
||||||
if not args.simple:
|
|
||||||
print_('Testing from %(isp)s (%(ip)s)...' % config['client'])
|
|
||||||
|
|
||||||
if args.server:
|
|
||||||
try:
|
|
||||||
best = getBestServer(filter(lambda x: x['id'] == args.server,
|
|
||||||
servers))
|
|
||||||
except IndexError:
|
|
||||||
print_('Invalid server ID')
|
|
||||||
sys.exit(1)
|
|
||||||
elif args.mini:
|
|
||||||
name, ext = os.path.splitext(args.mini)
|
|
||||||
if ext:
|
|
||||||
url = os.path.dirname(args.mini)
|
|
||||||
else:
|
|
||||||
url = args.mini
|
|
||||||
urlparts = urlparse(url)
|
|
||||||
try:
|
|
||||||
request = build_request(args.mini)
|
|
||||||
f = urlopen(request)
|
|
||||||
except:
|
|
||||||
print_('Invalid Speedtest Mini URL')
|
|
||||||
sys.exit(1)
|
|
||||||
else:
|
|
||||||
text = f.read()
|
|
||||||
f.close()
|
|
||||||
extension = re.findall('upload_extension: "([^"]+)"', text.decode())
|
|
||||||
if not extension:
|
|
||||||
for ext in ['php', 'asp', 'aspx', 'jsp']:
|
|
||||||
try:
|
|
||||||
request = build_request('%s/speedtest/upload.%s' %
|
|
||||||
(args.mini, ext))
|
|
||||||
f = urlopen(request)
|
|
||||||
except:
|
|
||||||
pass
|
|
||||||
else:
|
|
||||||
data = f.read().strip()
|
|
||||||
if (f.code == 200 and
|
|
||||||
len(data.splitlines()) == 1 and
|
|
||||||
re.match('size=[0-9]', data)):
|
|
||||||
extension = [ext]
|
|
||||||
break
|
|
||||||
if not urlparts or not extension:
|
|
||||||
print_('Please provide the full URL of your Speedtest Mini server')
|
|
||||||
sys.exit(1)
|
|
||||||
servers = [{
|
|
||||||
'sponsor': 'Speedtest Mini',
|
|
||||||
'name': urlparts[1],
|
|
||||||
'd': 0,
|
|
||||||
'url': '%s/speedtest/upload.%s' % (url.rstrip('/'), extension[0]),
|
|
||||||
'latency': 0,
|
|
||||||
'id': 0
|
|
||||||
}]
|
|
||||||
try:
|
|
||||||
best = getBestServer(servers)
|
|
||||||
except:
|
|
||||||
best = servers[0]
|
|
||||||
else:
|
|
||||||
if not args.simple:
|
|
||||||
print_('Selecting best server based on latency...')
|
|
||||||
best = getBestServer(servers)
|
|
||||||
|
|
||||||
if not args.simple:
|
|
||||||
# Python 2.7 and newer seem to be ok with the resultant encoding
|
|
||||||
# from parsing the XML, but older versions have some issues.
|
|
||||||
# This block should detect whether we need to encode or not
|
|
||||||
try:
|
|
||||||
unicode()
|
|
||||||
print_(('Hosted by %(sponsor)s (%(name)s) [%(d)0.2f km]: '
|
|
||||||
'%(latency)s ms' % best).encode('utf-8', 'ignore'))
|
|
||||||
except NameError:
|
|
||||||
print_('Hosted by %(sponsor)s (%(name)s) [%(d)0.2f km]: '
|
|
||||||
'%(latency)s ms' % best)
|
|
||||||
else:
|
|
||||||
print_('Ping: %(latency)s ms' % best)
|
|
||||||
|
|
||||||
sizes = [350, 500, 750, 1000, 1500, 2000, 2500, 3000, 3500, 4000]
|
|
||||||
urls = []
|
|
||||||
for size in sizes:
|
|
||||||
for i in range(0, 4):
|
|
||||||
urls.append('%s/random%sx%s.jpg' %
|
|
||||||
(os.path.dirname(best['url']), size, size))
|
|
||||||
if not args.simple:
|
|
||||||
print_('Testing download speed', end='')
|
|
||||||
dlspeed = downloadSpeed(urls, args.simple)
|
|
||||||
if not args.simple:
|
|
||||||
print_()
|
|
||||||
print_('Download: %0.2f M%s/s' %
|
|
||||||
((dlspeed / 1000 / 1000) * args.units[1], args.units[0]))
|
|
||||||
|
|
||||||
sizesizes = [int(.25 * 1000 * 1000), int(.5 * 1000 * 1000)]
|
|
||||||
sizes = []
|
|
||||||
for size in sizesizes:
|
|
||||||
for i in range(0, 25):
|
|
||||||
sizes.append(size)
|
|
||||||
if not args.simple:
|
|
||||||
print_('Testing upload speed', end='')
|
|
||||||
ulspeed = uploadSpeed(best['url'], sizes, args.simple)
|
|
||||||
if not args.simple:
|
|
||||||
print_()
|
|
||||||
print_('Upload: %0.2f M%s/s' %
|
|
||||||
((ulspeed / 1000 / 1000) * args.units[1], args.units[0]))
|
|
||||||
|
|
||||||
if args.share and args.mini:
|
|
||||||
print_('Cannot generate a speedtest.net share results image while '
|
|
||||||
'testing against a Speedtest Mini server')
|
|
||||||
elif args.share:
|
|
||||||
dlspeedk = int(round((dlspeed / 1000) * 8, 0))
|
|
||||||
ping = int(round(best['latency'], 0))
|
|
||||||
ulspeedk = int(round((ulspeed / 1000) * 8, 0))
|
|
||||||
|
|
||||||
# Build the request to send results back to speedtest.net
|
|
||||||
# We use a list instead of a dict because the API expects parameters
|
|
||||||
# in a certain order
|
|
||||||
apiData = [
|
|
||||||
'download=%s' % dlspeedk,
|
|
||||||
'ping=%s' % ping,
|
|
||||||
'upload=%s' % ulspeedk,
|
|
||||||
'promo=',
|
|
||||||
'startmode=%s' % 'pingselect',
|
|
||||||
'recommendedserverid=%s' % best['id'],
|
|
||||||
'accuracy=%s' % 1,
|
|
||||||
'serverid=%s' % best['id'],
|
|
||||||
'hash=%s' % md5(('%s-%s-%s-%s' %
|
|
||||||
(ping, ulspeedk, dlspeedk, '297aae72'))
|
|
||||||
.encode()).hexdigest()]
|
|
||||||
|
|
||||||
headers = {'Referer': 'http://c.speedtest.net/flash/speedtest.swf'}
|
|
||||||
request = build_request('http://www.speedtest.net/api/api.php',
|
|
||||||
data='&'.join(apiData).encode(),
|
|
||||||
headers=headers)
|
|
||||||
f = catch_request(request)
|
|
||||||
if f is False:
|
|
||||||
print_('Could not submit results to speedtest.net')
|
|
||||||
sys.exit(1)
|
|
||||||
response = f.read()
|
|
||||||
code = f.code
|
|
||||||
f.close()
|
|
||||||
|
|
||||||
if int(code) != 200:
|
|
||||||
print_('Could not submit results to speedtest.net')
|
|
||||||
sys.exit(1)
|
|
||||||
|
|
||||||
qsargs = parse_qs(response.decode())
|
|
||||||
resultid = qsargs.get('resultid')
|
|
||||||
if not resultid or len(resultid) != 1:
|
|
||||||
print_('Could not submit results to speedtest.net')
|
|
||||||
sys.exit(1)
|
|
||||||
|
|
||||||
print_('Share results: http://www.speedtest.net/result/%s.png' %
|
|
||||||
resultid[0])
|
|
||||||
|
|
||||||
|
|
||||||
def main():
|
|
||||||
try:
|
|
||||||
speedtest()
|
|
||||||
except KeyboardInterrupt:
|
|
||||||
print_('\nCancelling...')
|
|
||||||
|
|
||||||
|
|
||||||
if __name__ == '__main__':
|
|
||||||
main()
|
|
||||||
|
|
||||||
# vim:ts=4:sw=4:expandtab
|
|
|
@ -0,0 +1,37 @@
|
||||||
|
#!/usr/bin/env python
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
# Copyright 2018 Matt Martz
|
||||||
|
# All Rights Reserved.
|
||||||
|
#
|
||||||
|
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
||||||
|
# not use this file except in compliance with the License. You may obtain
|
||||||
|
# a copy of the License at
|
||||||
|
#
|
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
|
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
||||||
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||||
|
# License for the specific language governing permissions and limitations
|
||||||
|
# under the License.
|
||||||
|
|
||||||
|
import sys
|
||||||
|
import subprocess
|
||||||
|
|
||||||
|
cmd = [sys.executable, 'speedtest.py', '--source', '127.0.0.1']
|
||||||
|
|
||||||
|
p = subprocess.Popen(
|
||||||
|
cmd,
|
||||||
|
stdout=subprocess.PIPE,
|
||||||
|
stderr=subprocess.PIPE
|
||||||
|
)
|
||||||
|
|
||||||
|
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()
|
||||||
|
)
|
12
tox.ini
12
tox.ini
|
@ -4,16 +4,22 @@ skipsdist=true
|
||||||
[testenv]
|
[testenv]
|
||||||
commands =
|
commands =
|
||||||
{envpython} -V
|
{envpython} -V
|
||||||
{envpython} speedtest_cli.py
|
{envpython} -m compileall speedtest.py
|
||||||
|
{envpython} speedtest.py
|
||||||
|
{envpython} speedtest.py --source 172.17.0.1
|
||||||
|
{envpython} tests/scripts/source.py
|
||||||
|
|
||||||
[testenv:flake8]
|
[testenv:flake8]
|
||||||
basepython=python
|
basepython=python
|
||||||
deps=flake8
|
deps=flake8
|
||||||
commands =
|
commands =
|
||||||
{envpython} -V
|
{envpython} -V
|
||||||
flake8 speedtest_cli.py
|
flake8 speedtest.py
|
||||||
|
|
||||||
[testenv:pypy]
|
[testenv:pypy]
|
||||||
commands =
|
commands =
|
||||||
pypy -V
|
pypy -V
|
||||||
pypy speedtest_cli.py
|
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