Skip to content

Commit

Permalink
Merge pull request #254 from cas--/refactor/pretty-format-json
Browse files Browse the repository at this point in the history
Refactor/pretty format json
  • Loading branch information
asottile authored Dec 11, 2017
2 parents f3ff331 + 3e1b954 commit cf04ab0
Show file tree
Hide file tree
Showing 3 changed files with 31 additions and 43 deletions.
54 changes: 22 additions & 32 deletions pre_commit_hooks/pretty_format_json.py
Original file line number Diff line number Diff line change
Expand Up @@ -2,11 +2,11 @@

import argparse
import io
import json
import sys
from collections import OrderedDict

import simplejson
import six
from six import text_type


def _get_pretty_format(contents, indent, ensure_ascii=True, sort_keys=True, top_keys=[]):
Expand All @@ -17,40 +17,28 @@ def pairs_first(pairs):
if sort_keys:
after = sorted(after, key=lambda x: x[0])
return OrderedDict(before + after)
return six.text_type(simplejson.dumps(
simplejson.loads(
contents,
object_pairs_hook=pairs_first,
),
json_pretty = json.dumps(
json.loads(contents, object_pairs_hook=pairs_first),
indent=indent,
ensure_ascii=ensure_ascii,
)) + "\n" # dumps does not end with a newline
separators=(',', ': '), # Workaround for https://bugs.python.org/issue16333
)
# Ensure unicode (Py2) and add the newline that dumps does not end with.
return text_type(json_pretty) + '\n'


def _autofix(filename, new_contents):
print("Fixing file {}".format(filename))
print('Fixing file {}'.format(filename))
with io.open(filename, 'w', encoding='UTF-8') as f:
f.write(new_contents)


def parse_indent(s):
# type: (str) -> str
def parse_num_to_int(s):
"""Convert string numbers to int, leaving strings as is."""
try:
int_indentation_spec = int(s)
return int(s)
except ValueError:
if not s.strip():
return s
else:
raise ValueError(
'Non-whitespace JSON indentation delimiter supplied. ',
)
else:
if int_indentation_spec >= 0:
return int_indentation_spec * ' '
else:
raise ValueError(
'Negative integer supplied to construct JSON indentation delimiter. ',
)
return s


def parse_topkeys(s):
Expand All @@ -68,9 +56,12 @@ def pretty_format_json(argv=None):
)
parser.add_argument(
'--indent',
type=parse_indent,
default=' ',
help='String used as delimiter for one indentation level',
type=parse_num_to_int,
default='2',
help=(
'The number of indent spaces or a string to be used as delimiter'
' for indentation level e.g. 4 or "\t" (Default: 2)'
),
)
parser.add_argument(
'--no-ensure-ascii',
Expand Down Expand Up @@ -110,16 +101,15 @@ def pretty_format_json(argv=None):
)

if contents != pretty_contents:
print("File {} is not pretty-formatted".format(json_file))
print('File {} is not pretty-formatted'.format(json_file))

if args.autofix:
_autofix(json_file, pretty_contents)

status = 1

except simplejson.JSONDecodeError:
except ValueError:
print(
"Input File {} is not a valid JSON, consider using check-json"
'Input File {} is not a valid JSON, consider using check-json'
.format(json_file),
)
return 1
Expand Down
1 change: 0 additions & 1 deletion setup.py
Original file line number Diff line number Diff line change
Expand Up @@ -28,7 +28,6 @@
'flake8!=2.5.3',
'autopep8>=1.3',
'pyyaml',
'simplejson',
'six',
],
entry_points={
Expand Down
19 changes: 9 additions & 10 deletions tests/pretty_format_json_test.py
Original file line number Diff line number Diff line change
@@ -1,20 +1,18 @@
import shutil

import pytest
from six import PY2

from pre_commit_hooks.pretty_format_json import parse_indent
from pre_commit_hooks.pretty_format_json import parse_num_to_int
from pre_commit_hooks.pretty_format_json import pretty_format_json
from testing.util import get_resource_path


def test_parse_indent():
assert parse_indent('0') == ''
assert parse_indent('2') == ' '
assert parse_indent('\t') == '\t'
with pytest.raises(ValueError):
parse_indent('a')
with pytest.raises(ValueError):
parse_indent('-2')
def test_parse_num_to_int():
assert parse_num_to_int('0') == 0
assert parse_num_to_int('2') == 2
assert parse_num_to_int('\t') == '\t'
assert parse_num_to_int(' ') == ' '


@pytest.mark.parametrize(
Expand Down Expand Up @@ -43,6 +41,7 @@ def test_unsorted_pretty_format_json(filename, expected_retval):
assert ret == expected_retval


@pytest.mark.skipif(PY2, reason="Requires Python3")
@pytest.mark.parametrize(
('filename', 'expected_retval'), (
('not_pretty_formatted_json.json', 1),
Expand All @@ -52,7 +51,7 @@ def test_unsorted_pretty_format_json(filename, expected_retval):
('tab_pretty_formatted_json.json', 0),
),
)
def test_tab_pretty_format_json(filename, expected_retval):
def test_tab_pretty_format_json(filename, expected_retval): # pragma: no cover
ret = pretty_format_json(['--indent', '\t', get_resource_path(filename)])
assert ret == expected_retval

Expand Down

0 comments on commit cf04ab0

Please sign in to comment.