Add list-ordering rule
This is based on the key-ordering rule and behaves the same with regards to locales. Closes #454pull/541/head
parent
06db2af9b0
commit
f014ea9942
@ -0,0 +1,133 @@
|
|||||||
|
# Copyright (C) 2023 Johannes F. Knauf and Kevin Wojniak
|
||||||
|
#
|
||||||
|
# This program is free software: you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU General Public License as published by
|
||||||
|
# the Free Software Foundation, either version 3 of the License, or
|
||||||
|
# (at your option) any later version.
|
||||||
|
#
|
||||||
|
# This program is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
# GNU General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU General Public License
|
||||||
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
import locale
|
||||||
|
|
||||||
|
from tests.common import RuleTestCase
|
||||||
|
|
||||||
|
|
||||||
|
class ListOrderingTestCase(RuleTestCase):
|
||||||
|
rule_id = 'list-ordering'
|
||||||
|
|
||||||
|
def test_disabled(self):
|
||||||
|
conf = 'list-ordering: disable'
|
||||||
|
self.check('---\n'
|
||||||
|
'- seconditem\n'
|
||||||
|
'- firstitem\n', conf)
|
||||||
|
self.check('---\n'
|
||||||
|
'[seconditem, firstitem]\n', conf)
|
||||||
|
self.check('---\n'
|
||||||
|
'- second\n'
|
||||||
|
'- at\n', conf)
|
||||||
|
|
||||||
|
def test_enabled(self):
|
||||||
|
conf = 'list-ordering: enable'
|
||||||
|
self.check('---\n'
|
||||||
|
'- seconditem\n'
|
||||||
|
'- firstitem\n', conf,
|
||||||
|
problem=(3, 3))
|
||||||
|
self.check('---\n'
|
||||||
|
'[seconditem, firstitem]\n', conf,
|
||||||
|
problem=(2, 14))
|
||||||
|
self.check('---\n'
|
||||||
|
'- second\n'
|
||||||
|
'- at\n', conf,
|
||||||
|
problem=(3, 3))
|
||||||
|
self.check('---\n'
|
||||||
|
'nested but OK:\n'
|
||||||
|
' - third: [second]\n'
|
||||||
|
' - first\n', conf)
|
||||||
|
self.check('---\n'
|
||||||
|
'nested failure:\n'
|
||||||
|
' - third\n'
|
||||||
|
' - first:\n'
|
||||||
|
' items:\n'
|
||||||
|
' - z\n'
|
||||||
|
' - a\n', conf,
|
||||||
|
problem=(7, 9))
|
||||||
|
|
||||||
|
def test_word_length(self):
|
||||||
|
conf = 'list-ordering: enable'
|
||||||
|
self.check('---\n'
|
||||||
|
'- a\n'
|
||||||
|
'- ab\n'
|
||||||
|
'- abc\n', conf)
|
||||||
|
self.check('---\n'
|
||||||
|
'- a\n'
|
||||||
|
'- abc\n'
|
||||||
|
'- ab\n', conf,
|
||||||
|
problem=(4, 3))
|
||||||
|
|
||||||
|
def test_case(self):
|
||||||
|
conf = 'list-ordering: enable'
|
||||||
|
self.check('---\n'
|
||||||
|
'- T-shirt\n'
|
||||||
|
'- T-shirts\n'
|
||||||
|
'- t-shirt\n'
|
||||||
|
'- t-shirts\n', conf)
|
||||||
|
self.check('---\n'
|
||||||
|
'- T-shirt\n'
|
||||||
|
'- t-shirt\n'
|
||||||
|
'- T-shirts\n'
|
||||||
|
'- t-shirts\n', conf,
|
||||||
|
problem=(4, 3))
|
||||||
|
|
||||||
|
def test_accents(self):
|
||||||
|
conf = 'list-ordering: enable'
|
||||||
|
self.check('---\n'
|
||||||
|
'- hair\n'
|
||||||
|
'- hais\n'
|
||||||
|
'- haïr\n'
|
||||||
|
'- haïssable\n', conf)
|
||||||
|
self.check('---\n'
|
||||||
|
'- haïr\n'
|
||||||
|
'- hais\n', conf,
|
||||||
|
problem=(3, 3))
|
||||||
|
|
||||||
|
def test_locale_case(self):
|
||||||
|
self.addCleanup(locale.setlocale, locale.LC_ALL, (None, None))
|
||||||
|
try:
|
||||||
|
locale.setlocale(locale.LC_ALL, 'en_US.UTF-8')
|
||||||
|
except locale.Error: # pragma: no cover
|
||||||
|
self.skipTest('locale en_US.UTF-8 not available')
|
||||||
|
conf = ('list-ordering: enable')
|
||||||
|
self.check('---\n'
|
||||||
|
'- t-shirt\n'
|
||||||
|
'- T-shirt\n'
|
||||||
|
'- t-shirts\n'
|
||||||
|
'- T-shirts\n', conf)
|
||||||
|
self.check('---\n'
|
||||||
|
'- t-shirt\n'
|
||||||
|
'- t-shirts\n'
|
||||||
|
'- T-shirt\n'
|
||||||
|
'- T-shirts\n', conf,
|
||||||
|
problem=(4, 3))
|
||||||
|
|
||||||
|
def test_locale_accents(self):
|
||||||
|
self.addCleanup(locale.setlocale, locale.LC_ALL, (None, None))
|
||||||
|
try:
|
||||||
|
locale.setlocale(locale.LC_ALL, 'en_US.UTF-8')
|
||||||
|
except locale.Error: # pragma: no cover
|
||||||
|
self.skipTest('locale en_US.UTF-8 not available')
|
||||||
|
conf = ('list-ordering: enable')
|
||||||
|
self.check('---\n'
|
||||||
|
'- hair\n'
|
||||||
|
'- haïr\n'
|
||||||
|
'- hais\n'
|
||||||
|
'- haïssable\n', conf)
|
||||||
|
self.check('---\n'
|
||||||
|
'- hais\n'
|
||||||
|
'- haïr\n', conf,
|
||||||
|
problem=(3, 3))
|
@ -0,0 +1,119 @@
|
|||||||
|
# Copyright (C) 2023 Johannes F. Knauf and Kevin Wojniak
|
||||||
|
#
|
||||||
|
# This program is free software: you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU General Public License as published by
|
||||||
|
# the Free Software Foundation, either version 3 of the License, or
|
||||||
|
# (at your option) any later version.
|
||||||
|
#
|
||||||
|
# This program is distributed in the hope that it will be useful,
|
||||||
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
# GNU General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU General Public License
|
||||||
|
# along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
"""
|
||||||
|
Use this rule to enforce alphabetical ordering of items in lists. The sorting
|
||||||
|
order uses the Unicode code point number as a default. As a result, the
|
||||||
|
ordering is case-sensitive and not accent-friendly (see examples below).
|
||||||
|
This can be changed by setting the global ``locale`` option. This allows one
|
||||||
|
to sort case and accents properly.
|
||||||
|
|
||||||
|
.. rubric:: Examples
|
||||||
|
|
||||||
|
#. With ``list-ordering: {}``
|
||||||
|
|
||||||
|
the following code snippets would **PASS**:
|
||||||
|
::
|
||||||
|
|
||||||
|
- key 1
|
||||||
|
- key 2
|
||||||
|
- key 3
|
||||||
|
|
||||||
|
- [a, b, c]
|
||||||
|
|
||||||
|
- T-shirt
|
||||||
|
- T-shirts
|
||||||
|
- t-shirt
|
||||||
|
- t-shirts
|
||||||
|
|
||||||
|
- hair
|
||||||
|
- hais
|
||||||
|
- haïr
|
||||||
|
- haïssable
|
||||||
|
|
||||||
|
the following code snippets would **FAIL**:
|
||||||
|
::
|
||||||
|
|
||||||
|
- key 2
|
||||||
|
- key 1
|
||||||
|
|
||||||
|
- [b, a]
|
||||||
|
|
||||||
|
- T-shirt
|
||||||
|
- t-shirt
|
||||||
|
- T-shirts
|
||||||
|
- t-shirts
|
||||||
|
|
||||||
|
- haïr
|
||||||
|
- hais
|
||||||
|
|
||||||
|
#. With global option ``locale: "en_US.UTF-8"`` and rule ``list-ordering: {}``
|
||||||
|
|
||||||
|
as opposed to before, the following code snippets would now **PASS**:
|
||||||
|
::
|
||||||
|
|
||||||
|
- t-shirt
|
||||||
|
- T-shirt
|
||||||
|
- t-shirts
|
||||||
|
- T-shirts
|
||||||
|
|
||||||
|
- hair
|
||||||
|
- haïr
|
||||||
|
- hais
|
||||||
|
- haïssable
|
||||||
|
"""
|
||||||
|
|
||||||
|
from locale import strcoll
|
||||||
|
|
||||||
|
import yaml
|
||||||
|
|
||||||
|
from yamllint.linter import LintProblem
|
||||||
|
|
||||||
|
|
||||||
|
ID = 'list-ordering'
|
||||||
|
TYPE = 'token'
|
||||||
|
|
||||||
|
MAP, SEQ = range(2)
|
||||||
|
|
||||||
|
|
||||||
|
class Parent:
|
||||||
|
def __init__(self, type):
|
||||||
|
self.type = type
|
||||||
|
self.items = []
|
||||||
|
|
||||||
|
|
||||||
|
def check(conf, token, prev, next, nextnext, context):
|
||||||
|
if 'stack' not in context:
|
||||||
|
context['stack'] = []
|
||||||
|
|
||||||
|
if isinstance(token, (yaml.BlockMappingStartToken,
|
||||||
|
yaml.FlowMappingStartToken)):
|
||||||
|
context['stack'].append(Parent(MAP))
|
||||||
|
elif isinstance(token, (yaml.BlockSequenceStartToken,
|
||||||
|
yaml.FlowSequenceStartToken)):
|
||||||
|
context['stack'].append(Parent(SEQ))
|
||||||
|
elif isinstance(token, (yaml.BlockEndToken,
|
||||||
|
yaml.FlowMappingEndToken,
|
||||||
|
yaml.FlowSequenceEndToken)):
|
||||||
|
context['stack'].pop()
|
||||||
|
elif isinstance(token, yaml.ScalarToken):
|
||||||
|
if len(context['stack']) > 0 and context['stack'][-1].type == SEQ:
|
||||||
|
if any(strcoll(token.value, item) < 0
|
||||||
|
for item in context['stack'][-1].items):
|
||||||
|
yield LintProblem(
|
||||||
|
token.start_mark.line + 1, token.start_mark.column + 1,
|
||||||
|
'wrong list item order "%s"' % token.value)
|
||||||
|
else:
|
||||||
|
context['stack'][-1].items.append(token.value)
|
Loading…
Reference in New Issue