diff --git a/bear-languages.yaml b/bear-languages.yaml index b8095b55a9..9144835d06 100644 --- a/bear-languages.yaml +++ b/bear-languages.yaml @@ -329,6 +329,7 @@ RadonBear: - Python - Python 2 - Python 3 +RegexLintBear: RuboCopBear: - Ruby RubyFastererBear: diff --git a/bear-requirements.txt b/bear-requirements.txt index 8d6a60a46f..30aafdbb7d 100644 --- a/bear-requirements.txt +++ b/bear-requirements.txt @@ -31,6 +31,7 @@ pylint~=1.7.2 pyroma~=2.2.0 pyyaml~=3.12 radon==1.4.0 +regexlint~=1.6 restructuredtext-lint~=1.0 rstcheck~=3.1 safety~=1.8.2 diff --git a/bear-requirements.yaml b/bear-requirements.yaml index d168d4535c..deda87c2e7 100644 --- a/bear-requirements.yaml +++ b/bear-requirements.yaml @@ -68,6 +68,8 @@ pip_requirements: version: ~=3.12 radon: version: ==1.4.0 + regexlint: + version: ~=1.6 restructuredtext-lint: version: ~=1.0 rstcheck: diff --git a/bears/general/RegexLintBear.py b/bears/general/RegexLintBear.py new file mode 100644 index 0000000000..c8f23c6f75 --- /dev/null +++ b/bears/general/RegexLintBear.py @@ -0,0 +1,46 @@ +import re + +from sarge import run, Capture + +from bears.general.AnnotationBear import AnnotationBear +from coalib.bears.LocalBear import LocalBear +from dependency_management.requirements.PipRequirement import PipRequirement + + +class RegexLintBear(LocalBear): + LANGUAGES = {'All'} + REQUIREMENTS = {PipRequirement('regexlint', '1.6')} + AUTHORS = {'The coala developers'} + AUTHORS_EMAILS = {'coala-devel@googlegroups.com'} + LICENSE = 'AGPL-3.0' + CAN_DETECT = {'Formatting'} + BEAR_DEPS = {AnnotationBear} + + def run(self, filename, file, dependency_results): + """ + Bear for linting regex through regexlint. + + :param dependency_results: + Results given by the AnnotationBear. + """ + annotation_dict = dependency_results[AnnotationBear.name][0].contents + for src_range in annotation_dict['strings']: + src_line = src_range.affected_source({filename: file})[0] + regex = src_line[src_range.start.column:src_range.end.column-1] + try: + re.compile(regex) + except re.error: + continue + out = run('regexlint --regex "{}"'.format(regex), + stdout=Capture()).stdout.text + if out[-3:-1] == 'OK': + continue + yield Result.from_values( + origin=self, + message=out, + file=filename, + line=src_range.start.line, + column=src_range.start.column, + end_line=src_range.end.line, + end_column=src_range.end.column, + ) diff --git a/tests/general/RegexLintBearTest.py b/tests/general/RegexLintBearTest.py new file mode 100644 index 0000000000..96bdb5ed15 --- /dev/null +++ b/tests/general/RegexLintBearTest.py @@ -0,0 +1,103 @@ +from os.path import abspath +from queue import Queue + +from bears.general.AnnotationBear import AnnotationBear +from bears.general.RegexLintBear import RegexLintBear +from coalib.results.Result import Result +from coalib.settings.Section import Section +from coalib.settings.Setting import Setting +from coalib.testing.LocalBearTestHelper import ( + LocalBearTestHelper, execute_bear) + + +good_py_file = """ +some_regex = r'[a-zA-Z]' +""" + +bad_py_file = """ +some_regex = r'(else|elseif)' +""" + +good_cpp_file = """ +char some_regex[] = "[a-zA-Z]"; +""" + +bad_cpp_file = """ +char some_regex[13] = "(else|elseif)"; +""" + +re_error_file = """ +some_regex = r'*ab' # This should be skipped +some_other_regex = r'[a-z]' +""" + +BAD_MESSAGE = """ +E105:argv:root:0: Potential out of order alternation between 'else' and 'elseif' + '(else|elseif)' + ^ here +""".lstrip() + + +class RegexLintBearTest(LocalBearTestHelper): + + def setUp(self): + self.section = Section('') + self.uut = RegexLintBear(self.section, Queue()) + self.dep_uut = AnnotationBear(self.section, Queue()) + + def test_good_python_file(self): + valid_file = good_py_file.splitlines() + self.section.append(Setting('language', 'python')) + dep_results = {AnnotationBear.name: + list(self.dep_uut.execute('file', valid_file))} + with execute_bear(self.uut, abspath('file'), valid_file, + dependency_results=dep_results) as results: + self.assertEqual(len(results), 0) + + def test_bad_python_file(self): + invalid_file = bad_py_file.splitlines() + self.section.append(Setting('language', 'python')) + dep_results = {AnnotationBear.name: + list(self.dep_uut.execute('file', invalid_file))} + with execute_bear(self.uut, abspath('file'), invalid_file, + dependency_results=dep_results) as results: + self.assertEqual(len(results), 1) + self.assertEqual(results[0], + Result.from_values(origin=self.uut, + message=BAD_MESSAGE, + file='file', + line=2, column=15, + end_line=2, end_column=29)) + + def test_good_cpp_file(self): + valid_file = good_cpp_file.splitlines() + self.section.append(Setting('language', 'cpp')) + dep_results = {AnnotationBear.name: + list(self.dep_uut.execute('file', valid_file))} + with execute_bear(self.uut, abspath('file'), valid_file, + dependency_results=dep_results) as results: + self.assertEqual(len(results), 0) + + def test_bad_cpp_file(self): + invalid_file = bad_cpp_file.splitlines() + self.section.append(Setting('language', 'cpp')) + dep_results = {AnnotationBear.name: + list(self.dep_uut.execute('file', invalid_file))} + with execute_bear(self.uut, abspath('file'), invalid_file, + dependency_results=dep_results) as results: + self.assertEqual(len(results), 1) + self.assertEqual(results[0], + Result.from_values(origin=self.uut, + message=BAD_MESSAGE, + file='file', + line=2, column=23, + end_line=2, end_column=37)) + + def test_re_error_file(self): + valid_file = re_error_file.splitlines() + self.section.append(Setting('language', 'python')) + dep_results = {AnnotationBear.name: + list(self.dep_uut.execute('file', valid_file))} + with execute_bear(self.uut, abspath('file'), valid_file, + dependency_results=dep_results) as results: + self.assertEqual(len(results), 0)