setup.py 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191
  1. """A setuptools based setup module.
  2. """
  3. from __future__ import print_function
  4. import os
  5. import fnmatch
  6. import re
  7. import yaml
  8. # Always prefer setuptools over distutils
  9. from setuptools import setup, Command
  10. from setuptools_lint.setuptools_command import PylintCommand
  11. from six import string_types
  12. from yamllint.config import YamlLintConfig
  13. from yamllint.cli import Format
  14. from yamllint import linter
  15. def find_files(base_dir, exclude_dirs, include_dirs, file_regex):
  16. ''' find files matching file_regex '''
  17. found = []
  18. exclude_regex = ''
  19. include_regex = ''
  20. if exclude_dirs is not None:
  21. exclude_regex = r'|'.join([fnmatch.translate(x) for x in exclude_dirs]) or r'$.'
  22. if include_dirs is not None:
  23. include_regex = r'|'.join([fnmatch.translate(x) for x in include_dirs]) or r'$.'
  24. for root, dirs, files in os.walk(base_dir):
  25. if exclude_dirs is not None:
  26. # filter out excludes for dirs
  27. dirs[:] = [d for d in dirs if not re.match(exclude_regex, d)]
  28. if include_dirs is not None:
  29. # filter for includes for dirs
  30. dirs[:] = [d for d in dirs if re.match(include_regex, d)]
  31. matches = [os.path.join(root, f) for f in files if re.search(file_regex, f) is not None]
  32. found.extend(matches)
  33. return found
  34. class OpenShiftAnsibleYamlLint(Command):
  35. ''' Command to run yamllint '''
  36. description = "Run yamllint tests"
  37. user_options = [
  38. ('excludes=', 'e', 'directories to exclude'),
  39. ('config-file=', 'c', 'config file to use'),
  40. ('format=', 'f', 'format to use (standard, parsable)'),
  41. ]
  42. def initialize_options(self):
  43. ''' initialize_options '''
  44. # Reason: Defining these attributes as a part of initialize_options is
  45. # consistent with upstream usage
  46. # Status: permanently disabled
  47. # pylint: disable=attribute-defined-outside-init
  48. self.excludes = None
  49. self.config_file = None
  50. self.format = None
  51. def finalize_options(self):
  52. ''' finalize_options '''
  53. # Reason: These attributes are defined in initialize_options and this
  54. # usage is consistant with upstream usage
  55. # Status: permanently disabled
  56. # pylint: disable=attribute-defined-outside-init
  57. if isinstance(self.excludes, string_types):
  58. self.excludes = self.excludes.split(',')
  59. if self.format is None:
  60. self.format = 'standard'
  61. assert (self.format in ['standard', 'parsable']), (
  62. 'unknown format {0}.'.format(self.format))
  63. if self.config_file is None:
  64. self.config_file = '.yamllint'
  65. assert os.path.isfile(self.config_file), (
  66. 'yamllint config file {0} does not exist.'.format(self.config_file))
  67. def run(self):
  68. ''' run command '''
  69. if self.excludes is not None:
  70. print("Excludes:\n{0}".format(yaml.dump(self.excludes, default_flow_style=False)))
  71. config = YamlLintConfig(file=self.config_file)
  72. has_errors = False
  73. has_warnings = False
  74. if self.format == 'parsable':
  75. format_method = Format.parsable
  76. else:
  77. format_method = Format.standard_color
  78. for yaml_file in find_files(os.getcwd(), self.excludes, None, r'\.ya?ml$'):
  79. first = True
  80. with open(yaml_file, 'r') as contents:
  81. for problem in linter.run(contents, config):
  82. if first and self.format != 'parsable':
  83. print('\n{0}:'.format(os.path.relpath(yaml_file)))
  84. first = False
  85. print(format_method(problem, yaml_file))
  86. if problem.level == linter.PROBLEM_LEVELS[2]:
  87. has_errors = True
  88. elif problem.level == linter.PROBLEM_LEVELS[1]:
  89. has_warnings = True
  90. if has_errors or has_warnings:
  91. print('yammlint issues found')
  92. exit(1)
  93. class OpenShiftAnsiblePylint(PylintCommand):
  94. ''' Class to override the default behavior of PylintCommand '''
  95. # Reason: This method needs to be an instance method to conform to the
  96. # overridden method's signature
  97. # Status: permanently disabled
  98. # pylint: disable=no-self-use
  99. def find_all_modules(self):
  100. ''' find all python files to test '''
  101. exclude_dirs = ['.tox', 'utils', 'test', 'tests', 'git']
  102. modules = []
  103. for match in find_files(os.getcwd(), exclude_dirs, None, r'\.py$'):
  104. package = os.path.basename(match).replace('.py', '')
  105. modules.append(('openshift_ansible', package, match))
  106. return modules
  107. def get_finalized_command(self, cmd):
  108. ''' override get_finalized_command to ensure we use our
  109. find_all_modules method '''
  110. if cmd == 'build_py':
  111. return self
  112. # Reason: This method needs to be an instance method to conform to the
  113. # overridden method's signature
  114. # Status: permanently disabled
  115. # pylint: disable=no-self-use
  116. def with_project_on_sys_path(self, func, func_args, func_kwargs):
  117. ''' override behavior, since we don't need to build '''
  118. return func(*func_args, **func_kwargs)
  119. class UnsupportedCommand(Command):
  120. ''' Basic Command to override unsupported commands '''
  121. user_options = []
  122. # Reason: This method needs to be an instance method to conform to the
  123. # overridden method's signature
  124. # Status: permanently disabled
  125. # pylint: disable=no-self-use
  126. def initialize_options(self):
  127. ''' initialize_options '''
  128. pass
  129. # Reason: This method needs to be an instance method to conform to the
  130. # overridden method's signature
  131. # Status: permanently disabled
  132. # pylint: disable=no-self-use
  133. def finalize_options(self):
  134. ''' initialize_options '''
  135. pass
  136. # Reason: This method needs to be an instance method to conform to the
  137. # overridden method's signature
  138. # Status: permanently disabled
  139. # pylint: disable=no-self-use
  140. def run(self):
  141. ''' run command '''
  142. print("Unsupported command for openshift-ansible")
  143. setup(
  144. name='openshift-ansible',
  145. license="Apache 2.0",
  146. cmdclass={
  147. 'install': UnsupportedCommand,
  148. 'develop': UnsupportedCommand,
  149. 'build': UnsupportedCommand,
  150. 'build_py': UnsupportedCommand,
  151. 'build_ext': UnsupportedCommand,
  152. 'egg_info': UnsupportedCommand,
  153. 'sdist': UnsupportedCommand,
  154. 'lint': OpenShiftAnsiblePylint,
  155. 'yamllint': OpenShiftAnsibleYamlLint,
  156. },
  157. packages=[],
  158. )