| # Copyright (c) 2017 The WebRTC project authors. All Rights Reserved. | 
 | # | 
 | # Use of this source code is governed by a BSD-style license | 
 | # that can be found in the LICENSE file in the root of the source | 
 | # tree. An additional intellectual property rights grant can be found | 
 | # in the file PATENTS.  All contributing project authors may | 
 | # be found in the AUTHORS file in the root of the source tree. | 
 |  | 
 | # This file is inspired to [1]. | 
 | # [1] - https://cs.chromium.org/chromium/src/PRESUBMIT_test_mocks.py | 
 |  | 
 | import os.path | 
 | import re | 
 |  | 
 |  | 
 | class MockInputApi(object): | 
 |   """Mock class for the InputApi class. | 
 |  | 
 |   This class can be used for unittests for presubmit by initializing the files | 
 |   attribute as the list of changed files. | 
 |   """ | 
 |  | 
 |   def __init__(self): | 
 |     self.change = MockChange([], []) | 
 |     self.files = [] | 
 |     self.presubmit_local_path = os.path.dirname(__file__) | 
 |  | 
 |   def AffectedSourceFiles(self, file_filter=None): | 
 |     return self.AffectedFiles(file_filter=file_filter) | 
 |  | 
 |   def AffectedFiles(self, file_filter=None, include_deletes=False): | 
 |     # pylint: disable=unused-argument | 
 |     return self.files | 
 |  | 
 |   @classmethod | 
 |   def FilterSourceFile(cls, affected_file, white_list=(), black_list=()): | 
 |     # pylint: disable=unused-argument | 
 |     return True | 
 |  | 
 |   def PresubmitLocalPath(self): | 
 |     return self.presubmit_local_path | 
 |  | 
 |   def ReadFile(self, affected_file, mode='rU'): | 
 |     filename = affected_file.AbsoluteLocalPath() | 
 |     for f in self.files: | 
 |       if f.LocalPath() == filename: | 
 |         with open(filename, mode) as f: | 
 |           return f.read() | 
 |     # Otherwise, file is not in our mock API. | 
 |     raise IOError, "No such file or directory: '%s'" % filename | 
 |  | 
 |  | 
 | class MockOutputApi(object): | 
 |   """Mock class for the OutputApi class. | 
 |  | 
 |   An instance of this class can be passed to presubmit unittests for outputing | 
 |   various types of results. | 
 |   """ | 
 |  | 
 |   class PresubmitResult(object): | 
 |     def __init__(self, message, items=None, long_text=''): | 
 |       self.message = message | 
 |       self.items = items | 
 |       self.long_text = long_text | 
 |  | 
 |     def __repr__(self): | 
 |       return self.message | 
 |  | 
 |   class PresubmitError(PresubmitResult): | 
 |     def __init__(self, message, items=None, long_text=''): | 
 |       MockOutputApi.PresubmitResult.__init__(self, message, items, long_text) | 
 |       self.type = 'error' | 
 |  | 
 |  | 
 | class MockChange(object): | 
 |   """Mock class for Change class. | 
 |  | 
 |   This class can be used in presubmit unittests to mock the query of the | 
 |   current change. | 
 |   """ | 
 |  | 
 |   def __init__(self, changed_files, bugs_from_description, tags=None): | 
 |     self._changed_files = changed_files | 
 |     self._bugs_from_description = bugs_from_description | 
 |     self.tags = dict() if not tags else tags | 
 |  | 
 |   def BugsFromDescription(self): | 
 |     return self._bugs_from_description | 
 |  | 
 |   def __getattr__(self, attr): | 
 |     """Return tags directly as attributes on the object.""" | 
 |     if not re.match(r"^[A-Z_]*$", attr): | 
 |       raise AttributeError(self, attr) | 
 |     return self.tags.get(attr) | 
 |  | 
 |  | 
 | class MockFile(object): | 
 |   """Mock class for the File class. | 
 |  | 
 |   This class can be used to form the mock list of changed files in | 
 |   MockInputApi for presubmit unittests. | 
 |   """ | 
 |  | 
 |   def __init__(self, local_path, new_contents=None, old_contents=None, | 
 |       action='A'): | 
 |     if new_contents is None: | 
 |       new_contents = ["Data"] | 
 |     self._local_path = local_path | 
 |     self._new_contents = new_contents | 
 |     self._changed_contents = [(i + 1, l) for i, l in enumerate(new_contents)] | 
 |     self._action = action | 
 |     self._old_contents = old_contents | 
 |  | 
 |   def Action(self): | 
 |     return self._action | 
 |  | 
 |   def ChangedContents(self): | 
 |     return self._changed_contents | 
 |  | 
 |   def NewContents(self): | 
 |     return self._new_contents | 
 |  | 
 |   def LocalPath(self): | 
 |     return self._local_path | 
 |  | 
 |   def AbsoluteLocalPath(self): | 
 |     return self._local_path | 
 |  | 
 |   def OldContents(self): | 
 |     return self._old_contents |