Skip to content

Commit 359c105

Browse files
Merge pull request #1693 from PyCQA/issue/1645
Issue/1645
2 parents 9042488 + cb896cd commit 359c105

File tree

6 files changed

+47
-3
lines changed

6 files changed

+47
-3
lines changed

CHANGELOG.md

+1
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,7 @@ Find out more about isort's release policy [here](https://pycqa.github.io/isort/
1515
- Implemented #1638 / #1644: Provide a flag `--overwrite-in-place` to ensure same file handle is used after sorting.
1616
- Implemented #1684: Added support for extending skips with `--extend-skip` and `--extend-skip-glob`.
1717
- Implemented #1688: Auto identification and skipping of some invalid import statements.
18+
- Implemented #1645: Ability to reverse the import sorting order.
1819
- Documented #1685: Skip doesn't support plain directory names, but skip_glob does.
1920

2021
### 5.7.0 December 30th 2020

isort/main.py

+6
Original file line numberDiff line numberDiff line change
@@ -562,6 +562,12 @@ def _build_arg_parser() -> argparse.ArgumentParser:
562562
action="store_true",
563563
help="Reverse order of relative imports.",
564564
)
565+
output_group.add_argument(
566+
"--reverse-sort",
567+
dest="reverse_sort",
568+
action="store_true",
569+
help="Reverses the ordering of imports.",
570+
)
565571
inline_args_group.add_argument(
566572
"--sl",
567573
"--force-single-line-imports",

isort/output.py

+5-1
Original file line numberDiff line numberDiff line change
@@ -54,12 +54,15 @@ def sorted_imports(
5454
key=lambda key: sorting.module_key(
5555
key, config, section_name=section, straight_import=True
5656
),
57+
reverse=config.reverse_sort,
5758
)
5859

5960
from_modules = parsed.imports[section]["from"]
6061
if not config.only_sections:
6162
from_modules = sorting.naturally(
62-
from_modules, key=lambda key: sorting.module_key(key, config, section_name=section)
63+
from_modules,
64+
key=lambda key: sorting.module_key(key, config, section_name=section),
65+
reverse=config.reverse_sort,
6366
)
6467

6568
straight_imports = _with_straight_imports(
@@ -233,6 +236,7 @@ def _with_from_imports(
233236
config.force_alphabetical_sort_within_sections,
234237
section_name=section,
235238
),
239+
reverse=config.reverse_sort,
236240
)
237241
if remove_imports:
238242
from_imports = [

isort/settings.py

+1
Original file line numberDiff line numberDiff line change
@@ -210,6 +210,7 @@ class _Config:
210210
honor_case_in_force_sorted_sections: bool = False
211211
sort_relative_in_force_sorted_sections: bool = False
212212
overwrite_in_place: bool = False
213+
reverse_sort: bool = False
213214

214215
def __post_init__(self):
215216
py_version = self.py_version

isort/sorting.py

+4-2
Original file line numberDiff line numberDiff line change
@@ -96,7 +96,9 @@ def section_key(line: str, config: Config) -> str:
9696
return f"{section}{len(line) if config.length_sort else ''}{line}"
9797

9898

99-
def naturally(to_sort: Iterable[str], key: Optional[Callable[[str], Any]] = None) -> List[str]:
99+
def naturally(
100+
to_sort: Iterable[str], key: Optional[Callable[[str], Any]] = None, reverse: bool = False
101+
) -> List[str]:
100102
"""Returns a naturally sorted list"""
101103
if key is None:
102104
key_callback = _natural_keys
@@ -105,7 +107,7 @@ def naturally(to_sort: Iterable[str], key: Optional[Callable[[str], Any]] = None
105107
def key_callback(text: str) -> List[Any]:
106108
return _natural_keys(key(text)) # type: ignore
107109

108-
return sorted(to_sort, key=key_callback)
110+
return sorted(to_sort, key=key_callback, reverse=reverse)
109111

110112

111113
def _atoi(text: str) -> Any:

tests/unit/test_ticketed_features.py

+30
Original file line numberDiff line numberDiff line change
@@ -1029,3 +1029,33 @@ def test_isort_auto_detects_and_ignores_invalid_from_imports_issue_1688():
10291029
from package3 import also_ok
10301030
"""
10311031
)
1032+
1033+
1034+
def test_isort_allows_reversing_sort_order_issue_1645():
1035+
"""isort allows reversing the sort order for those who prefer Z or longer imports first.
1036+
see: https://github.com/PyCQA/isort/issues/1688
1037+
"""
1038+
assert (
1039+
isort.code(
1040+
"""
1041+
from xxx import (
1042+
g,
1043+
hi,
1044+
def,
1045+
abcd,
1046+
)
1047+
""",
1048+
profile="black",
1049+
reverse_sort=True,
1050+
length_sort=True,
1051+
line_length=20,
1052+
)
1053+
== """
1054+
from xxx import (
1055+
abcd,
1056+
def,
1057+
hi,
1058+
g,
1059+
)
1060+
"""
1061+
)

0 commit comments

Comments
 (0)