diff options
Diffstat (limited to 'googlemock')
| -rw-r--r-- | googlemock/docs/cheat_sheet.md | 34 | ||||
| -rw-r--r-- | googlemock/docs/pump_manual.md | 190 | ||||
| -rw-r--r-- | googlemock/include/gmock/gmock-actions.h | 59 | ||||
| -rw-r--r-- | googlemock/include/gmock/gmock-matchers.h | 7 | ||||
| -rw-r--r-- | googlemock/scripts/README.md | 5 | ||||
| -rwxr-xr-x | googlemock/scripts/generator/cpp/ast.py | 65 | ||||
| -rwxr-xr-x | googlemock/scripts/generator/cpp/gmock_class.py | 367 | ||||
| -rwxr-xr-x | googlemock/scripts/generator/cpp/gmock_class_test.py | 450 | ||||
| -rwxr-xr-x | googlemock/scripts/generator/cpp/keywords.py | 3 | ||||
| -rwxr-xr-x | googlemock/scripts/generator/cpp/tokenize.py | 3 | ||||
| -rwxr-xr-x | googlemock/scripts/generator/cpp/utils.py | 4 | ||||
| -rwxr-xr-x | googlemock/scripts/generator/gmock_gen.py | 1 | ||||
| -rwxr-xr-x | googlemock/scripts/gmock-config.in | 303 | ||||
| -rwxr-xr-x | googlemock/scripts/gmock_doctor.py | 640 | ||||
| -rwxr-xr-x | googlemock/scripts/pump.py | 855 | ||||
| -rwxr-xr-x | googlemock/scripts/upload.py | 1387 | ||||
| -rwxr-xr-x | googlemock/scripts/upload_gmock.py | 78 | ||||
| -rw-r--r-- | googlemock/src/gmock_main.cc | 9 | ||||
| -rw-r--r-- | googlemock/test/gmock-actions_test.cc | 62 | ||||
| -rwxr-xr-x | googlemock/test/pump_test.py | 182 | 
20 files changed, 1873 insertions, 2831 deletions
| diff --git a/googlemock/docs/cheat_sheet.md b/googlemock/docs/cheat_sheet.md index 3236e6a9..975362bf 100644 --- a/googlemock/docs/cheat_sheet.md +++ b/googlemock/docs/cheat_sheet.md @@ -503,16 +503,17 @@ which must be a permanent callback.  #### Returning a Value  <!-- mdformat off(no multiline tables) --> -|                             |                                               | -| :-------------------------- | :-------------------------------------------- | -| `Return()`                  | Return from a `void` mock function.           | -| `Return(value)`             | Return `value`. If the type of `value` is     different to the mock function's return type, `value` is converted to the latter type <i>at the time the expectation is set</i>, not when the action is executed. | -| `ReturnArg<N>()`            | Return the `N`-th (0-based) argument.         | -| `ReturnNew<T>(a1, ..., ak)` | Return `new T(a1, ..., ak)`; a different      object is created each time. | -| `ReturnNull()`              | Return a null pointer.                        | -| `ReturnPointee(ptr)`        | Return the value pointed to by `ptr`.         | -| `ReturnRef(variable)`       | Return a reference to `variable`.             | -| `ReturnRefOfCopy(value)`    | Return a reference to a copy of `value`; the  copy lives as long as the action. | +|                                   |                                               | +| :-------------------------------- | :-------------------------------------------- | +| `Return()`                        | Return from a `void` mock function.           | +| `Return(value)`                   | Return `value`. If the type of `value` is     different to the mock function's return type, `value` is converted to the latter type <i>at the time the expectation is set</i>, not when the action is executed. | +| `ReturnArg<N>()`                  | Return the `N`-th (0-based) argument.         | +| `ReturnNew<T>(a1, ..., ak)`       | Return `new T(a1, ..., ak)`; a different      object is created each time. | +| `ReturnNull()`                    | Return a null pointer.                        | +| `ReturnPointee(ptr)`              | Return the value pointed to by `ptr`.         | +| `ReturnRef(variable)`             | Return a reference to `variable`.             | +| `ReturnRefOfCopy(value)`          | Return a reference to a copy of `value`; the  copy lives as long as the action. | +| `ReturnRoundRobin({a1, ..., ak})` | Each call will return the next `ai` in the list, starting at the beginning when the end of the list is reached. |  <!-- mdformat on -->  #### Side Effects @@ -613,19 +614,6 @@ composite action - trying to do so will result in a run-time error.  #### Defining Actions -<table border="1" cellspacing="0" cellpadding="1"> -  <tr> -    <td>`struct SumAction {` <br> -         `template <typename T>` <br> -         `T operator()(T x, Ty) { return x + y; }` <br> -        `};` -    </td> -    <td> Defines a generic functor that can be used as an action summing its -    arguments. </td> </tr> -  <tr> -  </tr> -</table> -  <!-- mdformat off(no multiline tables) -->  |                                    |                                         |  | :--------------------------------- | :-------------------------------------- | diff --git a/googlemock/docs/pump_manual.md b/googlemock/docs/pump_manual.md new file mode 100644 index 00000000..10b3c5ff --- /dev/null +++ b/googlemock/docs/pump_manual.md @@ -0,0 +1,190 @@ +<b>P</b>ump is <b>U</b>seful for <b>M</b>eta <b>P</b>rogramming. + +# The Problem + +Template and macro libraries often need to define many classes, functions, or +macros that vary only (or almost only) in the number of arguments they take. +It's a lot of repetitive, mechanical, and error-prone work. + +Variadic templates and variadic macros can alleviate the problem. However, while +both are being considered by the C++ committee, neither is in the standard yet +or widely supported by compilers. Thus they are often not a good choice, +especially when your code needs to be portable. And their capabilities are still +limited. + +As a result, authors of such libraries often have to write scripts to generate +their implementation. However, our experience is that it's tedious to write such +scripts, which tend to reflect the structure of the generated code poorly and +are often hard to read and edit. For example, a small change needed in the +generated code may require some non-intuitive, non-trivial changes in the +script. This is especially painful when experimenting with the code. + +# Our Solution + +Pump (for Pump is Useful for Meta Programming, Pretty Useful for Meta +Programming, or Practical Utility for Meta Programming, whichever you prefer) is +a simple meta-programming tool for C++. The idea is that a programmer writes a +`foo.pump` file which contains C++ code plus meta code that manipulates the C++ +code. The meta code can handle iterations over a range, nested iterations, local +meta variable definitions, simple arithmetic, and conditional expressions. You +can view it as a small Domain-Specific Language. The meta language is designed +to be non-intrusive (s.t. it won't confuse Emacs' C++ mode, for example) and +concise, making Pump code intuitive and easy to maintain. + +## Highlights + +*   The implementation is in a single Python script and thus ultra portable: no +    build or installation is needed and it works cross platforms. +*   Pump tries to be smart with respect to +    [Google's style guide](https://github.com/google/styleguide): it breaks long +    lines (easy to have when they are generated) at acceptable places to fit +    within 80 columns and indent the continuation lines correctly. +*   The format is human-readable and more concise than XML. +*   The format works relatively well with Emacs' C++ mode. + +## Examples + +The following Pump code (where meta keywords start with `$`, `[[` and `]]` are +meta brackets, and `$$` starts a meta comment that ends with the line): + +``` +$var n = 3     $$ Defines a meta variable n. +$range i 0..n  $$ Declares the range of meta iterator i (inclusive). +$for i [[ +               $$ Meta loop. +// Foo$i does blah for $i-ary predicates. +$range j 1..i +template <size_t N $for j [[, typename A$j]]> +class Foo$i { +$if i == 0 [[ +  blah a; +]] $elif i <= 2 [[ +  blah b; +]] $else [[ +  blah c; +]] +}; + +]] +``` + +will be translated by the Pump compiler to: + +```cpp +// Foo0 does blah for 0-ary predicates. +template <size_t N> +class Foo0 { +  blah a; +}; + +// Foo1 does blah for 1-ary predicates. +template <size_t N, typename A1> +class Foo1 { +  blah b; +}; + +// Foo2 does blah for 2-ary predicates. +template <size_t N, typename A1, typename A2> +class Foo2 { +  blah b; +}; + +// Foo3 does blah for 3-ary predicates. +template <size_t N, typename A1, typename A2, typename A3> +class Foo3 { +  blah c; +}; +``` + +In another example, + +``` +$range i 1..n +Func($for i + [[a$i]]); +$$ The text between i and [[ is the separator between iterations. +``` + +will generate one of the following lines (without the comments), depending on +the value of `n`: + +```cpp +Func();              // If n is 0. +Func(a1);            // If n is 1. +Func(a1 + a2);       // If n is 2. +Func(a1 + a2 + a3);  // If n is 3. +// And so on... +``` + +## Constructs + +We support the following meta programming constructs: + +| `$var id = exp`                  | Defines a named constant value. `$id` is | +:                                  : valid util the end of the current meta   : +:                                  : lexical block.                           : +| :------------------------------- | :--------------------------------------- | +| `$range id exp..exp`             | Sets the range of an iteration variable, | +:                                  : which can be reused in multiple loops    : +:                                  : later.                                   : +| `$for id sep [[ code ]]`         | Iteration. The range of `id` must have   | +:                                  : been defined earlier. `$id` is valid in  : +:                                  : `code`.                                  : +| `$($)`                           | Generates a single `$` character.        | +| `$id`                            | Value of the named constant or iteration | +:                                  : variable.                                : +| `$(exp)`                         | Value of the expression.                 | +| `$if exp [[ code ]] else_branch` | Conditional.                             | +| `[[ code ]]`                     | Meta lexical block.                      | +| `cpp_code`                       | Raw C++ code.                            | +| `$$ comment`                     | Meta comment.                            | + +**Note:** To give the user some freedom in formatting the Pump source code, Pump +ignores a new-line character if it's right after `$for foo` or next to `[[` or +`]]`. Without this rule you'll often be forced to write very long lines to get +the desired output. Therefore sometimes you may need to insert an extra new-line +in such places for a new-line to show up in your output. + +## Grammar + +```ebnf +code ::= atomic_code* +atomic_code ::= $var id = exp +    | $var id = [[ code ]] +    | $range id exp..exp +    | $for id sep [[ code ]] +    | $($) +    | $id +    | $(exp) +    | $if exp [[ code ]] else_branch +    | [[ code ]] +    | cpp_code +sep ::= cpp_code | empty_string +else_branch ::= $else [[ code ]] +    | $elif exp [[ code ]] else_branch +    | empty_string +exp ::= simple_expression_in_Python_syntax +``` + +## Code + +You can find the source code of Pump in [scripts/pump.py](../scripts/pump.py). +It is still very unpolished and lacks automated tests, although it has been +successfully used many times. If you find a chance to use it in your project, +please let us know what you think! We also welcome help on improving Pump. + +## Real Examples + +You can find real-world applications of Pump in +[Google Test](https://github.com/google/googletest/tree/master/googletest) and +[Google Mock](https://github.com/google/googletest/tree/master/googlemock). The +source file `foo.h.pump` generates `foo.h`. + +## Tips + +*   If a meta variable is followed by a letter or digit, you can separate them +    using `[[]]`, which inserts an empty string. For example `Foo$j[[]]Helper` +    generate `Foo1Helper` when `j` is 1. +*   To avoid extra-long Pump source lines, you can break a line anywhere you +    want by inserting `[[]]` followed by a new line. Since any new-line +    character next to `[[` or `]]` is ignored, the generated code won't contain +    this new line. diff --git a/googlemock/include/gmock/gmock-actions.h b/googlemock/include/gmock/gmock-actions.h index f12d39be..b040004a 100644 --- a/googlemock/include/gmock/gmock-actions.h +++ b/googlemock/include/gmock/gmock-actions.h @@ -716,6 +716,36 @@ class ReturnRefOfCopyAction {    GTEST_DISALLOW_ASSIGN_(ReturnRefOfCopyAction);  }; +// Implements the polymorphic ReturnRoundRobin(v) action, which can be +// used in any function that returns the element_type of v. +template <typename T> +class ReturnRoundRobinAction { + public: +  explicit ReturnRoundRobinAction(std::vector<T> values) { +    GTEST_CHECK_(!values.empty()) +        << "ReturnRoundRobin requires at least one element."; +    state_->values = std::move(values); +  } + +  template <typename... Args> +  T operator()(Args&&...) const { +     return state_->Next(); +  } + + private: +  struct State { +    T Next() { +      T ret_val = values[i++]; +      if (i == values.size()) i = 0; +      return ret_val; +    } + +    std::vector<T> values; +    size_t i = 0; +  }; +  std::shared_ptr<State> state_ = std::make_shared<State>(); +}; +  // Implements the polymorphic DoDefault() action.  class DoDefaultAction {   public: @@ -1022,6 +1052,10 @@ inline internal::ReturnRefAction<R> ReturnRef(R& x) {  // NOLINT    return internal::ReturnRefAction<R>(x);  } +// Prevent using ReturnRef on reference to temporary. +template <typename R, R* = nullptr> +internal::ReturnRefAction<R> ReturnRef(R&&) = delete; +  // Creates an action that returns the reference to a copy of the  // argument.  The copy is created when the action is constructed and  // lives as long as the action. @@ -1039,6 +1073,23 @@ internal::ByMoveWrapper<R> ByMove(R x) {    return internal::ByMoveWrapper<R>(std::move(x));  } +// Creates an action that returns an element of `vals`. Calling this action will +// repeatedly return the next value from `vals` until it reaches the end and +// will restart from the beginning. +template <typename T> +internal::ReturnRoundRobinAction<T> ReturnRoundRobin(std::vector<T> vals) { +  return internal::ReturnRoundRobinAction<T>(std::move(vals)); +} + +// Creates an action that returns an element of `vals`. Calling this action will +// repeatedly return the next value from `vals` until it reaches the end and +// will restart from the beginning. +template <typename T> +internal::ReturnRoundRobinAction<T> ReturnRoundRobin( +    std::initializer_list<T> vals) { +  return internal::ReturnRoundRobinAction<T>(std::vector<T>(vals)); +} +  // Creates an action that does the default action for the give mock function.  inline internal::DoDefaultAction DoDefault() {    return internal::DoDefaultAction(); @@ -1047,14 +1098,14 @@ inline internal::DoDefaultAction DoDefault() {  // Creates an action that sets the variable pointed by the N-th  // (0-based) function argument to 'value'.  template <size_t N, typename T> -internal::SetArgumentPointeeAction<N, T> SetArgPointee(T x) { -  return {std::move(x)}; +internal::SetArgumentPointeeAction<N, T> SetArgPointee(T value) { +  return {std::move(value)};  }  // The following version is DEPRECATED.  template <size_t N, typename T> -internal::SetArgumentPointeeAction<N, T> SetArgumentPointee(T x) { -  return {std::move(x)}; +internal::SetArgumentPointeeAction<N, T> SetArgumentPointee(T value) { +  return {std::move(value)};  }  // Creates an action that sets a pointer referent to a given value. diff --git a/googlemock/include/gmock/gmock-matchers.h b/googlemock/include/gmock/gmock-matchers.h index bb047da9..b8ec24dd 100644 --- a/googlemock/include/gmock/gmock-matchers.h +++ b/googlemock/include/gmock/gmock-matchers.h @@ -758,8 +758,7 @@ class HasSubstrMatcher {    template <typename MatcheeStringType>    bool MatchAndExplain(const MatcheeStringType& s,                         MatchResultListener* /* listener */) const { -    const StringType& s2(s); -    return s2.find(substring_) != StringType::npos; +    return StringType(s).find(substring_) != StringType::npos;    }    // Describes what this matcher matches. @@ -3028,12 +3027,14 @@ class UnorderedElementsAreMatcherImpl      element_printouts->clear();      ::std::vector<char> did_match;      size_t num_elements = 0; +    DummyMatchResultListener dummy;      for (; elem_first != elem_last; ++num_elements, ++elem_first) {        if (listener->IsInterested()) {          element_printouts->push_back(PrintToString(*elem_first));        }        for (size_t irhs = 0; irhs != matchers_.size(); ++irhs) { -        did_match.push_back(Matches(matchers_[irhs])(*elem_first)); +        did_match.push_back( +            matchers_[irhs].MatchAndExplain(*elem_first, &dummy));        }      } diff --git a/googlemock/scripts/README.md b/googlemock/scripts/README.md new file mode 100644 index 00000000..fa359fed --- /dev/null +++ b/googlemock/scripts/README.md @@ -0,0 +1,5 @@ +# Please Note: + +Files in this directory are no longer supported by the maintainers. They +represent mosty historical artifacts and supported by the community only. There +is no guarantee whatsoever that these scripts still work. diff --git a/googlemock/scripts/generator/cpp/ast.py b/googlemock/scripts/generator/cpp/ast.py index f14728b4..f6331d80 100755 --- a/googlemock/scripts/generator/cpp/ast.py +++ b/googlemock/scripts/generator/cpp/ast.py @@ -17,10 +17,7 @@  """Generate an Abstract Syntax Tree (AST) for C++.""" -__author__ = 'nnorwitz@google.com (Neal Norwitz)' - - -# TODO: +# FIXME:  #  * Tokens should never be exported, need to convert to Nodes  #    (return types, parameters, etc.)  #  * Handle static class data for templatized classes @@ -338,7 +335,7 @@ class Class(_GenericDeclaration):          # TODO(nnorwitz): handle namespaces, etc.          if self.bases:              for token_list in self.bases: -                # TODO(nnorwitz): bases are tokens, do name comparison. +                # TODO(nnorwitz): bases are tokens, do name comparision.                  for token in token_list:                      if token.name == node.name:                          return True @@ -381,7 +378,7 @@ class Function(_GenericDeclaration):      def Requires(self, node):          if self.parameters: -            # TODO(nnorwitz): parameters are tokens, do name comparison. +            # TODO(nnorwitz): parameters are tokens, do name comparision.              for p in self.parameters:                  if p.name == node.name:                      return True @@ -521,7 +518,7 @@ class TypeConverter(object):              elif token.name == '&':                  reference = True              elif token.name == '[': -               pointer = True +                pointer = True              elif token.name == ']':                  pass              else: @@ -579,7 +576,7 @@ class TypeConverter(object):              elif p.name not in ('*', '&', '>'):                  # Ensure that names have a space between them.                  if (type_name and type_name[-1].token_type == tokenize.NAME and -                    p.token_type == tokenize.NAME): +                        p.token_type == tokenize.NAME):                      type_name.append(tokenize.Token(tokenize.SYNTAX, ' ', 0, 0))                  type_name.append(p)              else: @@ -655,7 +652,7 @@ class TypeConverter(object):          start = return_type_seq[0].start          end = return_type_seq[-1].end          _, name, templated_types, modifiers, default, other_tokens = \ -           self.DeclarationToParts(return_type_seq, False) +            self.DeclarationToParts(return_type_seq, False)          names = [n.name for n in other_tokens]          reference = '&' in names          pointer = '*' in names @@ -739,6 +736,14 @@ class AstBuilder(object):          if token.token_type == tokenize.NAME:              if (keywords.IsKeyword(token.name) and                  not keywords.IsBuiltinType(token.name)): +                if token.name == 'enum': +                    # Pop the next token and only put it back if it's not +                    # 'class'.  This allows us to support the two-token +                    # 'enum class' keyword as if it were simply 'enum'. +                    next = self._GetNextToken() +                    if next.name != 'class': +                        self._AddBackToken(next) +                  method = getattr(self, 'handle_' + token.name)                  return method()              elif token.name == self.in_class_name_only: @@ -754,7 +759,8 @@ class AstBuilder(object):              # Handle data or function declaration/definition.              syntax = tokenize.SYNTAX              temp_tokens, last_token = \ -                self._GetVarTokensUpTo(syntax, '(', ';', '{', '[') +                self._GetVarTokensUpToIgnoringTemplates(syntax, +                                                        '(', ';', '{', '[')              temp_tokens.insert(0, token)              if last_token.name == '(':                  # If there is an assignment before the paren, @@ -810,7 +816,7 @@ class AstBuilder(object):                  # self.in_class can contain A::Name, but the dtor will only                  # be Name.  Make sure to compare against the right value.                  if (token.token_type == tokenize.NAME and -                    token.name == self.in_class_name_only): +                        token.name == self.in_class_name_only):                      return self._GetMethod([token], FUNCTION_DTOR, None, True)              # TODO(nnorwitz): handle a lot more syntax.          elif token.token_type == tokenize.PREPROCESSOR: @@ -858,7 +864,25 @@ class AstBuilder(object):              last_token = self._GetNextToken()          return tokens, last_token -    # TODO(nnorwitz): remove _IgnoreUpTo() it shouldn't be necessary. +    # Same as _GetVarTokensUpTo, but skips over '<...>' which could contain an +    # expected token. +    def _GetVarTokensUpToIgnoringTemplates(self, expected_token_type, +                                           *expected_tokens): +        last_token = self._GetNextToken() +        tokens = [] +        nesting = 0 +        while (nesting > 0 or +               last_token.token_type != expected_token_type or +               last_token.name not in expected_tokens): +            tokens.append(last_token) +            last_token = self._GetNextToken() +            if last_token.name == '<': +                nesting += 1 +            elif last_token.name == '>': +                nesting -= 1 +        return tokens, last_token + +    # TODO(nnorwitz): remove _IgnoreUpTo() it shouldn't be necesary.      def _IgnoreUpTo(self, token_type, token):          unused_tokens = self._GetTokensUpTo(token_type, token) @@ -905,7 +929,10 @@ class AstBuilder(object):      def _GetNextToken(self):          if self.token_queue:              return self.token_queue.pop() -        return next(self.tokens) +        try: +            return next(self.tokens) +        except StopIteration: +            return      def _AddBackToken(self, token):          if token.whence == tokenize.WHENCE_STREAM: @@ -1105,7 +1132,7 @@ class AstBuilder(object):          # Looks like we got a method, not a function.          if len(return_type) > 2 and return_type[-1].name == '::':              return_type, in_class = \ -                         self._GetReturnTypeAndClassName(return_type) +                self._GetReturnTypeAndClassName(return_type)              return Method(indices.start, indices.end, name.name, in_class,                            return_type, parameters, modifiers, templated_types,                            body, self.namespace_stack) @@ -1264,9 +1291,6 @@ class AstBuilder(object):          return self._GetNestedType(Union)      def handle_enum(self): -        token = self._GetNextToken() -        if not (token.token_type == tokenize.NAME and token.name == 'class'): -            self._AddBackToken(token)          return self._GetNestedType(Enum)      def handle_auto(self): @@ -1298,7 +1322,8 @@ class AstBuilder(object):          if token2.token_type == tokenize.SYNTAX and token2.name == '~':              return self.GetMethod(FUNCTION_VIRTUAL + FUNCTION_DTOR, None)          assert token.token_type == tokenize.NAME or token.name == '::', token -        return_type_and_name = self._GetTokensUpTo(tokenize.SYNTAX, '(')  # ) +        return_type_and_name, _ = self._GetVarTokensUpToIgnoringTemplates( +            tokenize.SYNTAX, '(')  # )          return_type_and_name.insert(0, token)          if token2 is not token:              return_type_and_name.insert(1, token2) @@ -1352,7 +1377,7 @@ class AstBuilder(object):      def handle_typedef(self):          token = self._GetNextToken()          if (token.token_type == tokenize.NAME and -            keywords.IsKeyword(token.name)): +                keywords.IsKeyword(token.name)):              # Token must be struct/enum/union/class.              method = getattr(self, 'handle_' + token.name)              self._handling_typedef = True @@ -1375,7 +1400,7 @@ class AstBuilder(object):          if name.name == ')':              # HACK(nnorwitz): Handle pointers to functions "properly".              if (len(tokens) >= 4 and -                tokens[1].name == '(' and tokens[2].name == '*'): +                    tokens[1].name == '(' and tokens[2].name == '*'):                  tokens.append(name)                  name = tokens[3]          elif name.name == ']': diff --git a/googlemock/scripts/generator/cpp/gmock_class.py b/googlemock/scripts/generator/cpp/gmock_class.py index f9966cbb..89862ae1 100755 --- a/googlemock/scripts/generator/cpp/gmock_class.py +++ b/googlemock/scripts/generator/cpp/gmock_class.py @@ -26,9 +26,6 @@ Usage:  Output is sent to stdout.  """ -__author__ = 'nnorwitz@google.com (Neal Norwitz)' - -  import os  import re  import sys @@ -38,190 +35,214 @@ from cpp import utils  # Preserve compatibility with Python 2.3.  try: -  _dummy = set +    _dummy = set  except NameError: -  import sets -  set = sets.Set +    import sets + +    set = sets.Set  _VERSION = (1, 0, 1)  # The version of this script.  # How many spaces to indent.  Can set me with the INDENT environment variable.  _INDENT = 2 -def _GenerateMethods(output_lines, source, class_node): -  function_type = (ast.FUNCTION_VIRTUAL | ast.FUNCTION_PURE_VIRTUAL | -                   ast.FUNCTION_OVERRIDE) -  ctor_or_dtor = ast.FUNCTION_CTOR | ast.FUNCTION_DTOR -  indent = ' ' * _INDENT - -  for node in class_node.body: -    # We only care about virtual functions. -    if (isinstance(node, ast.Function) and -        node.modifiers & function_type and -        not node.modifiers & ctor_or_dtor): -      # Pick out all the elements we need from the original function. -      const = '' -      if node.modifiers & ast.FUNCTION_CONST: -        const = 'CONST_' -      return_type = 'void' -      if node.return_type: -        # Add modifiers like 'const'. -        modifiers = '' -        if node.return_type.modifiers: -          modifiers = ' '.join(node.return_type.modifiers) + ' ' -        return_type = modifiers + node.return_type.name -        template_args = [arg.name for arg in node.return_type.templated_types] -        if template_args: -          return_type += '<' + ', '.join(template_args) + '>' -          if len(template_args) > 1: -            for line in [ -                '// The following line won\'t really compile, as the return', -                '// type has multiple template arguments.  To fix it, use a', -                '// typedef for the return type.']: -              output_lines.append(indent + line) -        if node.return_type.pointer: -          return_type += '*' -        if node.return_type.reference: -          return_type += '&' -        num_parameters = len(node.parameters) -        if len(node.parameters) == 1: -          first_param = node.parameters[0] -          if source[first_param.start:first_param.end].strip() == 'void': +def _RenderType(ast_type): +    """Renders the potentially recursively templated type into a string. + +  Args: +    ast_type: The AST of the type. + +  Returns: +    Rendered string and a boolean to indicate whether we have multiple args +    (which is not handled correctly). +  """ +    has_multiarg_error = False +    # Add modifiers like 'const'. +    modifiers = '' +    if ast_type.modifiers: +        modifiers = ' '.join(ast_type.modifiers) + ' ' +    return_type = modifiers + ast_type.name +    if ast_type.templated_types: +        # Collect template args. +        template_args = [] +        for arg in ast_type.templated_types: +            rendered_arg, e = _RenderType(arg) +            if e: has_multiarg_error = True +            template_args.append(rendered_arg) +        return_type += '<' + ', '.join(template_args) + '>' +        # We are actually not handling multi-template-args correctly. So mark it. +        if len(template_args) > 1: +            has_multiarg_error = True +    if ast_type.pointer: +        return_type += '*' +    if ast_type.reference: +        return_type += '&' +    return return_type, has_multiarg_error + + +def _GetNumParameters(parameters, source): +    num_parameters = len(parameters) +    if num_parameters == 1: +        first_param = parameters[0] +        if source[first_param.start:first_param.end].strip() == 'void':              # We must treat T(void) as a function with no parameters. -            num_parameters = 0 -      tmpl = '' -      if class_node.templated_types: -        tmpl = '_T' -      mock_method_macro = 'MOCK_%sMETHOD%d%s' % (const, num_parameters, tmpl) - -      args = '' -      if node.parameters: -        # Due to the parser limitations, it is impossible to keep comments -        # while stripping the default parameters.  When defaults are -        # present, we choose to strip them and comments (and produce -        # compilable code). -        # TODO(nnorwitz@google.com): Investigate whether it is possible to -        # preserve parameter name when reconstructing parameter text from -        # the AST. -        if len([param for param in node.parameters if param.default]) > 0: -          args = ', '.join(param.type.name for param in node.parameters) -        else: -          # Get the full text of the parameters from the start -          # of the first parameter to the end of the last parameter. -          start = node.parameters[0].start -          end = node.parameters[-1].end -          # Remove // comments. -          args_strings = re.sub(r'//.*', '', source[start:end]) -          # Condense multiple spaces and eliminate newlines putting the -          # parameters together on a single line.  Ensure there is a -          # space in an argument which is split by a newline without -          # intervening whitespace, e.g.: int\nBar -          args = re.sub('  +', ' ', args_strings.replace('\n', ' ')) - -      # Create the mock method definition. -      output_lines.extend(['%s%s(%s,' % (indent, mock_method_macro, node.name), -                           '%s%s(%s));' % (indent*3, return_type, args)]) +            return 0 +    return num_parameters + + +def _GenerateMethods(output_lines, source, class_node): +    function_type = (ast.FUNCTION_VIRTUAL | ast.FUNCTION_PURE_VIRTUAL | +                     ast.FUNCTION_OVERRIDE) +    ctor_or_dtor = ast.FUNCTION_CTOR | ast.FUNCTION_DTOR +    indent = ' ' * _INDENT + +    for node in class_node.body: +        # We only care about virtual functions. +        if (isinstance(node, ast.Function) and +                node.modifiers & function_type and +                not node.modifiers & ctor_or_dtor): +            # Pick out all the elements we need from the original function. +            const = '' +            if node.modifiers & ast.FUNCTION_CONST: +                const = 'CONST_' +            num_parameters = _GetNumParameters(node.parameters, source) +            return_type = 'void' +            if node.return_type: +                return_type, has_multiarg_error = _RenderType(node.return_type) +                if has_multiarg_error: +                    for line in [ +                        '// The following line won\'t really compile, as the return', +                        '// type has multiple template arguments.  To fix it, use a', +                        '// typedef for the return type.']: +                        output_lines.append(indent + line) +            tmpl = '' +            if class_node.templated_types: +                tmpl = '_T' +            mock_method_macro = 'MOCK_%sMETHOD%d%s' % (const, num_parameters, tmpl) + +            args = '' +            if node.parameters: +                # Get the full text of the parameters from the start +                # of the first parameter to the end of the last parameter. +                start = node.parameters[0].start +                end = node.parameters[-1].end +                # Remove // comments. +                args_strings = re.sub(r'//.*', '', source[start:end]) +                # Remove /* comments */. +                args_strings = re.sub(r'/\*.*\*/', '', args_strings) +                # Remove default arguments. +                args_strings = re.sub(r'=.*,', ',', args_strings) +                args_strings = re.sub(r'=.*', '', args_strings) +                # Condense multiple spaces and eliminate newlines putting the +                # parameters together on a single line.  Ensure there is a +                # space in an argument which is split by a newline without +                # intervening whitespace, e.g.: int\nBar +                args = re.sub('  +', ' ', args_strings.replace('\n', ' ')) + +            # Create the mock method definition. +            output_lines.extend(['%s%s(%s,' % (indent, mock_method_macro, node.name), +                                 '%s%s(%s));' % (indent * 3, return_type, args)])  def _GenerateMocks(filename, source, ast_list, desired_class_names): -  processed_class_names = set() -  lines = [] -  for node in ast_list: -    if (isinstance(node, ast.Class) and node.body and -        # desired_class_names being None means that all classes are selected. -        (not desired_class_names or node.name in desired_class_names)): -      class_name = node.name -      parent_name = class_name -      processed_class_names.add(class_name) -      class_node = node -      # Add namespace before the class. -      if class_node.namespace: -        lines.extend(['namespace %s {' % n for n in class_node.namespace])  # } -        lines.append('') - -      # Add template args for templated classes. -      if class_node.templated_types: -        # TODO(paulchang): The AST doesn't preserve template argument order, -        # so we have to make up names here. -        # TODO(paulchang): Handle non-type template arguments (e.g. -        # template<typename T, int N>). -        template_arg_count = len(class_node.templated_types.keys()) -        template_args = ['T%d' % n for n in range(template_arg_count)] -        template_decls = ['typename ' + arg for arg in template_args] -        lines.append('template <' + ', '.join(template_decls) + '>') -        parent_name += '<' + ', '.join(template_args) + '>' - -      # Add the class prolog. -      lines.append('class Mock%s : public %s {'  # } -                   % (class_name, parent_name)) -      lines.append('%spublic:' % (' ' * (_INDENT // 2))) - -      # Add all the methods. -      _GenerateMethods(lines, source, class_node) - -      # Close the class. -      if lines: -        # If there are no virtual methods, no need for a public label. -        if len(lines) == 2: -          del lines[-1] - -        # Only close the class if there really is a class. -        lines.append('};') -        lines.append('')  # Add an extra newline. - -      # Close the namespace. -      if class_node.namespace: -        for i in range(len(class_node.namespace)-1, -1, -1): -          lines.append('}  // namespace %s' % class_node.namespace[i]) -        lines.append('')  # Add an extra newline. - -  if desired_class_names: -    missing_class_name_list = list(desired_class_names - processed_class_names) -    if missing_class_name_list: -      missing_class_name_list.sort() -      sys.stderr.write('Class(es) not found in %s: %s\n' % -                       (filename, ', '.join(missing_class_name_list))) -  elif not processed_class_names: -    sys.stderr.write('No class found in %s\n' % filename) - -  return lines +    processed_class_names = set() +    lines = [] +    for node in ast_list: +        if (isinstance(node, ast.Class) and node.body and +                # desired_class_names being None means that all classes are selected. +                (not desired_class_names or node.name in desired_class_names)): +            class_name = node.name +            parent_name = class_name +            processed_class_names.add(class_name) +            class_node = node +            # Add namespace before the class. +            if class_node.namespace: +                lines.extend(['namespace %s {' % n for n in class_node.namespace])  # } +                lines.append('') + +            # Add template args for templated classes. +            if class_node.templated_types: +                # TODO(paulchang): The AST doesn't preserve template argument order, +                # so we have to make up names here. +                # TODO(paulchang): Handle non-type template arguments (e.g. +                # template<typename T, int N>). +                template_arg_count = len(class_node.templated_types.keys()) +                template_args = ['T%d' % n for n in range(template_arg_count)] +                template_decls = ['typename ' + arg for arg in template_args] +                lines.append('template <' + ', '.join(template_decls) + '>') +                parent_name += '<' + ', '.join(template_args) + '>' + +            # Add the class prolog. +            lines.append('class Mock%s : public %s {'  # } +                         % (class_name, parent_name)) +            lines.append('%spublic:' % (' ' * (_INDENT // 2))) + +            # Add all the methods. +            _GenerateMethods(lines, source, class_node) + +            # Close the class. +            if lines: +                # If there are no virtual methods, no need for a public label. +                if len(lines) == 2: +                    del lines[-1] + +                # Only close the class if there really is a class. +                lines.append('};') +                lines.append('')  # Add an extra newline. + +            # Close the namespace. +            if class_node.namespace: +                for i in range(len(class_node.namespace) - 1, -1, -1): +                    lines.append('}  // namespace %s' % class_node.namespace[i]) +                lines.append('')  # Add an extra newline. + +    if desired_class_names: +        missing_class_name_list = list(desired_class_names - processed_class_names) +        if missing_class_name_list: +            missing_class_name_list.sort() +            sys.stderr.write('Class(es) not found in %s: %s\n' % +                             (filename, ', '.join(missing_class_name_list))) +    elif not processed_class_names: +        sys.stderr.write('No class found in %s\n' % filename) + +    return lines  def main(argv=sys.argv): -  if len(argv) < 2: -    sys.stderr.write('Google Mock Class Generator v%s\n\n' % -                     '.'.join(map(str, _VERSION))) -    sys.stderr.write(__doc__) -    return 1 - -  global _INDENT -  try: -    _INDENT = int(os.environ['INDENT']) -  except KeyError: -    pass -  except: -    sys.stderr.write('Unable to use indent of %s\n' % os.environ.get('INDENT')) - -  filename = argv[1] -  desired_class_names = None  # None means all classes in the source file. -  if len(argv) >= 3: -    desired_class_names = set(argv[2:]) -  source = utils.ReadFile(filename) -  if source is None: -    return 1 - -  builder = ast.BuilderFromSource(source, filename) -  try: -    entire_ast = filter(None, builder.Generate()) -  except KeyboardInterrupt: -    return -  except: -    # An error message was already printed since we couldn't parse. -    sys.exit(1) -  else: -    lines = _GenerateMocks(filename, source, entire_ast, desired_class_names) -    sys.stdout.write('\n'.join(lines)) +    if len(argv) < 2: +        sys.stderr.write('Google Mock Class Generator v%s\n\n' % +                         '.'.join(map(str, _VERSION))) +        sys.stderr.write(__doc__) +        return 1 + +    global _INDENT +    try: +        _INDENT = int(os.environ['INDENT']) +    except KeyError: +        pass +    except: +        sys.stderr.write('Unable to use indent of %s\n' % os.environ.get('INDENT')) + +    filename = argv[1] +    desired_class_names = None  # None means all classes in the source file. +    if len(argv) >= 3: +        desired_class_names = set(argv[2:]) +    source = utils.ReadFile(filename) +    if source is None: +        return 1 + +    builder = ast.BuilderFromSource(source, filename) +    try: +        entire_ast = filter(None, builder.Generate()) +    except KeyboardInterrupt: +        return +    except: +        # An error message was already printed since we couldn't parse. +        sys.exit(1) +    else: +        lines = _GenerateMocks(filename, source, entire_ast, desired_class_names) +        sys.stdout.write('\n'.join(lines))  if __name__ == '__main__': -  main(sys.argv) +    main(sys.argv) diff --git a/googlemock/scripts/generator/cpp/gmock_class_test.py b/googlemock/scripts/generator/cpp/gmock_class_test.py index c53e6000..211a92dd 100755 --- a/googlemock/scripts/generator/cpp/gmock_class_test.py +++ b/googlemock/scripts/generator/cpp/gmock_class_test.py @@ -17,9 +17,6 @@  """Tests for gmock.scripts.generator.cpp.gmock_class.""" -__author__ = 'nnorwitz@google.com (Neal Norwitz)' - -  import os  import sys  import unittest @@ -32,41 +29,43 @@ from cpp import gmock_class  class TestCase(unittest.TestCase): -  """Helper class that adds assert methods.""" +    """Helper class that adds assert methods.""" -  def StripLeadingWhitespace(self, lines): -    """Strip leading whitespace in each line in 'lines'.""" -    return '\n'.join([s.lstrip() for s in lines.split('\n')]) +    @staticmethod +    def StripLeadingWhitespace(lines): +        """Strip leading whitespace in each line in 'lines'.""" +        return '\n'.join([s.lstrip() for s in lines.split('\n')]) -  def assertEqualIgnoreLeadingWhitespace(self, expected_lines, lines): -    """Specialized assert that ignores the indent level.""" -    self.assertEqual(expected_lines, self.StripLeadingWhitespace(lines)) +    def assertEqualIgnoreLeadingWhitespace(self, expected_lines, lines): +        """Specialized assert that ignores the indent level.""" +        self.assertEqual(expected_lines, self.StripLeadingWhitespace(lines))  class GenerateMethodsTest(TestCase): -  def GenerateMethodSource(self, cpp_source): -    """Convert C++ source to Google Mock output source lines.""" -    method_source_lines = [] -    # <test> is a pseudo-filename, it is not read or written. -    builder = ast.BuilderFromSource(cpp_source, '<test>') -    ast_list = list(builder.Generate()) -    gmock_class._GenerateMethods(method_source_lines, cpp_source, ast_list[0]) -    return '\n'.join(method_source_lines) - -  def testSimpleMethod(self): -    source = """ +    @staticmethod +    def GenerateMethodSource(cpp_source): +        """Convert C++ source to Google Mock output source lines.""" +        method_source_lines = [] +        # <test> is a pseudo-filename, it is not read or written. +        builder = ast.BuilderFromSource(cpp_source, '<test>') +        ast_list = list(builder.Generate()) +        gmock_class._GenerateMethods(method_source_lines, cpp_source, ast_list[0]) +        return '\n'.join(method_source_lines) + +    def testSimpleMethod(self): +        source = """  class Foo {   public:    virtual int Bar();  };  """ -    self.assertEqualIgnoreLeadingWhitespace( -        'MOCK_METHOD0(Bar,\nint());', -        self.GenerateMethodSource(source)) +        self.assertEqualIgnoreLeadingWhitespace( +            'MOCK_METHOD0(Bar,\nint());', +            self.GenerateMethodSource(source)) -  def testSimpleConstructorsAndDestructor(self): -    source = """ +    def testSimpleConstructorsAndDestructor(self): +        source = """  class Foo {   public:    Foo(); @@ -77,26 +76,26 @@ class Foo {    virtual int Bar() = 0;  };  """ -    # The constructors and destructor should be ignored. -    self.assertEqualIgnoreLeadingWhitespace( -        'MOCK_METHOD0(Bar,\nint());', -        self.GenerateMethodSource(source)) +        # The constructors and destructor should be ignored. +        self.assertEqualIgnoreLeadingWhitespace( +            'MOCK_METHOD0(Bar,\nint());', +            self.GenerateMethodSource(source)) -  def testVirtualDestructor(self): -    source = """ +    def testVirtualDestructor(self): +        source = """  class Foo {   public:    virtual ~Foo();    virtual int Bar() = 0;  };  """ -    # The destructor should be ignored. -    self.assertEqualIgnoreLeadingWhitespace( -        'MOCK_METHOD0(Bar,\nint());', -        self.GenerateMethodSource(source)) +        # The destructor should be ignored. +        self.assertEqualIgnoreLeadingWhitespace( +            'MOCK_METHOD0(Bar,\nint());', +            self.GenerateMethodSource(source)) -  def testExplicitlyDefaultedConstructorsAndDestructor(self): -    source = """ +    def testExplicitlyDefaultedConstructorsAndDestructor(self): +        source = """  class Foo {   public:    Foo() = default; @@ -106,13 +105,13 @@ class Foo {    virtual int Bar() = 0;  };  """ -    # The constructors and destructor should be ignored. -    self.assertEqualIgnoreLeadingWhitespace( -        'MOCK_METHOD0(Bar,\nint());', -        self.GenerateMethodSource(source)) +        # The constructors and destructor should be ignored. +        self.assertEqualIgnoreLeadingWhitespace( +            'MOCK_METHOD0(Bar,\nint());', +            self.GenerateMethodSource(source)) -  def testExplicitlyDeletedConstructorsAndDestructor(self): -    source = """ +    def testExplicitlyDeletedConstructorsAndDestructor(self): +        source = """  class Foo {   public:    Foo() = delete; @@ -122,92 +121,121 @@ class Foo {    virtual int Bar() = 0;  };  """ -    # The constructors and destructor should be ignored. -    self.assertEqualIgnoreLeadingWhitespace( -        'MOCK_METHOD0(Bar,\nint());', -        self.GenerateMethodSource(source)) +        # The constructors and destructor should be ignored. +        self.assertEqualIgnoreLeadingWhitespace( +            'MOCK_METHOD0(Bar,\nint());', +            self.GenerateMethodSource(source)) -  def testSimpleOverrideMethod(self): -    source = """ +    def testSimpleOverrideMethod(self): +        source = """  class Foo {   public:    int Bar() override;  };  """ -    self.assertEqualIgnoreLeadingWhitespace( -        'MOCK_METHOD0(Bar,\nint());', -        self.GenerateMethodSource(source)) +        self.assertEqualIgnoreLeadingWhitespace( +            'MOCK_METHOD0(Bar,\nint());', +            self.GenerateMethodSource(source)) -  def testSimpleConstMethod(self): -    source = """ +    def testSimpleConstMethod(self): +        source = """  class Foo {   public:    virtual void Bar(bool flag) const;  };  """ -    self.assertEqualIgnoreLeadingWhitespace( -        'MOCK_CONST_METHOD1(Bar,\nvoid(bool flag));', -        self.GenerateMethodSource(source)) +        self.assertEqualIgnoreLeadingWhitespace( +            'MOCK_CONST_METHOD1(Bar,\nvoid(bool flag));', +            self.GenerateMethodSource(source)) -  def testExplicitVoid(self): -    source = """ +    def testExplicitVoid(self): +        source = """  class Foo {   public:    virtual int Bar(void);  };  """ -    self.assertEqualIgnoreLeadingWhitespace( -        'MOCK_METHOD0(Bar,\nint(void));', -        self.GenerateMethodSource(source)) +        self.assertEqualIgnoreLeadingWhitespace( +            'MOCK_METHOD0(Bar,\nint(void));', +            self.GenerateMethodSource(source)) -  def testStrangeNewlineInParameter(self): -    source = """ +    def testStrangeNewlineInParameter(self): +        source = """  class Foo {   public:    virtual void Bar(int  a) = 0;  };  """ -    self.assertEqualIgnoreLeadingWhitespace( -        'MOCK_METHOD1(Bar,\nvoid(int a));', -        self.GenerateMethodSource(source)) +        self.assertEqualIgnoreLeadingWhitespace( +            'MOCK_METHOD1(Bar,\nvoid(int a));', +            self.GenerateMethodSource(source)) -  def testDefaultParameters(self): -    source = """ +    def testDefaultParameters(self): +        source = """  class Foo {   public:    virtual void Bar(int a, char c = 'x') = 0;  };  """ -    self.assertEqualIgnoreLeadingWhitespace( -        'MOCK_METHOD2(Bar,\nvoid(int, char));', -        self.GenerateMethodSource(source)) +        self.assertEqualIgnoreLeadingWhitespace( +            'MOCK_METHOD2(Bar,\nvoid(int a, char c ));', +            self.GenerateMethodSource(source)) -  def testMultipleDefaultParameters(self): -    source = """ +    def testMultipleDefaultParameters(self): +        source = """  class Foo {   public: -  virtual void Bar(int a = 42, char c = 'x') = 0; +  virtual void Bar( +        int a = 42,  +        char c = 'x',  +        const int* const p = nullptr,  +        const std::string& s = "42", +        char tab[] = {'4','2'}, +        int const *& rp = aDefaultPointer) = 0; +}; +""" +        self.assertEqualIgnoreLeadingWhitespace( +            "MOCK_METHOD7(Bar,\n" +            "void(int a , char c , const int* const p , const std::string& s , char tab[] , int const *& rp ));", +            self.GenerateMethodSource(source)) + +    def testConstDefaultParameter(self): +        source = """ +class Test { + public: +  virtual bool Bar(const int test_arg = 42) = 0; +}; +""" +        expected = 'MOCK_METHOD1(Bar,\nbool(const int test_arg ));' +        self.assertEqualIgnoreLeadingWhitespace( +            expected, self.GenerateMethodSource(source)) + +    def testConstRefDefaultParameter(self): +        source = """ +class Test { + public: +  virtual bool Bar(const std::string& test_arg = "42" ) = 0;  };  """ -    self.assertEqualIgnoreLeadingWhitespace( -        'MOCK_METHOD2(Bar,\nvoid(int, char));', -        self.GenerateMethodSource(source)) +        expected = 'MOCK_METHOD1(Bar,\nbool(const std::string& test_arg ));' +        self.assertEqualIgnoreLeadingWhitespace( +            expected, self.GenerateMethodSource(source)) -  def testRemovesCommentsWhenDefaultsArePresent(self): -    source = """ +    def testRemovesCommentsWhenDefaultsArePresent(self): +        source = """  class Foo {   public:    virtual void Bar(int a = 42 /* a comment */,                     char /* other comment */ c= 'x') = 0;  };  """ -    self.assertEqualIgnoreLeadingWhitespace( -        'MOCK_METHOD2(Bar,\nvoid(int, char));', -        self.GenerateMethodSource(source)) +        self.assertEqualIgnoreLeadingWhitespace( +            'MOCK_METHOD2(Bar,\nvoid(int a , char c));', +            self.GenerateMethodSource(source)) -  def testDoubleSlashCommentsInParameterListAreRemoved(self): -    source = """ +    def testDoubleSlashCommentsInParameterListAreRemoved(self): +        source = """  class Foo {   public:    virtual void Bar(int a,  // inline comments should be elided. @@ -215,116 +243,117 @@ class Foo {                     ) const = 0;  };  """ -    self.assertEqualIgnoreLeadingWhitespace( -        'MOCK_CONST_METHOD2(Bar,\nvoid(int a, int b));', -        self.GenerateMethodSource(source)) +        self.assertEqualIgnoreLeadingWhitespace( +            'MOCK_CONST_METHOD2(Bar,\nvoid(int a, int b));', +            self.GenerateMethodSource(source)) -  def testCStyleCommentsInParameterListAreNotRemoved(self): -    # NOTE(nnorwitz): I'm not sure if it's the best behavior to keep these -    # comments.  Also note that C style comments after the last parameter -    # are still elided. -    source = """ +    def testCStyleCommentsInParameterListAreNotRemoved(self): +        # NOTE(nnorwitz): I'm not sure if it's the best behavior to keep these +        # comments.  Also note that C style comments after the last parameter +        # are still elided. +        source = """  class Foo {   public:    virtual const string& Bar(int /* keeper */, int b);  };  """ -    self.assertEqualIgnoreLeadingWhitespace( -        'MOCK_METHOD2(Bar,\nconst string&(int /* keeper */, int b));', -        self.GenerateMethodSource(source)) +        self.assertEqualIgnoreLeadingWhitespace( +            'MOCK_METHOD2(Bar,\nconst string&(int , int b));', +            self.GenerateMethodSource(source)) -  def testArgsOfTemplateTypes(self): -    source = """ +    def testArgsOfTemplateTypes(self): +        source = """  class Foo {   public:    virtual int Bar(const vector<int>& v, map<int, string>* output);  };""" -    self.assertEqualIgnoreLeadingWhitespace( -        'MOCK_METHOD2(Bar,\n' -        'int(const vector<int>& v, map<int, string>* output));', -        self.GenerateMethodSource(source)) +        self.assertEqualIgnoreLeadingWhitespace( +            'MOCK_METHOD2(Bar,\n' +            'int(const vector<int>& v, map<int, string>* output));', +            self.GenerateMethodSource(source)) -  def testReturnTypeWithOneTemplateArg(self): -    source = """ +    def testReturnTypeWithOneTemplateArg(self): +        source = """  class Foo {   public:    virtual vector<int>* Bar(int n);  };""" -    self.assertEqualIgnoreLeadingWhitespace( -        'MOCK_METHOD1(Bar,\nvector<int>*(int n));', -        self.GenerateMethodSource(source)) +        self.assertEqualIgnoreLeadingWhitespace( +            'MOCK_METHOD1(Bar,\nvector<int>*(int n));', +            self.GenerateMethodSource(source)) -  def testReturnTypeWithManyTemplateArgs(self): -    source = """ +    def testReturnTypeWithManyTemplateArgs(self): +        source = """  class Foo {   public:    virtual map<int, string> Bar();  };""" -    # Comparing the comment text is brittle - we'll think of something -    # better in case this gets annoying, but for now let's keep it simple. -    self.assertEqualIgnoreLeadingWhitespace( -        '// The following line won\'t really compile, as the return\n' -        '// type has multiple template arguments.  To fix it, use a\n' -        '// typedef for the return type.\n' -        'MOCK_METHOD0(Bar,\nmap<int, string>());', -        self.GenerateMethodSource(source)) - -  def testSimpleMethodInTemplatedClass(self): -    source = """ +        # Comparing the comment text is brittle - we'll think of something +        # better in case this gets annoying, but for now let's keep it simple. +        self.assertEqualIgnoreLeadingWhitespace( +            '// The following line won\'t really compile, as the return\n' +            '// type has multiple template arguments.  To fix it, use a\n' +            '// typedef for the return type.\n' +            'MOCK_METHOD0(Bar,\nmap<int, string>());', +            self.GenerateMethodSource(source)) + +    def testSimpleMethodInTemplatedClass(self): +        source = """  template<class T>  class Foo {   public:    virtual int Bar();  };  """ -    self.assertEqualIgnoreLeadingWhitespace( -        'MOCK_METHOD0_T(Bar,\nint());', -        self.GenerateMethodSource(source)) +        self.assertEqualIgnoreLeadingWhitespace( +            'MOCK_METHOD0_T(Bar,\nint());', +            self.GenerateMethodSource(source)) -  def testPointerArgWithoutNames(self): -    source = """ +    def testPointerArgWithoutNames(self): +        source = """  class Foo {    virtual int Bar(C*);  };  """ -    self.assertEqualIgnoreLeadingWhitespace( -        'MOCK_METHOD1(Bar,\nint(C*));', -        self.GenerateMethodSource(source)) +        self.assertEqualIgnoreLeadingWhitespace( +            'MOCK_METHOD1(Bar,\nint(C*));', +            self.GenerateMethodSource(source)) -  def testReferenceArgWithoutNames(self): -    source = """ +    def testReferenceArgWithoutNames(self): +        source = """  class Foo {    virtual int Bar(C&);  };  """ -    self.assertEqualIgnoreLeadingWhitespace( -        'MOCK_METHOD1(Bar,\nint(C&));', -        self.GenerateMethodSource(source)) +        self.assertEqualIgnoreLeadingWhitespace( +            'MOCK_METHOD1(Bar,\nint(C&));', +            self.GenerateMethodSource(source)) -  def testArrayArgWithoutNames(self): -    source = """ +    def testArrayArgWithoutNames(self): +        source = """  class Foo {    virtual int Bar(C[]);  };  """ -    self.assertEqualIgnoreLeadingWhitespace( -        'MOCK_METHOD1(Bar,\nint(C[]));', -        self.GenerateMethodSource(source)) +        self.assertEqualIgnoreLeadingWhitespace( +            'MOCK_METHOD1(Bar,\nint(C[]));', +            self.GenerateMethodSource(source))  class GenerateMocksTest(TestCase): -  def GenerateMocks(self, cpp_source): -    """Convert C++ source to complete Google Mock output source.""" -    # <test> is a pseudo-filename, it is not read or written. -    filename = '<test>' -    builder = ast.BuilderFromSource(cpp_source, filename) -    ast_list = list(builder.Generate()) -    lines = gmock_class._GenerateMocks(filename, cpp_source, ast_list, None) -    return '\n'.join(lines) - -  def testNamespaces(self): -    source = """ +    @staticmethod +    def GenerateMocks(cpp_source): +        """Convert C++ source to complete Google Mock output source.""" +        # <test> is a pseudo-filename, it is not read or written. +        filename = '<test>' +        builder = ast.BuilderFromSource(cpp_source, filename) +        ast_list = list(builder.Generate()) +        lines = gmock_class._GenerateMocks(filename, cpp_source, ast_list, None) +        return '\n'.join(lines) + +    def testNamespaces(self): +        source = """  namespace Foo {  namespace Bar { class Forward; }  namespace Baz { @@ -337,7 +366,7 @@ class Test {  }  // namespace Baz  }  // namespace Foo  """ -    expected = """\ +        expected = """\  namespace Foo {  namespace Baz { @@ -350,53 +379,53 @@ void());  }  // namespace Baz  }  // namespace Foo  """ -    self.assertEqualIgnoreLeadingWhitespace( -        expected, self.GenerateMocks(source)) +        self.assertEqualIgnoreLeadingWhitespace( +            expected, self.GenerateMocks(source)) -  def testClassWithStorageSpecifierMacro(self): -    source = """ +    def testClassWithStorageSpecifierMacro(self): +        source = """  class STORAGE_SPECIFIER Test {   public:    virtual void Foo();  };  """ -    expected = """\ +        expected = """\  class MockTest : public Test {  public:  MOCK_METHOD0(Foo,  void());  };  """ -    self.assertEqualIgnoreLeadingWhitespace( -        expected, self.GenerateMocks(source)) +        self.assertEqualIgnoreLeadingWhitespace( +            expected, self.GenerateMocks(source)) -  def testTemplatedForwardDeclaration(self): -    source = """ +    def testTemplatedForwardDeclaration(self): +        source = """  template <class T> class Forward;  // Forward declaration should be ignored.  class Test {   public:    virtual void Foo();  };  """ -    expected = """\ +        expected = """\  class MockTest : public Test {  public:  MOCK_METHOD0(Foo,  void());  };  """ -    self.assertEqualIgnoreLeadingWhitespace( -        expected, self.GenerateMocks(source)) +        self.assertEqualIgnoreLeadingWhitespace( +            expected, self.GenerateMocks(source)) -  def testTemplatedClass(self): -    source = """ +    def testTemplatedClass(self): +        source = """  template <typename S, typename T>  class Test {   public:    virtual void Foo();  };  """ -    expected = """\ +        expected = """\  template <typename T0, typename T1>  class MockTest : public Test<T0, T1> {  public: @@ -404,29 +433,29 @@ MOCK_METHOD0_T(Foo,  void());  };  """ -    self.assertEqualIgnoreLeadingWhitespace( -        expected, self.GenerateMocks(source)) +        self.assertEqualIgnoreLeadingWhitespace( +            expected, self.GenerateMocks(source)) -  def testTemplateInATemplateTypedef(self): -    source = """ +    def testTemplateInATemplateTypedef(self): +        source = """  class Test {   public:    typedef std::vector<std::list<int>> FooType;    virtual void Bar(const FooType& test_arg);  };  """ -    expected = """\ +        expected = """\  class MockTest : public Test {  public:  MOCK_METHOD1(Bar,  void(const FooType& test_arg));  };  """ -    self.assertEqualIgnoreLeadingWhitespace( -        expected, self.GenerateMocks(source)) +        self.assertEqualIgnoreLeadingWhitespace( +            expected, self.GenerateMocks(source)) -  def testTemplateInATemplateTypedefWithComma(self): -    source = """ +    def testTemplateInATemplateTypedefWithComma(self): +        source = """  class Test {   public:    typedef std::function<void( @@ -434,33 +463,78 @@ class Test {    virtual void Bar(const FooType& test_arg);  };  """ -    expected = """\ +        expected = """\  class MockTest : public Test {  public:  MOCK_METHOD1(Bar,  void(const FooType& test_arg));  };  """ -    self.assertEqualIgnoreLeadingWhitespace( -        expected, self.GenerateMocks(source)) +        self.assertEqualIgnoreLeadingWhitespace( +            expected, self.GenerateMocks(source)) -  def testEnumClass(self): -    source = """ +    def testEnumType(self): +        source = """  class Test {   public: -  enum class Baz { BAZINGA }; -  virtual void Bar(const FooType& test_arg); +  enum Bar { +    BAZ, QUX, QUUX, QUUUX +  }; +  virtual void Foo();  };  """ -    expected = """\ +        expected = """\  class MockTest : public Test {  public: -MOCK_METHOD1(Bar, -void(const FooType& test_arg)); +MOCK_METHOD0(Foo, +void());  };  """ -    self.assertEqualIgnoreLeadingWhitespace( -        expected, self.GenerateMocks(source)) +        self.assertEqualIgnoreLeadingWhitespace( +            expected, self.GenerateMocks(source)) + +    def testEnumClassType(self): +        source = """ +class Test { + public: +  enum class Bar { +    BAZ, QUX, QUUX, QUUUX +  }; +  virtual void Foo(); +}; +""" +        expected = """\ +class MockTest : public Test { +public: +MOCK_METHOD0(Foo, +void()); +}; +""" +        self.assertEqualIgnoreLeadingWhitespace( +            expected, self.GenerateMocks(source)) + +    def testStdFunction(self): +        source = """ +class Test { + public: +  Test(std::function<int(std::string)> foo) : foo_(foo) {} + +  virtual std::function<int(std::string)> foo(); + + private: +  std::function<int(std::string)> foo_; +}; +""" +        expected = """\ +class MockTest : public Test { +public: +MOCK_METHOD0(foo, +std::function<int (std::string)>()); +}; +""" +        self.assertEqualIgnoreLeadingWhitespace( +            expected, self.GenerateMocks(source)) +  if __name__ == '__main__': -  unittest.main() +    unittest.main() diff --git a/googlemock/scripts/generator/cpp/keywords.py b/googlemock/scripts/generator/cpp/keywords.py index f694450e..e4282714 100755 --- a/googlemock/scripts/generator/cpp/keywords.py +++ b/googlemock/scripts/generator/cpp/keywords.py @@ -17,9 +17,6 @@  """C++ keywords and helper utilities for determining keywords.""" -__author__ = 'nnorwitz@google.com (Neal Norwitz)' - -  try:      # Python 3.x      import builtins diff --git a/googlemock/scripts/generator/cpp/tokenize.py b/googlemock/scripts/generator/cpp/tokenize.py index 359d5562..a75edcb1 100755 --- a/googlemock/scripts/generator/cpp/tokenize.py +++ b/googlemock/scripts/generator/cpp/tokenize.py @@ -17,9 +17,6 @@  """Tokenize C++ source code.""" -__author__ = 'nnorwitz@google.com (Neal Norwitz)' - -  try:      # Python 3.x      import builtins diff --git a/googlemock/scripts/generator/cpp/utils.py b/googlemock/scripts/generator/cpp/utils.py index eab36eec..6f5fc097 100755 --- a/googlemock/scripts/generator/cpp/utils.py +++ b/googlemock/scripts/generator/cpp/utils.py @@ -17,12 +17,8 @@  """Generic utilities for C++ parsing.""" -__author__ = 'nnorwitz@google.com (Neal Norwitz)' - -  import sys -  # Set to True to see the start/end token indices.  DEBUG = True diff --git a/googlemock/scripts/generator/gmock_gen.py b/googlemock/scripts/generator/gmock_gen.py index 8cc0d135..9d528a56 100755 --- a/googlemock/scripts/generator/gmock_gen.py +++ b/googlemock/scripts/generator/gmock_gen.py @@ -16,7 +16,6 @@  """Driver for starting up Google Mock class generator.""" -__author__ = 'nnorwitz@google.com (Neal Norwitz)'  import os  import sys diff --git a/googlemock/scripts/gmock-config.in b/googlemock/scripts/gmock-config.in deleted file mode 100755 index 2baefe94..00000000 --- a/googlemock/scripts/gmock-config.in +++ /dev/null @@ -1,303 +0,0 @@ -#!/bin/sh - -# These variables are automatically filled in by the configure script. -name="@PACKAGE_TARNAME@" -version="@PACKAGE_VERSION@" - -show_usage() -{ -  echo "Usage: gmock-config [OPTIONS...]" -} - -show_help() -{ -  show_usage -  cat <<\EOF - -The `gmock-config' script provides access to the necessary compile and linking -flags to connect with Google C++ Mocking Framework, both in a build prior to -installation, and on the system proper after installation. The installation -overrides may be issued in combination with any other queries, but will only -affect installation queries if called on a built but not installed gmock. The -installation queries may not be issued with any other types of queries, and -only one installation query may be made at a time. The version queries and -compiler flag queries may be combined as desired but not mixed. Different -version queries are always combined with logical "and" semantics, and only the -last of any particular query is used while all previous ones ignored. All -versions must be specified as a sequence of numbers separated by periods. -Compiler flag queries output the union of the sets of flags when combined. - - Examples: -  gmock-config --min-version=1.0 || echo "Insufficient Google Mock version." - -  g++ $(gmock-config --cppflags --cxxflags) -o foo.o -c foo.cpp -  g++ $(gmock-config --ldflags --libs) -o foo foo.o - -  # When using a built but not installed Google Mock: -  g++ $(../../my_gmock_build/scripts/gmock-config ...) ... - -  # When using an installed Google Mock, but with installation overrides: -  export GMOCK_PREFIX="/opt" -  g++ $(gmock-config --libdir="/opt/lib64" ...) ... - - Help: -  --usage                    brief usage information -  --help                     display this help message - - Installation Overrides: -  --prefix=<dir>             overrides the installation prefix -  --exec-prefix=<dir>        overrides the executable installation prefix -  --libdir=<dir>             overrides the library installation prefix -  --includedir=<dir>         overrides the header file installation prefix - - Installation Queries: -  --prefix                   installation prefix -  --exec-prefix              executable installation prefix -  --libdir                   library installation directory -  --includedir               header file installation directory -  --version                  the version of the Google Mock installation - - Version Queries: -  --min-version=VERSION      return 0 if the version is at least VERSION -  --exact-version=VERSION    return 0 if the version is exactly VERSION -  --max-version=VERSION      return 0 if the version is at most VERSION - - Compilation Flag Queries: -  --cppflags                 compile flags specific to the C-like preprocessors -  --cxxflags                 compile flags appropriate for C++ programs -  --ldflags                  linker flags -  --libs                     libraries for linking - -EOF -} - -# This function bounds our version with a min and a max. It uses some clever -# POSIX-compliant variable expansion to portably do all the work in the shell -# and avoid any dependency on a particular "sed" or "awk" implementation. -# Notable is that it will only ever compare the first 3 components of versions. -# Further components will be cleanly stripped off. All versions must be -# unadorned, so "v1.0" will *not* work. The minimum version must be in $1, and -# the max in $2. TODO(chandlerc@google.com): If this ever breaks, we should -# investigate expanding this via autom4te from AS_VERSION_COMPARE rather than -# continuing to maintain our own shell version. -check_versions() -{ -  major_version=${version%%.*} -  minor_version="0" -  point_version="0" -  if test "${version#*.}" != "${version}"; then -    minor_version=${version#*.} -    minor_version=${minor_version%%.*} -  fi -  if test "${version#*.*.}" != "${version}"; then -    point_version=${version#*.*.} -    point_version=${point_version%%.*} -  fi - -  min_version="$1" -  min_major_version=${min_version%%.*} -  min_minor_version="0" -  min_point_version="0" -  if test "${min_version#*.}" != "${min_version}"; then -    min_minor_version=${min_version#*.} -    min_minor_version=${min_minor_version%%.*} -  fi -  if test "${min_version#*.*.}" != "${min_version}"; then -    min_point_version=${min_version#*.*.} -    min_point_version=${min_point_version%%.*} -  fi - -  max_version="$2" -  max_major_version=${max_version%%.*} -  max_minor_version="0" -  max_point_version="0" -  if test "${max_version#*.}" != "${max_version}"; then -    max_minor_version=${max_version#*.} -    max_minor_version=${max_minor_version%%.*} -  fi -  if test "${max_version#*.*.}" != "${max_version}"; then -    max_point_version=${max_version#*.*.} -    max_point_version=${max_point_version%%.*} -  fi - -  test $(($major_version)) -lt $(($min_major_version)) && exit 1 -  if test $(($major_version)) -eq $(($min_major_version)); then -    test $(($minor_version)) -lt $(($min_minor_version)) && exit 1 -    if test $(($minor_version)) -eq $(($min_minor_version)); then -      test $(($point_version)) -lt $(($min_point_version)) && exit 1 -    fi -  fi - -  test $(($major_version)) -gt $(($max_major_version)) && exit 1 -  if test $(($major_version)) -eq $(($max_major_version)); then -    test $(($minor_version)) -gt $(($max_minor_version)) && exit 1 -    if test $(($minor_version)) -eq $(($max_minor_version)); then -      test $(($point_version)) -gt $(($max_point_version)) && exit 1 -    fi -  fi - -  exit 0 -} - -# Show the usage line when no arguments are specified. -if test $# -eq 0; then -  show_usage -  exit 1 -fi - -while test $# -gt 0; do -  case $1 in -    --usage)          show_usage;         exit 0;; -    --help)           show_help;          exit 0;; - -    # Installation overrides -    --prefix=*)       GMOCK_PREFIX=${1#--prefix=};; -    --exec-prefix=*)  GMOCK_EXEC_PREFIX=${1#--exec-prefix=};; -    --libdir=*)       GMOCK_LIBDIR=${1#--libdir=};; -    --includedir=*)   GMOCK_INCLUDEDIR=${1#--includedir=};; - -    # Installation queries -    --prefix|--exec-prefix|--libdir|--includedir|--version) -      if test -n "${do_query}"; then -        show_usage -        exit 1 -      fi -      do_query=${1#--} -      ;; - -    # Version checking -    --min-version=*) -      do_check_versions=yes -      min_version=${1#--min-version=} -      ;; -    --max-version=*) -      do_check_versions=yes -      max_version=${1#--max-version=} -      ;; -    --exact-version=*) -      do_check_versions=yes -      exact_version=${1#--exact-version=} -      ;; - -    # Compiler flag output -    --cppflags)       echo_cppflags=yes;; -    --cxxflags)       echo_cxxflags=yes;; -    --ldflags)        echo_ldflags=yes;; -    --libs)           echo_libs=yes;; - -    # Everything else is an error -    *)                show_usage;         exit 1;; -  esac -  shift -done - -# These have defaults filled in by the configure script but can also be -# overridden by environment variables or command line parameters. -prefix="${GMOCK_PREFIX:-@prefix@}" -exec_prefix="${GMOCK_EXEC_PREFIX:-@exec_prefix@}" -libdir="${GMOCK_LIBDIR:-@libdir@}" -includedir="${GMOCK_INCLUDEDIR:-@includedir@}" - -# We try and detect if our binary is not located at its installed location. If -# it's not, we provide variables pointing to the source and build tree rather -# than to the install tree. We also locate Google Test using the configured -# gtest-config script rather than searching the PATH and our bindir for one. -# This allows building against a just-built gmock rather than an installed -# gmock. -bindir="@bindir@" -this_relative_bindir=`dirname $0` -this_bindir=`cd ${this_relative_bindir}; pwd -P` -if test "${this_bindir}" = "${this_bindir%${bindir}}"; then -  # The path to the script doesn't end in the bindir sequence from Autoconf, -  # assume that we are in a build tree. -  build_dir=`dirname ${this_bindir}` -  src_dir=`cd ${this_bindir}/@top_srcdir@; pwd -P` - -  # TODO(chandlerc@google.com): This is a dangerous dependency on libtool, we -  # should work to remove it, and/or remove libtool altogether, replacing it -  # with direct references to the library and a link path. -  gmock_libs="${build_dir}/lib/libgmock.la" -  gmock_ldflags="" - -  # We provide hooks to include from either the source or build dir, where the -  # build dir is always preferred. This will potentially allow us to write -  # build rules for generated headers and have them automatically be preferred -  # over provided versions. -  gmock_cppflags="-I${build_dir}/include -I${src_dir}/include" -  gmock_cxxflags="" - -  # Directly invoke the gtest-config script used during the build process. -  gtest_config="@GTEST_CONFIG@" -else -  # We're using an installed gmock, although it may be staged under some -  # prefix. Assume (as our own libraries do) that we can resolve the prefix, -  # and are present in the dynamic link paths. -  gmock_ldflags="-L${libdir}" -  gmock_libs="-l${name}" -  gmock_cppflags="-I${includedir}" -  gmock_cxxflags="" - -  # We also prefer any gtest-config script installed in our prefix. Lacking -  # one, we look in the PATH for one. -  gtest_config="${bindir}/gtest-config" -  if test ! -x "${gtest_config}"; then -    gtest_config=`which gtest-config` -  fi -fi - -# Ensure that we have located a Google Test to link against. -if ! test -x "${gtest_config}"; then -  echo "Unable to locate Google Test, check your Google Mock configuration" \ -       "and installation" >&2 -  exit 1 -elif ! "${gtest_config}" "--exact-version=@GTEST_VERSION@"; then -  echo "The Google Test found is not the same version as Google Mock was " \ -       "built against" >&2 -  exit 1 -fi - -# Add the necessary Google Test bits into the various flag variables -gmock_cppflags="${gmock_cppflags} `${gtest_config} --cppflags`" -gmock_cxxflags="${gmock_cxxflags} `${gtest_config} --cxxflags`" -gmock_ldflags="${gmock_ldflags} `${gtest_config} --ldflags`" -gmock_libs="${gmock_libs} `${gtest_config} --libs`" - -# Do an installation query if requested. -if test -n "$do_query"; then -  case $do_query in -    prefix)           echo $prefix;       exit 0;; -    exec-prefix)      echo $exec_prefix;  exit 0;; -    libdir)           echo $libdir;       exit 0;; -    includedir)       echo $includedir;   exit 0;; -    version)          echo $version;      exit 0;; -    *)                show_usage;         exit 1;; -  esac -fi - -# Do a version check if requested. -if test "$do_check_versions" = "yes"; then -  # Make sure we didn't receive a bad combination of parameters. -  test "$echo_cppflags" = "yes" && show_usage && exit 1 -  test "$echo_cxxflags" = "yes" && show_usage && exit 1 -  test "$echo_ldflags" = "yes"  && show_usage && exit 1 -  test "$echo_libs" = "yes"     && show_usage && exit 1 - -  if test "$exact_version" != ""; then -    check_versions $exact_version $exact_version -    # unreachable -  else -    check_versions ${min_version:-0.0.0} ${max_version:-9999.9999.9999} -    # unreachable -  fi -fi - -# Do the output in the correct order so that these can be used in-line of -# a compiler invocation. -output="" -test "$echo_cppflags" = "yes" && output="$output $gmock_cppflags" -test "$echo_cxxflags" = "yes" && output="$output $gmock_cxxflags" -test "$echo_ldflags" = "yes"  && output="$output $gmock_ldflags" -test "$echo_libs" = "yes"     && output="$output $gmock_libs" -echo $output - -exit 0 diff --git a/googlemock/scripts/gmock_doctor.py b/googlemock/scripts/gmock_doctor.py deleted file mode 100755 index 74992bc7..00000000 --- a/googlemock/scripts/gmock_doctor.py +++ /dev/null @@ -1,640 +0,0 @@ -#!/usr/bin/env python -# -# Copyright 2008, Google Inc. -# All rights reserved. -# -# Redistribution and use in source and binary forms, with or without -# modification, are permitted provided that the following conditions are -# met: -# -#     * Redistributions of source code must retain the above copyright -# notice, this list of conditions and the following disclaimer. -#     * Redistributions in binary form must reproduce the above -# copyright notice, this list of conditions and the following disclaimer -# in the documentation and/or other materials provided with the -# distribution. -#     * Neither the name of Google Inc. nor the names of its -# contributors may be used to endorse or promote products derived from -# this software without specific prior written permission. -# -# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS -# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT -# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR -# A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT -# OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, -# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT -# LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, -# DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY -# THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT -# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE -# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. - -"""Converts compiler's errors in code using Google Mock to plain English.""" - -__author__ = 'wan@google.com (Zhanyong Wan)' - -import re -import sys - -_VERSION = '1.0.3' - -_EMAIL = 'googlemock@googlegroups.com' - -_COMMON_GMOCK_SYMBOLS = [ -    # Matchers -    '_', -    'A', -    'AddressSatisfies', -    'AllOf', -    'An', -    'AnyOf', -    'ContainerEq', -    'Contains', -    'ContainsRegex', -    'DoubleEq', -    'ElementsAre', -    'ElementsAreArray', -    'EndsWith', -    'Eq', -    'Field', -    'FloatEq', -    'Ge', -    'Gt', -    'HasSubstr', -    'IsInitializedProto', -    'Le', -    'Lt', -    'MatcherCast', -    'Matches', -    'MatchesRegex', -    'NanSensitiveDoubleEq', -    'NanSensitiveFloatEq', -    'Ne', -    'Not', -    'NotNull', -    'Pointee', -    'Property', -    'Ref', -    'ResultOf', -    'SafeMatcherCast', -    'StartsWith', -    'StrCaseEq', -    'StrCaseNe', -    'StrEq', -    'StrNe', -    'Truly', -    'TypedEq', -    'Value', - -    # Actions -    'Assign', -    'ByRef', -    'DeleteArg', -    'DoAll', -    'DoDefault', -    'IgnoreResult', -    'Invoke', -    'InvokeArgument', -    'InvokeWithoutArgs', -    'Return', -    'ReturnNew', -    'ReturnNull', -    'ReturnRef', -    'SaveArg', -    'SetArgReferee', -    'SetArgPointee', -    'SetArgumentPointee', -    'SetArrayArgument', -    'SetErrnoAndReturn', -    'Throw', -    'WithArg', -    'WithArgs', -    'WithoutArgs', - -    # Cardinalities -    'AnyNumber', -    'AtLeast', -    'AtMost', -    'Between', -    'Exactly', - -    # Sequences -    'InSequence', -    'Sequence', - -    # Misc -    'DefaultValue', -    'Mock', -    ] - -# Regex for matching source file path and line number in the compiler's errors. -_GCC_FILE_LINE_RE = r'(?P<file>.*):(?P<line>\d+):(\d+:)?\s+' -_CLANG_FILE_LINE_RE = r'(?P<file>.*):(?P<line>\d+):(?P<column>\d+):\s+' -_CLANG_NON_GMOCK_FILE_LINE_RE = ( -    r'(?P<file>.*[/\\^](?!gmock-)[^/\\]+):(?P<line>\d+):(?P<column>\d+):\s+') - - -def _FindAllMatches(regex, s): -  """Generates all matches of regex in string s.""" - -  r = re.compile(regex) -  return r.finditer(s) - - -def _GenericDiagnoser(short_name, long_name, diagnoses, msg): -  """Diagnoses the given disease by pattern matching. - -  Can provide different diagnoses for different patterns. - -  Args: -    short_name: Short name of the disease. -    long_name:  Long name of the disease. -    diagnoses:  A list of pairs (regex, pattern for formatting the diagnosis -                for matching regex). -    msg:        Compiler's error messages. -  Yields: -    Tuples of the form -      (short name of disease, long name of disease, diagnosis). -  """ -  for regex, diagnosis in diagnoses: -    if re.search(regex, msg): -      diagnosis = '%(file)s:%(line)s:' + diagnosis -      for m in _FindAllMatches(regex, msg): -        yield (short_name, long_name, diagnosis % m.groupdict()) - - -def _NeedToReturnReferenceDiagnoser(msg): -  """Diagnoses the NRR disease, given the error messages by the compiler.""" - -  gcc_regex = (r'In member function \'testing::internal::ReturnAction<R>.*\n' -               + _GCC_FILE_LINE_RE + r'instantiated from here\n' -               r'.*gmock-actions\.h.*error: creating array with negative size') -  clang_regex = (r'error:.*array.*negative.*\r?\n' -                 r'(.*\n)*?' + -                 _CLANG_NON_GMOCK_FILE_LINE_RE + -                 r'note: in instantiation of function template specialization ' -                 r'\'testing::internal::ReturnAction<(?P<type>.*)>' -                 r'::operator Action<.*>\' requested here') -  clang11_re = (r'use_ReturnRef_instead_of_Return_to_return_a_reference.*' -                r'(.*\n)*?' + _CLANG_NON_GMOCK_FILE_LINE_RE) - -  diagnosis = """ -You are using a Return() action in a function that returns a reference to -%(type)s.  Please use ReturnRef() instead.""" -  return _GenericDiagnoser('NRR', 'Need to Return Reference', -                           [(clang_regex, diagnosis), -                            (clang11_re, diagnosis % {'type': 'a type'}), -                            (gcc_regex, diagnosis % {'type': 'a type'})], -                           msg) - - -def _NeedToReturnSomethingDiagnoser(msg): -  """Diagnoses the NRS disease, given the error messages by the compiler.""" - -  gcc_regex = (_GCC_FILE_LINE_RE + r'(instantiated from here\n.' -               r'*gmock.*actions\.h.*error: void value not ignored)' -               r'|(error: control reaches end of non-void function)') -  clang_regex1 = (_CLANG_FILE_LINE_RE + -                  r'error: cannot initialize return object ' -                  r'of type \'Result\' \(aka \'(?P<return_type>.*)\'\) ' -                  r'with an rvalue of type \'void\'') -  clang_regex2 = (_CLANG_FILE_LINE_RE + -                  r'error: cannot initialize return object ' -                  r'of type \'(?P<return_type>.*)\' ' -                  r'with an rvalue of type \'void\'') -  diagnosis = """ -You are using an action that returns void, but it needs to return -%(return_type)s.  Please tell it *what* to return.  Perhaps you can use -the pattern DoAll(some_action, Return(some_value))?""" -  return _GenericDiagnoser( -      'NRS', -      'Need to Return Something', -      [(gcc_regex, diagnosis % {'return_type': '*something*'}), -       (clang_regex1, diagnosis), -       (clang_regex2, diagnosis)], -      msg) - - -def _NeedToReturnNothingDiagnoser(msg): -  """Diagnoses the NRN disease, given the error messages by the compiler.""" - -  gcc_regex = (_GCC_FILE_LINE_RE + r'instantiated from here\n' -               r'.*gmock-actions\.h.*error: instantiation of ' -               r'\'testing::internal::ReturnAction<R>::Impl<F>::value_\' ' -               r'as type \'void\'') -  clang_regex1 = (r'error: field has incomplete type ' -                  r'\'Result\' \(aka \'void\'\)(\r)?\n' -                  r'(.*\n)*?' + -                  _CLANG_NON_GMOCK_FILE_LINE_RE + r'note: in instantiation ' -                  r'of function template specialization ' -                  r'\'testing::internal::ReturnAction<(?P<return_type>.*)>' -                  r'::operator Action<void \(.*\)>\' requested here') -  clang_regex2 = (r'error: field has incomplete type ' -                  r'\'Result\' \(aka \'void\'\)(\r)?\n' -                  r'(.*\n)*?' + -                  _CLANG_NON_GMOCK_FILE_LINE_RE + r'note: in instantiation ' -                  r'of function template specialization ' -                  r'\'testing::internal::DoBothAction<.*>' -                  r'::operator Action<(?P<return_type>.*) \(.*\)>\' ' -                  r'requested here') -  diagnosis = """ -You are using an action that returns %(return_type)s, but it needs to return -void.  Please use a void-returning action instead. - -All actions but the last in DoAll(...) must return void.  Perhaps you need -to re-arrange the order of actions in a DoAll(), if you are using one?""" -  return _GenericDiagnoser( -      'NRN', -      'Need to Return Nothing', -      [(gcc_regex, diagnosis % {'return_type': '*something*'}), -       (clang_regex1, diagnosis), -       (clang_regex2, diagnosis)], -      msg) - - -def _IncompleteByReferenceArgumentDiagnoser(msg): -  """Diagnoses the IBRA disease, given the error messages by the compiler.""" - -  gcc_regex = (_GCC_FILE_LINE_RE + r'instantiated from here\n' -               r'.*gtest-printers\.h.*error: invalid application of ' -               r'\'sizeof\' to incomplete type \'(?P<type>.*)\'') - -  clang_regex = (r'.*gtest-printers\.h.*error: invalid application of ' -                 r'\'sizeof\' to an incomplete type ' -                 r'\'(?P<type>.*)( const)?\'\r?\n' -                 r'(.*\n)*?' + -                 _CLANG_NON_GMOCK_FILE_LINE_RE + -                 r'note: in instantiation of member function ' -                 r'\'testing::internal2::TypeWithoutFormatter<.*>::' -                 r'PrintValue\' requested here') -  diagnosis = """ -In order to mock this function, Google Mock needs to see the definition -of type "%(type)s" - declaration alone is not enough.  Either #include -the header that defines it, or change the argument to be passed -by pointer.""" - -  return _GenericDiagnoser('IBRA', 'Incomplete By-Reference Argument Type', -                           [(gcc_regex, diagnosis), -                            (clang_regex, diagnosis)], -                           msg) - - -def _OverloadedFunctionMatcherDiagnoser(msg): -  """Diagnoses the OFM disease, given the error messages by the compiler.""" - -  gcc_regex = (_GCC_FILE_LINE_RE + r'error: no matching function for ' -               r'call to \'Truly\(<unresolved overloaded function type>\)') -  clang_regex = (_CLANG_FILE_LINE_RE + r'error: no matching function for ' -                 r'call to \'Truly') -  diagnosis = """ -The argument you gave to Truly() is an overloaded function.  Please tell -your compiler which overloaded version you want to use. - -For example, if you want to use the version whose signature is -  bool Foo(int n); -you should write -  Truly(static_cast<bool (*)(int n)>(Foo))""" -  return _GenericDiagnoser('OFM', 'Overloaded Function Matcher', -                           [(gcc_regex, diagnosis), -                            (clang_regex, diagnosis)], -                           msg) - - -def _OverloadedFunctionActionDiagnoser(msg): -  """Diagnoses the OFA disease, given the error messages by the compiler.""" - -  gcc_regex = (_GCC_FILE_LINE_RE + r'error: no matching function for call to ' -               r'\'Invoke\(<unresolved overloaded function type>') -  clang_regex = (_CLANG_FILE_LINE_RE + r'error: no matching ' -                 r'function for call to \'Invoke\'\r?\n' -                 r'(.*\n)*?' -                 r'.*\bgmock-generated-actions\.h:\d+:\d+:\s+' -                 r'note: candidate template ignored:\s+' -                 r'couldn\'t infer template argument \'FunctionImpl\'') -  diagnosis = """ -Function you are passing to Invoke is overloaded.  Please tell your compiler -which overloaded version you want to use. - -For example, if you want to use the version whose signature is -  bool MyFunction(int n, double x); -you should write something like -  Invoke(static_cast<bool (*)(int n, double x)>(MyFunction))""" -  return _GenericDiagnoser('OFA', 'Overloaded Function Action', -                           [(gcc_regex, diagnosis), -                            (clang_regex, diagnosis)], -                           msg) - - -def _OverloadedMethodActionDiagnoser(msg): -  """Diagnoses the OMA disease, given the error messages by the compiler.""" - -  gcc_regex = (_GCC_FILE_LINE_RE + r'error: no matching function for ' -               r'call to \'Invoke\(.+, <unresolved overloaded function ' -               r'type>\)') -  clang_regex = (_CLANG_FILE_LINE_RE + r'error: no matching function ' -                 r'for call to \'Invoke\'\r?\n' -                 r'(.*\n)*?' -                 r'.*\bgmock-generated-actions\.h:\d+:\d+: ' -                 r'note: candidate function template not viable: ' -                 r'requires .*, but 2 (arguments )?were provided') -  diagnosis = """ -The second argument you gave to Invoke() is an overloaded method.  Please -tell your compiler which overloaded version you want to use. - -For example, if you want to use the version whose signature is -  class Foo { -    ... -    bool Bar(int n, double x); -  }; -you should write something like -  Invoke(foo, static_cast<bool (Foo::*)(int n, double x)>(&Foo::Bar))""" -  return _GenericDiagnoser('OMA', 'Overloaded Method Action', -                           [(gcc_regex, diagnosis), -                            (clang_regex, diagnosis)], -                           msg) - - -def _MockObjectPointerDiagnoser(msg): -  """Diagnoses the MOP disease, given the error messages by the compiler.""" - -  gcc_regex = (_GCC_FILE_LINE_RE + r'error: request for member ' -               r'\'gmock_(?P<method>.+)\' in \'(?P<mock_object>.+)\', ' -               r'which is of non-class type \'(.*::)*(?P<class_name>.+)\*\'') -  clang_regex = (_CLANG_FILE_LINE_RE + r'error: member reference type ' -                 r'\'(?P<class_name>.*?) *\' is a pointer; ' -                 r'(did you mean|maybe you meant) to use \'->\'\?') -  diagnosis = """ -The first argument to ON_CALL() and EXPECT_CALL() must be a mock *object*, -not a *pointer* to it.  Please write '*(%(mock_object)s)' instead of -'%(mock_object)s' as your first argument. - -For example, given the mock class: - -  class %(class_name)s : public ... { -    ... -    MOCK_METHOD0(%(method)s, ...); -  }; - -and the following mock instance: - -  %(class_name)s* mock_ptr = ... - -you should use the EXPECT_CALL like this: - -  EXPECT_CALL(*mock_ptr, %(method)s(...));""" - -  return _GenericDiagnoser( -      'MOP', -      'Mock Object Pointer', -      [(gcc_regex, diagnosis), -       (clang_regex, diagnosis % {'mock_object': 'mock_object', -                                  'method': 'method', -                                  'class_name': '%(class_name)s'})], -       msg) - - -def _NeedToUseSymbolDiagnoser(msg): -  """Diagnoses the NUS disease, given the error messages by the compiler.""" - -  gcc_regex = (_GCC_FILE_LINE_RE + r'error: \'(?P<symbol>.+)\' ' -               r'(was not declared in this scope|has not been declared)') -  clang_regex = (_CLANG_FILE_LINE_RE + -                 r'error: (use of undeclared identifier|unknown type name|' -                 r'no template named) \'(?P<symbol>[^\']+)\'') -  diagnosis = """ -'%(symbol)s' is defined by Google Mock in the testing namespace. -Did you forget to write -  using testing::%(symbol)s; -?""" -  for m in (list(_FindAllMatches(gcc_regex, msg)) + -            list(_FindAllMatches(clang_regex, msg))): -    symbol = m.groupdict()['symbol'] -    if symbol in _COMMON_GMOCK_SYMBOLS: -      yield ('NUS', 'Need to Use Symbol', diagnosis % m.groupdict()) - - -def _NeedToUseReturnNullDiagnoser(msg): -  """Diagnoses the NRNULL disease, given the error messages by the compiler.""" - -  gcc_regex = ('instantiated from \'testing::internal::ReturnAction<R>' -               '::operator testing::Action<Func>\(\) const.*\n' + -               _GCC_FILE_LINE_RE + r'instantiated from here\n' -               r'.*error: no matching function for call to \'ImplicitCast_\(' -               r'(:?long )?int&\)') -  clang_regex = (r'\bgmock-actions.h:.* error: no matching function for ' -                 r'call to \'ImplicitCast_\'\r?\n' -                 r'(.*\n)*?' + -                 _CLANG_NON_GMOCK_FILE_LINE_RE + r'note: in instantiation ' -                 r'of function template specialization ' -                 r'\'testing::internal::ReturnAction<(int|long)>::operator ' -                 r'Action<(?P<type>.*)\(\)>\' requested here') -  diagnosis = """ -You are probably calling Return(NULL) and the compiler isn't sure how to turn -NULL into %(type)s. Use ReturnNull() instead. -Note: the line number may be off; please fix all instances of Return(NULL).""" -  return _GenericDiagnoser( -      'NRNULL', 'Need to use ReturnNull', -      [(clang_regex, diagnosis), -       (gcc_regex, diagnosis % {'type': 'the right type'})], -      msg) - - -def _TypeInTemplatedBaseDiagnoser(msg): -  """Diagnoses the TTB disease, given the error messages by the compiler.""" - -  # This version works when the type is used as the mock function's return -  # type. -  gcc_4_3_1_regex_type_in_retval = ( -      r'In member function \'int .*\n' + _GCC_FILE_LINE_RE + -      r'error: a function call cannot appear in a constant-expression') -  gcc_4_4_0_regex_type_in_retval = ( -      r'error: a function call cannot appear in a constant-expression' -      + _GCC_FILE_LINE_RE + r'error: template argument 1 is invalid\n') -  # This version works when the type is used as the mock function's sole -  # parameter type. -  gcc_regex_type_of_sole_param = ( -      _GCC_FILE_LINE_RE + -      r'error: \'(?P<type>.+)\' was not declared in this scope\n' -      r'.*error: template argument 1 is invalid\n') -  # This version works when the type is used as a parameter of a mock -  # function that has multiple parameters. -  gcc_regex_type_of_a_param = ( -      r'error: expected `;\' before \'::\' token\n' -      + _GCC_FILE_LINE_RE + -      r'error: \'(?P<type>.+)\' was not declared in this scope\n' -      r'.*error: template argument 1 is invalid\n' -      r'.*error: \'.+\' was not declared in this scope') -  clang_regex_type_of_retval_or_sole_param = ( -      _CLANG_FILE_LINE_RE + -      r'error: use of undeclared identifier \'(?P<type>.*)\'\n' -      r'(.*\n)*?' -      r'(?P=file):(?P=line):\d+: error: ' -      r'non-friend class member \'Result\' cannot have a qualified name' -      ) -  clang_regex_type_of_a_param = ( -      _CLANG_FILE_LINE_RE + -      r'error: C\+\+ requires a type specifier for all declarations\n' -      r'(.*\n)*?' -      r'(?P=file):(?P=line):(?P=column): error: ' -      r'C\+\+ requires a type specifier for all declarations' -      ) -  clang_regex_unknown_type = ( -      _CLANG_FILE_LINE_RE + -      r'error: unknown type name \'(?P<type>[^\']+)\'' -      ) - -  diagnosis = """ -In a mock class template, types or typedefs defined in the base class -template are *not* automatically visible.  This is how C++ works.  Before -you can use a type or typedef named %(type)s defined in base class Base<T>, you -need to make it visible.  One way to do it is: - -  typedef typename Base<T>::%(type)s %(type)s;""" - -  for diag in _GenericDiagnoser( -      'TTB', 'Type in Template Base', -      [(gcc_4_3_1_regex_type_in_retval, diagnosis % {'type': 'Foo'}), -       (gcc_4_4_0_regex_type_in_retval, diagnosis % {'type': 'Foo'}), -       (gcc_regex_type_of_sole_param, diagnosis), -       (gcc_regex_type_of_a_param, diagnosis), -       (clang_regex_type_of_retval_or_sole_param, diagnosis), -       (clang_regex_type_of_a_param, diagnosis % {'type': 'Foo'})], -      msg): -    yield diag -  # Avoid overlap with the NUS pattern. -  for m in _FindAllMatches(clang_regex_unknown_type, msg): -    type_ = m.groupdict()['type'] -    if type_ not in _COMMON_GMOCK_SYMBOLS: -      yield ('TTB', 'Type in Template Base', diagnosis % m.groupdict()) - - -def _WrongMockMethodMacroDiagnoser(msg): -  """Diagnoses the WMM disease, given the error messages by the compiler.""" - -  gcc_regex = (_GCC_FILE_LINE_RE + -               r'.*this_method_does_not_take_(?P<wrong_args>\d+)_argument.*\n' -               r'.*\n' -               r'.*candidates are.*FunctionMocker<[^>]+A(?P<args>\d+)\)>') -  clang_regex = (_CLANG_NON_GMOCK_FILE_LINE_RE + -                 r'error:.*array.*negative.*r?\n' -                 r'(.*\n)*?' -                 r'(?P=file):(?P=line):(?P=column): error: too few arguments ' -                 r'to function call, expected (?P<args>\d+), ' -                 r'have (?P<wrong_args>\d+)') -  clang11_re = (_CLANG_NON_GMOCK_FILE_LINE_RE + -                r'.*this_method_does_not_take_' -                r'(?P<wrong_args>\d+)_argument.*') -  diagnosis = """ -You are using MOCK_METHOD%(wrong_args)s to define a mock method that has -%(args)s arguments. Use MOCK_METHOD%(args)s (or MOCK_CONST_METHOD%(args)s, -MOCK_METHOD%(args)s_T, MOCK_CONST_METHOD%(args)s_T as appropriate) instead.""" -  return _GenericDiagnoser('WMM', 'Wrong MOCK_METHODn Macro', -                           [(gcc_regex, diagnosis), -                            (clang11_re, diagnosis % {'wrong_args': 'm', -                                                      'args': 'n'}), -                            (clang_regex, diagnosis)], -                           msg) - - -def _WrongParenPositionDiagnoser(msg): -  """Diagnoses the WPP disease, given the error messages by the compiler.""" - -  gcc_regex = (_GCC_FILE_LINE_RE + -               r'error:.*testing::internal::MockSpec<.* has no member named \'' -               r'(?P<method>\w+)\'') -  clang_regex = (_CLANG_NON_GMOCK_FILE_LINE_RE + -                 r'error: no member named \'(?P<method>\w+)\' in ' -                 r'\'testing::internal::MockSpec<.*>\'') -  diagnosis = """ -The closing parenthesis of ON_CALL or EXPECT_CALL should be *before* -".%(method)s".  For example, you should write: -  EXPECT_CALL(my_mock, Foo(_)).%(method)s(...); -instead of: -  EXPECT_CALL(my_mock, Foo(_).%(method)s(...));""" -  return _GenericDiagnoser('WPP', 'Wrong Parenthesis Position', -                           [(gcc_regex, diagnosis), -                            (clang_regex, diagnosis)], -                           msg) - - -_DIAGNOSERS = [ -    _IncompleteByReferenceArgumentDiagnoser, -    _MockObjectPointerDiagnoser, -    _NeedToReturnNothingDiagnoser, -    _NeedToReturnReferenceDiagnoser, -    _NeedToReturnSomethingDiagnoser, -    _NeedToUseReturnNullDiagnoser, -    _NeedToUseSymbolDiagnoser, -    _OverloadedFunctionActionDiagnoser, -    _OverloadedFunctionMatcherDiagnoser, -    _OverloadedMethodActionDiagnoser, -    _TypeInTemplatedBaseDiagnoser, -    _WrongMockMethodMacroDiagnoser, -    _WrongParenPositionDiagnoser, -    ] - - -def Diagnose(msg): -  """Generates all possible diagnoses given the compiler error message.""" - -  msg = re.sub(r'\x1b\[[^m]*m', '', msg)  # Strips all color formatting. -  # Assuming the string is using the UTF-8 encoding, replaces the left and -  # the right single quote characters with apostrophes. -  msg = re.sub(r'(\xe2\x80\x98|\xe2\x80\x99)', "'", msg) - -  diagnoses = [] -  for diagnoser in _DIAGNOSERS: -    for diag in diagnoser(msg): -      diagnosis = '[%s - %s]\n%s' % diag -      if not diagnosis in diagnoses: -        diagnoses.append(diagnosis) -  return diagnoses - - -def main(): -  print ('Google Mock Doctor v%s - ' -         'diagnoses problems in code using Google Mock.' % _VERSION) - -  if sys.stdin.isatty(): -    print ('Please copy and paste the compiler errors here.  Press c-D when ' -           'you are done:') -  else: -    print ('Waiting for compiler errors on stdin . . .') - -  msg = sys.stdin.read().strip() -  diagnoses = Diagnose(msg) -  count = len(diagnoses) -  if not count: -    print (""" -Your compiler complained: -8<------------------------------------------------------------ -%s ------------------------------------------------------------->8 - -Uh-oh, I'm not smart enough to figure out what the problem is. :-( -However... -If you send your source code and the compiler's error messages to -%s, you can be helped and I can get smarter -- -win-win for us!""" % (msg, _EMAIL)) -  else: -    print ('------------------------------------------------------------') -    print ('Your code appears to have the following',) -    if count > 1: -      print ('%s diseases:' % (count,)) -    else: -      print ('disease:') -    i = 0 -    for d in diagnoses: -      i += 1 -      if count > 1: -        print ('\n#%s:' % (i,)) -      print (d) -    print (""" -How did I do?  If you think I'm wrong or unhelpful, please send your -source code and the compiler's error messages to %s. -Then you can be helped and I can get smarter -- I promise I won't be upset!""" % -           _EMAIL) - - -if __name__ == '__main__': -  main() diff --git a/googlemock/scripts/pump.py b/googlemock/scripts/pump.py new file mode 100755 index 00000000..66e32170 --- /dev/null +++ b/googlemock/scripts/pump.py @@ -0,0 +1,855 @@ +#!/usr/bin/env python2.7 +# +# Copyright 2008, Google Inc. +# All rights reserved. +# +# Redistribution and use in source and binary forms, with or without +# modification, are permitted provided that the following conditions are +# met: +# +#     * Redistributions of source code must retain the above copyright +# notice, this list of conditions and the following disclaimer. +#     * Redistributions in binary form must reproduce the above +# copyright notice, this list of conditions and the following disclaimer +# in the documentation and/or other materials provided with the +# distribution. +#     * Neither the name of Google Inc. nor the names of its +# contributors may be used to endorse or promote products derived from +# this software without specific prior written permission. +# +# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +# A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +# OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +# LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +# DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +# THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +"""pump v0.2.0 - Pretty Useful for Meta Programming. + +A tool for preprocessor meta programming.  Useful for generating +repetitive boilerplate code.  Especially useful for writing C++ +classes, functions, macros, and templates that need to work with +various number of arguments. + +USAGE: +       pump.py SOURCE_FILE + +EXAMPLES: +       pump.py foo.cc.pump +         Converts foo.cc.pump to foo.cc. + +GRAMMAR: +       CODE ::= ATOMIC_CODE* +       ATOMIC_CODE ::= $var ID = EXPRESSION +           | $var ID = [[ CODE ]] +           | $range ID EXPRESSION..EXPRESSION +           | $for ID SEPARATOR [[ CODE ]] +           | $($) +           | $ID +           | $(EXPRESSION) +           | $if EXPRESSION [[ CODE ]] ELSE_BRANCH +           | [[ CODE ]] +           | RAW_CODE +       SEPARATOR ::= RAW_CODE | EMPTY +       ELSE_BRANCH ::= $else [[ CODE ]] +           | $elif EXPRESSION [[ CODE ]] ELSE_BRANCH +           | EMPTY +       EXPRESSION has Python syntax. +""" + +from __future__ import print_function + +import os +import re +import sys + + +TOKEN_TABLE = [ +    (re.compile(r'\$var\s+'), '$var'), +    (re.compile(r'\$elif\s+'), '$elif'), +    (re.compile(r'\$else\s+'), '$else'), +    (re.compile(r'\$for\s+'), '$for'), +    (re.compile(r'\$if\s+'), '$if'), +    (re.compile(r'\$range\s+'), '$range'), +    (re.compile(r'\$[_A-Za-z]\w*'), '$id'), +    (re.compile(r'\$\(\$\)'), '$($)'), +    (re.compile(r'\$'), '$'), +    (re.compile(r'\[\[\n?'), '[['), +    (re.compile(r'\]\]\n?'), ']]'), +    ] + + +class Cursor: +  """Represents a position (line and column) in a text file.""" + +  def __init__(self, line=-1, column=-1): +    self.line = line +    self.column = column + +  def __eq__(self, rhs): +    return self.line == rhs.line and self.column == rhs.column + +  def __ne__(self, rhs): +    return not self == rhs + +  def __lt__(self, rhs): +    return self.line < rhs.line or ( +        self.line == rhs.line and self.column < rhs.column) + +  def __le__(self, rhs): +    return self < rhs or self == rhs + +  def __gt__(self, rhs): +    return rhs < self + +  def __ge__(self, rhs): +    return rhs <= self + +  def __str__(self): +    if self == Eof(): +      return 'EOF' +    else: +      return '%s(%s)' % (self.line + 1, self.column) + +  def __add__(self, offset): +    return Cursor(self.line, self.column + offset) + +  def __sub__(self, offset): +    return Cursor(self.line, self.column - offset) + +  def Clone(self): +    """Returns a copy of self.""" + +    return Cursor(self.line, self.column) + + +# Special cursor to indicate the end-of-file. +def Eof(): +  """Returns the special cursor to denote the end-of-file.""" +  return Cursor(-1, -1) + + +class Token: +  """Represents a token in a Pump source file.""" + +  def __init__(self, start=None, end=None, value=None, token_type=None): +    if start is None: +      self.start = Eof() +    else: +      self.start = start +    if end is None: +      self.end = Eof() +    else: +      self.end = end +    self.value = value +    self.token_type = token_type + +  def __str__(self): +    return 'Token @%s: \'%s\' type=%s' % ( +        self.start, self.value, self.token_type) + +  def Clone(self): +    """Returns a copy of self.""" + +    return Token(self.start.Clone(), self.end.Clone(), self.value, +                 self.token_type) + + +def StartsWith(lines, pos, string): +  """Returns True iff the given position in lines starts with 'string'.""" + +  return lines[pos.line][pos.column:].startswith(string) + + +def FindFirstInLine(line, token_table): +  best_match_start = -1 +  for (regex, token_type) in token_table: +    m = regex.search(line) +    if m: +      # We found regex in lines +      if best_match_start < 0 or m.start() < best_match_start: +        best_match_start = m.start() +        best_match_length = m.end() - m.start() +        best_match_token_type = token_type + +  if best_match_start < 0: +    return None + +  return (best_match_start, best_match_length, best_match_token_type) + + +def FindFirst(lines, token_table, cursor): +  """Finds the first occurrence of any string in strings in lines.""" + +  start = cursor.Clone() +  cur_line_number = cursor.line +  for line in lines[start.line:]: +    if cur_line_number == start.line: +      line = line[start.column:] +    m = FindFirstInLine(line, token_table) +    if m: +      # We found a regex in line. +      (start_column, length, token_type) = m +      if cur_line_number == start.line: +        start_column += start.column +      found_start = Cursor(cur_line_number, start_column) +      found_end = found_start + length +      return MakeToken(lines, found_start, found_end, token_type) +    cur_line_number += 1 +  # We failed to find str in lines +  return None + + +def SubString(lines, start, end): +  """Returns a substring in lines.""" + +  if end == Eof(): +    end = Cursor(len(lines) - 1, len(lines[-1])) + +  if start >= end: +    return '' + +  if start.line == end.line: +    return lines[start.line][start.column:end.column] + +  result_lines = ([lines[start.line][start.column:]] + +                  lines[start.line + 1:end.line] + +                  [lines[end.line][:end.column]]) +  return ''.join(result_lines) + + +def StripMetaComments(str): +  """Strip meta comments from each line in the given string.""" + +  # First, completely remove lines containing nothing but a meta +  # comment, including the trailing \n. +  str = re.sub(r'^\s*\$\$.*\n', '', str) + +  # Then, remove meta comments from contentful lines. +  return re.sub(r'\s*\$\$.*', '', str) + + +def MakeToken(lines, start, end, token_type): +  """Creates a new instance of Token.""" + +  return Token(start, end, SubString(lines, start, end), token_type) + + +def ParseToken(lines, pos, regex, token_type): +  line = lines[pos.line][pos.column:] +  m = regex.search(line) +  if m and not m.start(): +    return MakeToken(lines, pos, pos + m.end(), token_type) +  else: +    print('ERROR: %s expected at %s.' % (token_type, pos)) +    sys.exit(1) + + +ID_REGEX = re.compile(r'[_A-Za-z]\w*') +EQ_REGEX = re.compile(r'=') +REST_OF_LINE_REGEX = re.compile(r'.*?(?=$|\$\$)') +OPTIONAL_WHITE_SPACES_REGEX = re.compile(r'\s*') +WHITE_SPACE_REGEX = re.compile(r'\s') +DOT_DOT_REGEX = re.compile(r'\.\.') + + +def Skip(lines, pos, regex): +  line = lines[pos.line][pos.column:] +  m = re.search(regex, line) +  if m and not m.start(): +    return pos + m.end() +  else: +    return pos + + +def SkipUntil(lines, pos, regex, token_type): +  line = lines[pos.line][pos.column:] +  m = re.search(regex, line) +  if m: +    return pos + m.start() +  else: +    print ('ERROR: %s expected on line %s after column %s.' % +           (token_type, pos.line + 1, pos.column)) +    sys.exit(1) + + +def ParseExpTokenInParens(lines, pos): +  def ParseInParens(pos): +    pos = Skip(lines, pos, OPTIONAL_WHITE_SPACES_REGEX) +    pos = Skip(lines, pos, r'\(') +    pos = Parse(pos) +    pos = Skip(lines, pos, r'\)') +    return pos + +  def Parse(pos): +    pos = SkipUntil(lines, pos, r'\(|\)', ')') +    if SubString(lines, pos, pos + 1) == '(': +      pos = Parse(pos + 1) +      pos = Skip(lines, pos, r'\)') +      return Parse(pos) +    else: +      return pos + +  start = pos.Clone() +  pos = ParseInParens(pos) +  return MakeToken(lines, start, pos, 'exp') + + +def RStripNewLineFromToken(token): +  if token.value.endswith('\n'): +    return Token(token.start, token.end, token.value[:-1], token.token_type) +  else: +    return token + + +def TokenizeLines(lines, pos): +  while True: +    found = FindFirst(lines, TOKEN_TABLE, pos) +    if not found: +      yield MakeToken(lines, pos, Eof(), 'code') +      return + +    if found.start == pos: +      prev_token = None +      prev_token_rstripped = None +    else: +      prev_token = MakeToken(lines, pos, found.start, 'code') +      prev_token_rstripped = RStripNewLineFromToken(prev_token) + +    if found.token_type == '$var': +      if prev_token_rstripped: +        yield prev_token_rstripped +      yield found +      id_token = ParseToken(lines, found.end, ID_REGEX, 'id') +      yield id_token +      pos = Skip(lines, id_token.end, OPTIONAL_WHITE_SPACES_REGEX) + +      eq_token = ParseToken(lines, pos, EQ_REGEX, '=') +      yield eq_token +      pos = Skip(lines, eq_token.end, r'\s*') + +      if SubString(lines, pos, pos + 2) != '[[': +        exp_token = ParseToken(lines, pos, REST_OF_LINE_REGEX, 'exp') +        yield exp_token +        pos = Cursor(exp_token.end.line + 1, 0) +    elif found.token_type == '$for': +      if prev_token_rstripped: +        yield prev_token_rstripped +      yield found +      id_token = ParseToken(lines, found.end, ID_REGEX, 'id') +      yield id_token +      pos = Skip(lines, id_token.end, WHITE_SPACE_REGEX) +    elif found.token_type == '$range': +      if prev_token_rstripped: +        yield prev_token_rstripped +      yield found +      id_token = ParseToken(lines, found.end, ID_REGEX, 'id') +      yield id_token +      pos = Skip(lines, id_token.end, OPTIONAL_WHITE_SPACES_REGEX) + +      dots_pos = SkipUntil(lines, pos, DOT_DOT_REGEX, '..') +      yield MakeToken(lines, pos, dots_pos, 'exp') +      yield MakeToken(lines, dots_pos, dots_pos + 2, '..') +      pos = dots_pos + 2 +      new_pos = Cursor(pos.line + 1, 0) +      yield MakeToken(lines, pos, new_pos, 'exp') +      pos = new_pos +    elif found.token_type == '$': +      if prev_token: +        yield prev_token +      yield found +      exp_token = ParseExpTokenInParens(lines, found.end) +      yield exp_token +      pos = exp_token.end +    elif (found.token_type == ']]' or found.token_type == '$if' or +          found.token_type == '$elif' or found.token_type == '$else'): +      if prev_token_rstripped: +        yield prev_token_rstripped +      yield found +      pos = found.end +    else: +      if prev_token: +        yield prev_token +      yield found +      pos = found.end + + +def Tokenize(s): +  """A generator that yields the tokens in the given string.""" +  if s != '': +    lines = s.splitlines(True) +    for token in TokenizeLines(lines, Cursor(0, 0)): +      yield token + + +class CodeNode: +  def __init__(self, atomic_code_list=None): +    self.atomic_code = atomic_code_list + + +class VarNode: +  def __init__(self, identifier=None, atomic_code=None): +    self.identifier = identifier +    self.atomic_code = atomic_code + + +class RangeNode: +  def __init__(self, identifier=None, exp1=None, exp2=None): +    self.identifier = identifier +    self.exp1 = exp1 +    self.exp2 = exp2 + + +class ForNode: +  def __init__(self, identifier=None, sep=None, code=None): +    self.identifier = identifier +    self.sep = sep +    self.code = code + + +class ElseNode: +  def __init__(self, else_branch=None): +    self.else_branch = else_branch + + +class IfNode: +  def __init__(self, exp=None, then_branch=None, else_branch=None): +    self.exp = exp +    self.then_branch = then_branch +    self.else_branch = else_branch + + +class RawCodeNode: +  def __init__(self, token=None): +    self.raw_code = token + + +class LiteralDollarNode: +  def __init__(self, token): +    self.token = token + + +class ExpNode: +  def __init__(self, token, python_exp): +    self.token = token +    self.python_exp = python_exp + + +def PopFront(a_list): +  head = a_list[0] +  a_list[:1] = [] +  return head + + +def PushFront(a_list, elem): +  a_list[:0] = [elem] + + +def PopToken(a_list, token_type=None): +  token = PopFront(a_list) +  if token_type is not None and token.token_type != token_type: +    print('ERROR: %s expected at %s' % (token_type, token.start)) +    print('ERROR: %s found instead' % (token,)) +    sys.exit(1) + +  return token + + +def PeekToken(a_list): +  if not a_list: +    return None + +  return a_list[0] + + +def ParseExpNode(token): +  python_exp = re.sub(r'([_A-Za-z]\w*)', r'self.GetValue("\1")', token.value) +  return ExpNode(token, python_exp) + + +def ParseElseNode(tokens): +  def Pop(token_type=None): +    return PopToken(tokens, token_type) + +  next = PeekToken(tokens) +  if not next: +    return None +  if next.token_type == '$else': +    Pop('$else') +    Pop('[[') +    code_node = ParseCodeNode(tokens) +    Pop(']]') +    return code_node +  elif next.token_type == '$elif': +    Pop('$elif') +    exp = Pop('code') +    Pop('[[') +    code_node = ParseCodeNode(tokens) +    Pop(']]') +    inner_else_node = ParseElseNode(tokens) +    return CodeNode([IfNode(ParseExpNode(exp), code_node, inner_else_node)]) +  elif not next.value.strip(): +    Pop('code') +    return ParseElseNode(tokens) +  else: +    return None + + +def ParseAtomicCodeNode(tokens): +  def Pop(token_type=None): +    return PopToken(tokens, token_type) + +  head = PopFront(tokens) +  t = head.token_type +  if t == 'code': +    return RawCodeNode(head) +  elif t == '$var': +    id_token = Pop('id') +    Pop('=') +    next = PeekToken(tokens) +    if next.token_type == 'exp': +      exp_token = Pop() +      return VarNode(id_token, ParseExpNode(exp_token)) +    Pop('[[') +    code_node = ParseCodeNode(tokens) +    Pop(']]') +    return VarNode(id_token, code_node) +  elif t == '$for': +    id_token = Pop('id') +    next_token = PeekToken(tokens) +    if next_token.token_type == 'code': +      sep_token = next_token +      Pop('code') +    else: +      sep_token = None +    Pop('[[') +    code_node = ParseCodeNode(tokens) +    Pop(']]') +    return ForNode(id_token, sep_token, code_node) +  elif t == '$if': +    exp_token = Pop('code') +    Pop('[[') +    code_node = ParseCodeNode(tokens) +    Pop(']]') +    else_node = ParseElseNode(tokens) +    return IfNode(ParseExpNode(exp_token), code_node, else_node) +  elif t == '$range': +    id_token = Pop('id') +    exp1_token = Pop('exp') +    Pop('..') +    exp2_token = Pop('exp') +    return RangeNode(id_token, ParseExpNode(exp1_token), +                     ParseExpNode(exp2_token)) +  elif t == '$id': +    return ParseExpNode(Token(head.start + 1, head.end, head.value[1:], 'id')) +  elif t == '$($)': +    return LiteralDollarNode(head) +  elif t == '$': +    exp_token = Pop('exp') +    return ParseExpNode(exp_token) +  elif t == '[[': +    code_node = ParseCodeNode(tokens) +    Pop(']]') +    return code_node +  else: +    PushFront(tokens, head) +    return None + + +def ParseCodeNode(tokens): +  atomic_code_list = [] +  while True: +    if not tokens: +      break +    atomic_code_node = ParseAtomicCodeNode(tokens) +    if atomic_code_node: +      atomic_code_list.append(atomic_code_node) +    else: +      break +  return CodeNode(atomic_code_list) + + +def ParseToAST(pump_src_text): +  """Convert the given Pump source text into an AST.""" +  tokens = list(Tokenize(pump_src_text)) +  code_node = ParseCodeNode(tokens) +  return code_node + + +class Env: +  def __init__(self): +    self.variables = [] +    self.ranges = [] + +  def Clone(self): +    clone = Env() +    clone.variables = self.variables[:] +    clone.ranges = self.ranges[:] +    return clone + +  def PushVariable(self, var, value): +    # If value looks like an int, store it as an int. +    try: +      int_value = int(value) +      if ('%s' % int_value) == value: +        value = int_value +    except Exception: +      pass +    self.variables[:0] = [(var, value)] + +  def PopVariable(self): +    self.variables[:1] = [] + +  def PushRange(self, var, lower, upper): +    self.ranges[:0] = [(var, lower, upper)] + +  def PopRange(self): +    self.ranges[:1] = [] + +  def GetValue(self, identifier): +    for (var, value) in self.variables: +      if identifier == var: +        return value + +    print('ERROR: meta variable %s is undefined.' % (identifier,)) +    sys.exit(1) + +  def EvalExp(self, exp): +    try: +      result = eval(exp.python_exp) +    except Exception as e:  # pylint: disable=broad-except +      print('ERROR: caught exception %s: %s' % (e.__class__.__name__, e)) +      print('ERROR: failed to evaluate meta expression %s at %s' % +            (exp.python_exp, exp.token.start)) +      sys.exit(1) +    return result + +  def GetRange(self, identifier): +    for (var, lower, upper) in self.ranges: +      if identifier == var: +        return (lower, upper) + +    print('ERROR: range %s is undefined.' % (identifier,)) +    sys.exit(1) + + +class Output: +  def __init__(self): +    self.string = '' + +  def GetLastLine(self): +    index = self.string.rfind('\n') +    if index < 0: +      return '' + +    return self.string[index + 1:] + +  def Append(self, s): +    self.string += s + + +def RunAtomicCode(env, node, output): +  if isinstance(node, VarNode): +    identifier = node.identifier.value.strip() +    result = Output() +    RunAtomicCode(env.Clone(), node.atomic_code, result) +    value = result.string +    env.PushVariable(identifier, value) +  elif isinstance(node, RangeNode): +    identifier = node.identifier.value.strip() +    lower = int(env.EvalExp(node.exp1)) +    upper = int(env.EvalExp(node.exp2)) +    env.PushRange(identifier, lower, upper) +  elif isinstance(node, ForNode): +    identifier = node.identifier.value.strip() +    if node.sep is None: +      sep = '' +    else: +      sep = node.sep.value +    (lower, upper) = env.GetRange(identifier) +    for i in range(lower, upper + 1): +      new_env = env.Clone() +      new_env.PushVariable(identifier, i) +      RunCode(new_env, node.code, output) +      if i != upper: +        output.Append(sep) +  elif isinstance(node, RawCodeNode): +    output.Append(node.raw_code.value) +  elif isinstance(node, IfNode): +    cond = env.EvalExp(node.exp) +    if cond: +      RunCode(env.Clone(), node.then_branch, output) +    elif node.else_branch is not None: +      RunCode(env.Clone(), node.else_branch, output) +  elif isinstance(node, ExpNode): +    value = env.EvalExp(node) +    output.Append('%s' % (value,)) +  elif isinstance(node, LiteralDollarNode): +    output.Append('$') +  elif isinstance(node, CodeNode): +    RunCode(env.Clone(), node, output) +  else: +    print('BAD') +    print(node) +    sys.exit(1) + + +def RunCode(env, code_node, output): +  for atomic_code in code_node.atomic_code: +    RunAtomicCode(env, atomic_code, output) + + +def IsSingleLineComment(cur_line): +  return '//' in cur_line + + +def IsInPreprocessorDirective(prev_lines, cur_line): +  if cur_line.lstrip().startswith('#'): +    return True +  return prev_lines and prev_lines[-1].endswith('\\') + + +def WrapComment(line, output): +  loc = line.find('//') +  before_comment = line[:loc].rstrip() +  if before_comment == '': +    indent = loc +  else: +    output.append(before_comment) +    indent = len(before_comment) - len(before_comment.lstrip()) +  prefix = indent*' ' + '// ' +  max_len = 80 - len(prefix) +  comment = line[loc + 2:].strip() +  segs = [seg for seg in re.split(r'(\w+\W*)', comment) if seg != ''] +  cur_line = '' +  for seg in segs: +    if len((cur_line + seg).rstrip()) < max_len: +      cur_line += seg +    else: +      if cur_line.strip() != '': +        output.append(prefix + cur_line.rstrip()) +      cur_line = seg.lstrip() +  if cur_line.strip() != '': +    output.append(prefix + cur_line.strip()) + + +def WrapCode(line, line_concat, output): +  indent = len(line) - len(line.lstrip()) +  prefix = indent*' '  # Prefix of the current line +  max_len = 80 - indent - len(line_concat)  # Maximum length of the current line +  new_prefix = prefix + 4*' '  # Prefix of a continuation line +  new_max_len = max_len - 4  # Maximum length of a continuation line +  # Prefers to wrap a line after a ',' or ';'. +  segs = [seg for seg in re.split(r'([^,;]+[,;]?)', line.strip()) if seg != ''] +  cur_line = ''  # The current line without leading spaces. +  for seg in segs: +    # If the line is still too long, wrap at a space. +    while cur_line == '' and len(seg.strip()) > max_len: +      seg = seg.lstrip() +      split_at = seg.rfind(' ', 0, max_len) +      output.append(prefix + seg[:split_at].strip() + line_concat) +      seg = seg[split_at + 1:] +      prefix = new_prefix +      max_len = new_max_len + +    if len((cur_line + seg).rstrip()) < max_len: +      cur_line = (cur_line + seg).lstrip() +    else: +      output.append(prefix + cur_line.rstrip() + line_concat) +      prefix = new_prefix +      max_len = new_max_len +      cur_line = seg.lstrip() +  if cur_line.strip() != '': +    output.append(prefix + cur_line.strip()) + + +def WrapPreprocessorDirective(line, output): +  WrapCode(line, ' \\', output) + + +def WrapPlainCode(line, output): +  WrapCode(line, '', output) + + +def IsMultiLineIWYUPragma(line): +  return re.search(r'/\* IWYU pragma: ', line) + + +def IsHeaderGuardIncludeOrOneLineIWYUPragma(line): +  return (re.match(r'^#(ifndef|define|endif\s*//)\s*[\w_]+\s*$', line) or +          re.match(r'^#include\s', line) or +          # Don't break IWYU pragmas, either; that causes iwyu.py problems. +          re.search(r'// IWYU pragma: ', line)) + + +def WrapLongLine(line, output): +  line = line.rstrip() +  if len(line) <= 80: +    output.append(line) +  elif IsSingleLineComment(line): +    if IsHeaderGuardIncludeOrOneLineIWYUPragma(line): +      # The style guide made an exception to allow long header guard lines, +      # includes and IWYU pragmas. +      output.append(line) +    else: +      WrapComment(line, output) +  elif IsInPreprocessorDirective(output, line): +    if IsHeaderGuardIncludeOrOneLineIWYUPragma(line): +      # The style guide made an exception to allow long header guard lines, +      # includes and IWYU pragmas. +      output.append(line) +    else: +      WrapPreprocessorDirective(line, output) +  elif IsMultiLineIWYUPragma(line): +    output.append(line) +  else: +    WrapPlainCode(line, output) + + +def BeautifyCode(string): +  lines = string.splitlines() +  output = [] +  for line in lines: +    WrapLongLine(line, output) +  output2 = [line.rstrip() for line in output] +  return '\n'.join(output2) + '\n' + + +def ConvertFromPumpSource(src_text): +  """Return the text generated from the given Pump source text.""" +  ast = ParseToAST(StripMetaComments(src_text)) +  output = Output() +  RunCode(Env(), ast, output) +  return BeautifyCode(output.string) + + +def main(argv): +  if len(argv) == 1: +    print(__doc__) +    sys.exit(1) + +  file_path = argv[-1] +  output_str = ConvertFromPumpSource(file(file_path, 'r').read()) +  if file_path.endswith('.pump'): +    output_file_path = file_path[:-5] +  else: +    output_file_path = '-' +  if output_file_path == '-': +    print(output_str,) +  else: +    output_file = file(output_file_path, 'w') +    output_file.write('// This file was GENERATED by command:\n') +    output_file.write('//     %s %s\n' % +                      (os.path.basename(__file__), os.path.basename(file_path))) +    output_file.write('// DO NOT EDIT BY HAND!!!\n\n') +    output_file.write(output_str) +    output_file.close() + + +if __name__ == '__main__': +  main(sys.argv) diff --git a/googlemock/scripts/upload.py b/googlemock/scripts/upload.py deleted file mode 100755 index 95239dc2..00000000 --- a/googlemock/scripts/upload.py +++ /dev/null @@ -1,1387 +0,0 @@ -#!/usr/bin/env python -# -# Copyright 2007 Google Inc. -# -# Licensed under the Apache License, Version 2.0 (the "License"); -# you may not use this file except in compliance with the License. -# You may obtain a copy of the License at -# -#     http://www.apache.org/licenses/LICENSE-2.0 -# -# Unless required by applicable law or agreed to in writing, software -# distributed under the License is distributed on an "AS IS" BASIS, -# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -# See the License for the specific language governing permissions and -# limitations under the License. - -"""Tool for uploading diffs from a version control system to the codereview app. - -Usage summary: upload.py [options] [-- diff_options] - -Diff options are passed to the diff command of the underlying system. - -Supported version control systems: -  Git -  Mercurial -  Subversion - -It is important for Git/Mercurial users to specify a tree/node/branch to diff -against by using the '--rev' option. -""" -# This code is derived from appcfg.py in the App Engine SDK (open source), -# and from ASPN recipe #146306. - -import cookielib -import getpass -import logging -import md5 -import mimetypes -import optparse -import os -import re -import socket -import subprocess -import sys -import urllib -import urllib2 -import urlparse - -try: -  import readline -except ImportError: -  pass - -# The logging verbosity: -#  0: Errors only. -#  1: Status messages. -#  2: Info logs. -#  3: Debug logs. -verbosity = 1 - -# Max size of patch or base file. -MAX_UPLOAD_SIZE = 900 * 1024 - - -def GetEmail(prompt): -  """Prompts the user for their email address and returns it. - -  The last used email address is saved to a file and offered up as a suggestion -  to the user. If the user presses enter without typing in anything the last -  used email address is used. If the user enters a new address, it is saved -  for next time we prompt. - -  """ -  last_email_file_name = os.path.expanduser("~/.last_codereview_email_address") -  last_email = "" -  if os.path.exists(last_email_file_name): -    try: -      last_email_file = open(last_email_file_name, "r") -      last_email = last_email_file.readline().strip("\n") -      last_email_file.close() -      prompt += " [%s]" % last_email -    except IOError, e: -      pass -  email = raw_input(prompt + ": ").strip() -  if email: -    try: -      last_email_file = open(last_email_file_name, "w") -      last_email_file.write(email) -      last_email_file.close() -    except IOError, e: -      pass -  else: -    email = last_email -  return email - - -def StatusUpdate(msg): -  """Print a status message to stdout. - -  If 'verbosity' is greater than 0, print the message. - -  Args: -    msg: The string to print. -  """ -  if verbosity > 0: -    print msg - - -def ErrorExit(msg): -  """Print an error message to stderr and exit.""" -  print >>sys.stderr, msg -  sys.exit(1) - - -class ClientLoginError(urllib2.HTTPError): -  """Raised to indicate there was an error authenticating with ClientLogin.""" - -  def __init__(self, url, code, msg, headers, args): -    urllib2.HTTPError.__init__(self, url, code, msg, headers, None) -    self.args = args -    self.reason = args["Error"] - - -class AbstractRpcServer(object): -  """Provides a common interface for a simple RPC server.""" - -  def __init__(self, host, auth_function, host_override=None, extra_headers={}, -               save_cookies=False): -    """Creates a new HttpRpcServer. - -    Args: -      host: The host to send requests to. -      auth_function: A function that takes no arguments and returns an -        (email, password) tuple when called. Will be called if authentication -        is required. -      host_override: The host header to send to the server (defaults to host). -      extra_headers: A dict of extra headers to append to every request. -      save_cookies: If True, save the authentication cookies to local disk. -        If False, use an in-memory cookiejar instead.  Subclasses must -        implement this functionality.  Defaults to False. -    """ -    self.host = host -    self.host_override = host_override -    self.auth_function = auth_function -    self.authenticated = False -    self.extra_headers = extra_headers -    self.save_cookies = save_cookies -    self.opener = self._GetOpener() -    if self.host_override: -      logging.info("Server: %s; Host: %s", self.host, self.host_override) -    else: -      logging.info("Server: %s", self.host) - -  def _GetOpener(self): -    """Returns an OpenerDirector for making HTTP requests. - -    Returns: -      A urllib2.OpenerDirector object. -    """ -    raise NotImplementedError() - -  def _CreateRequest(self, url, data=None): -    """Creates a new urllib request.""" -    logging.debug("Creating request for: '%s' with payload:\n%s", url, data) -    req = urllib2.Request(url, data=data) -    if self.host_override: -      req.add_header("Host", self.host_override) -    for key, value in self.extra_headers.iteritems(): -      req.add_header(key, value) -    return req - -  def _GetAuthToken(self, email, password): -    """Uses ClientLogin to authenticate the user, returning an auth token. - -    Args: -      email:    The user's email address -      password: The user's password - -    Raises: -      ClientLoginError: If there was an error authenticating with ClientLogin. -      HTTPError: If there was some other form of HTTP error. - -    Returns: -      The authentication token returned by ClientLogin. -    """ -    account_type = "GOOGLE" -    if self.host.endswith(".google.com"): -      # Needed for use inside Google. -      account_type = "HOSTED" -    req = self._CreateRequest( -        url="https://www.google.com/accounts/ClientLogin", -        data=urllib.urlencode({ -            "Email": email, -            "Passwd": password, -            "service": "ah", -            "source": "rietveld-codereview-upload", -            "accountType": account_type, -        }), -    ) -    try: -      response = self.opener.open(req) -      response_body = response.read() -      response_dict = dict(x.split("=") -                           for x in response_body.split("\n") if x) -      return response_dict["Auth"] -    except urllib2.HTTPError, e: -      if e.code == 403: -        body = e.read() -        response_dict = dict(x.split("=", 1) for x in body.split("\n") if x) -        raise ClientLoginError(req.get_full_url(), e.code, e.msg, -                               e.headers, response_dict) -      else: -        raise - -  def _GetAuthCookie(self, auth_token): -    """Fetches authentication cookies for an authentication token. - -    Args: -      auth_token: The authentication token returned by ClientLogin. - -    Raises: -      HTTPError: If there was an error fetching the authentication cookies. -    """ -    # This is a dummy value to allow us to identify when we're successful. -    continue_location = "http://localhost/" -    args = {"continue": continue_location, "auth": auth_token} -    req = self._CreateRequest("http://%s/_ah/login?%s" % -                              (self.host, urllib.urlencode(args))) -    try: -      response = self.opener.open(req) -    except urllib2.HTTPError, e: -      response = e -    if (response.code != 302 or -        response.info()["location"] != continue_location): -      raise urllib2.HTTPError(req.get_full_url(), response.code, response.msg, -                              response.headers, response.fp) -    self.authenticated = True - -  def _Authenticate(self): -    """Authenticates the user. - -    The authentication process works as follows: -     1) We get a username and password from the user -     2) We use ClientLogin to obtain an AUTH token for the user -        (see https://developers.google.com/identity/protocols/AuthForInstalledApps). -     3) We pass the auth token to /_ah/login on the server to obtain an -        authentication cookie. If login was successful, it tries to redirect -        us to the URL we provided. - -    If we attempt to access the upload API without first obtaining an -    authentication cookie, it returns a 401 response and directs us to -    authenticate ourselves with ClientLogin. -    """ -    for i in range(3): -      credentials = self.auth_function() -      try: -        auth_token = self._GetAuthToken(credentials[0], credentials[1]) -      except ClientLoginError, e: -        if e.reason == "BadAuthentication": -          print >>sys.stderr, "Invalid username or password." -          continue -        if e.reason == "CaptchaRequired": -          print >>sys.stderr, ( -              "Please go to\n" -              "https://www.google.com/accounts/DisplayUnlockCaptcha\n" -              "and verify you are a human.  Then try again.") -          break -        if e.reason == "NotVerified": -          print >>sys.stderr, "Account not verified." -          break -        if e.reason == "TermsNotAgreed": -          print >>sys.stderr, "User has not agreed to TOS." -          break -        if e.reason == "AccountDeleted": -          print >>sys.stderr, "The user account has been deleted." -          break -        if e.reason == "AccountDisabled": -          print >>sys.stderr, "The user account has been disabled." -          break -        if e.reason == "ServiceDisabled": -          print >>sys.stderr, ("The user's access to the service has been " -                               "disabled.") -          break -        if e.reason == "ServiceUnavailable": -          print >>sys.stderr, "The service is not available; try again later." -          break -        raise -      self._GetAuthCookie(auth_token) -      return - -  def Send(self, request_path, payload=None, -           content_type="application/octet-stream", -           timeout=None, -           **kwargs): -    """Sends an RPC and returns the response. - -    Args: -      request_path: The path to send the request to, eg /api/appversion/create. -      payload: The body of the request, or None to send an empty request. -      content_type: The Content-Type header to use. -      timeout: timeout in seconds; default None i.e. no timeout. -        (Note: for large requests on OS X, the timeout doesn't work right.) -      kwargs: Any keyword arguments are converted into query string parameters. - -    Returns: -      The response body, as a string. -    """ -    # TODO: Don't require authentication.  Let the server say -    # whether it is necessary. -    if not self.authenticated: -      self._Authenticate() - -    old_timeout = socket.getdefaulttimeout() -    socket.setdefaulttimeout(timeout) -    try: -      tries = 0 -      while True: -        tries += 1 -        args = dict(kwargs) -        url = "http://%s%s" % (self.host, request_path) -        if args: -          url += "?" + urllib.urlencode(args) -        req = self._CreateRequest(url=url, data=payload) -        req.add_header("Content-Type", content_type) -        try: -          f = self.opener.open(req) -          response = f.read() -          f.close() -          return response -        except urllib2.HTTPError, e: -          if tries > 3: -            raise -          elif e.code == 401: -            self._Authenticate() -##           elif e.code >= 500 and e.code < 600: -##             # Server Error - try again. -##             continue -          else: -            raise -    finally: -      socket.setdefaulttimeout(old_timeout) - - -class HttpRpcServer(AbstractRpcServer): -  """Provides a simplified RPC-style interface for HTTP requests.""" - -  def _Authenticate(self): -    """Save the cookie jar after authentication.""" -    super(HttpRpcServer, self)._Authenticate() -    if self.save_cookies: -      StatusUpdate("Saving authentication cookies to %s" % self.cookie_file) -      self.cookie_jar.save() - -  def _GetOpener(self): -    """Returns an OpenerDirector that supports cookies and ignores redirects. - -    Returns: -      A urllib2.OpenerDirector object. -    """ -    opener = urllib2.OpenerDirector() -    opener.add_handler(urllib2.ProxyHandler()) -    opener.add_handler(urllib2.UnknownHandler()) -    opener.add_handler(urllib2.HTTPHandler()) -    opener.add_handler(urllib2.HTTPDefaultErrorHandler()) -    opener.add_handler(urllib2.HTTPSHandler()) -    opener.add_handler(urllib2.HTTPErrorProcessor()) -    if self.save_cookies: -      self.cookie_file = os.path.expanduser("~/.codereview_upload_cookies") -      self.cookie_jar = cookielib.MozillaCookieJar(self.cookie_file) -      if os.path.exists(self.cookie_file): -        try: -          self.cookie_jar.load() -          self.authenticated = True -          StatusUpdate("Loaded authentication cookies from %s" % -                       self.cookie_file) -        except (cookielib.LoadError, IOError): -          # Failed to load cookies - just ignore them. -          pass -      else: -        # Create an empty cookie file with mode 600 -        fd = os.open(self.cookie_file, os.O_CREAT, 0600) -        os.close(fd) -      # Always chmod the cookie file -      os.chmod(self.cookie_file, 0600) -    else: -      # Don't save cookies across runs of update.py. -      self.cookie_jar = cookielib.CookieJar() -    opener.add_handler(urllib2.HTTPCookieProcessor(self.cookie_jar)) -    return opener - - -parser = optparse.OptionParser(usage="%prog [options] [-- diff_options]") -parser.add_option("-y", "--assume_yes", action="store_true", -                  dest="assume_yes", default=False, -                  help="Assume that the answer to yes/no questions is 'yes'.") -# Logging -group = parser.add_option_group("Logging options") -group.add_option("-q", "--quiet", action="store_const", const=0, -                 dest="verbose", help="Print errors only.") -group.add_option("-v", "--verbose", action="store_const", const=2, -                 dest="verbose", default=1, -                 help="Print info level logs (default).") -group.add_option("--noisy", action="store_const", const=3, -                 dest="verbose", help="Print all logs.") -# Review server -group = parser.add_option_group("Review server options") -group.add_option("-s", "--server", action="store", dest="server", -                 default="codereview.appspot.com", -                 metavar="SERVER", -                 help=("The server to upload to. The format is host[:port]. " -                       "Defaults to 'codereview.appspot.com'.")) -group.add_option("-e", "--email", action="store", dest="email", -                 metavar="EMAIL", default=None, -                 help="The username to use. Will prompt if omitted.") -group.add_option("-H", "--host", action="store", dest="host", -                 metavar="HOST", default=None, -                 help="Overrides the Host header sent with all RPCs.") -group.add_option("--no_cookies", action="store_false", -                 dest="save_cookies", default=True, -                 help="Do not save authentication cookies to local disk.") -# Issue -group = parser.add_option_group("Issue options") -group.add_option("-d", "--description", action="store", dest="description", -                 metavar="DESCRIPTION", default=None, -                 help="Optional description when creating an issue.") -group.add_option("-f", "--description_file", action="store", -                 dest="description_file", metavar="DESCRIPTION_FILE", -                 default=None, -                 help="Optional path of a file that contains " -                      "the description when creating an issue.") -group.add_option("-r", "--reviewers", action="store", dest="reviewers", -                 metavar="REVIEWERS", default=None, -                 help="Add reviewers (comma separated email addresses).") -group.add_option("--cc", action="store", dest="cc", -                 metavar="CC", default=None, -                 help="Add CC (comma separated email addresses).") -# Upload options -group = parser.add_option_group("Patch options") -group.add_option("-m", "--message", action="store", dest="message", -                 metavar="MESSAGE", default=None, -                 help="A message to identify the patch. " -                      "Will prompt if omitted.") -group.add_option("-i", "--issue", type="int", action="store", -                 metavar="ISSUE", default=None, -                 help="Issue number to which to add. Defaults to new issue.") -group.add_option("--download_base", action="store_true", -                 dest="download_base", default=False, -                 help="Base files will be downloaded by the server " -                 "(side-by-side diffs may not work on files with CRs).") -group.add_option("--rev", action="store", dest="revision", -                 metavar="REV", default=None, -                 help="Branch/tree/revision to diff against (used by DVCS).") -group.add_option("--send_mail", action="store_true", -                 dest="send_mail", default=False, -                 help="Send notification email to reviewers.") - - -def GetRpcServer(options): -  """Returns an instance of an AbstractRpcServer. - -  Returns: -    A new AbstractRpcServer, on which RPC calls can be made. -  """ - -  rpc_server_class = HttpRpcServer - -  def GetUserCredentials(): -    """Prompts the user for a username and password.""" -    email = options.email -    if email is None: -      email = GetEmail("Email (login for uploading to %s)" % options.server) -    password = getpass.getpass("Password for %s: " % email) -    return (email, password) - -  # If this is the dev_appserver, use fake authentication. -  host = (options.host or options.server).lower() -  if host == "localhost" or host.startswith("localhost:"): -    email = options.email -    if email is None: -      email = "test@example.com" -      logging.info("Using debug user %s.  Override with --email" % email) -    server = rpc_server_class( -        options.server, -        lambda: (email, "password"), -        host_override=options.host, -        extra_headers={"Cookie": -                       'dev_appserver_login="%s:False"' % email}, -        save_cookies=options.save_cookies) -    # Don't try to talk to ClientLogin. -    server.authenticated = True -    return server - -  return rpc_server_class(options.server, GetUserCredentials, -                          host_override=options.host, -                          save_cookies=options.save_cookies) - - -def EncodeMultipartFormData(fields, files): -  """Encode form fields for multipart/form-data. - -  Args: -    fields: A sequence of (name, value) elements for regular form fields. -    files: A sequence of (name, filename, value) elements for data to be -           uploaded as files. -  Returns: -    (content_type, body) ready for httplib.HTTP instance. - -  Source: -    https://web.archive.org/web/20160116052001/code.activestate.com/recipes/146306 -  """ -  BOUNDARY = '-M-A-G-I-C---B-O-U-N-D-A-R-Y-' -  CRLF = '\r\n' -  lines = [] -  for (key, value) in fields: -    lines.append('--' + BOUNDARY) -    lines.append('Content-Disposition: form-data; name="%s"' % key) -    lines.append('') -    lines.append(value) -  for (key, filename, value) in files: -    lines.append('--' + BOUNDARY) -    lines.append('Content-Disposition: form-data; name="%s"; filename="%s"' % -             (key, filename)) -    lines.append('Content-Type: %s' % GetContentType(filename)) -    lines.append('') -    lines.append(value) -  lines.append('--' + BOUNDARY + '--') -  lines.append('') -  body = CRLF.join(lines) -  content_type = 'multipart/form-data; boundary=%s' % BOUNDARY -  return content_type, body - - -def GetContentType(filename): -  """Helper to guess the content-type from the filename.""" -  return mimetypes.guess_type(filename)[0] or 'application/octet-stream' - - -# Use a shell for subcommands on Windows to get a PATH search. -use_shell = sys.platform.startswith("win") - -def RunShellWithReturnCode(command, print_output=False, -                           universal_newlines=True): -  """Executes a command and returns the output from stdout and the return code. - -  Args: -    command: Command to execute. -    print_output: If True, the output is printed to stdout. -                  If False, both stdout and stderr are ignored. -    universal_newlines: Use universal_newlines flag (default: True). - -  Returns: -    Tuple (output, return code) -  """ -  logging.info("Running %s", command) -  p = subprocess.Popen(command, stdout=subprocess.PIPE, stderr=subprocess.PIPE, -                       shell=use_shell, universal_newlines=universal_newlines) -  if print_output: -    output_array = [] -    while True: -      line = p.stdout.readline() -      if not line: -        break -      print line.strip("\n") -      output_array.append(line) -    output = "".join(output_array) -  else: -    output = p.stdout.read() -  p.wait() -  errout = p.stderr.read() -  if print_output and errout: -    print >>sys.stderr, errout -  p.stdout.close() -  p.stderr.close() -  return output, p.returncode - - -def RunShell(command, silent_ok=False, universal_newlines=True, -             print_output=False): -  data, retcode = RunShellWithReturnCode(command, print_output, -                                         universal_newlines) -  if retcode: -    ErrorExit("Got error status from %s:\n%s" % (command, data)) -  if not silent_ok and not data: -    ErrorExit("No output from %s" % command) -  return data - - -class VersionControlSystem(object): -  """Abstract base class providing an interface to the VCS.""" - -  def __init__(self, options): -    """Constructor. - -    Args: -      options: Command line options. -    """ -    self.options = options - -  def GenerateDiff(self, args): -    """Return the current diff as a string. - -    Args: -      args: Extra arguments to pass to the diff command. -    """ -    raise NotImplementedError( -        "abstract method -- subclass %s must override" % self.__class__) - -  def GetUnknownFiles(self): -    """Return a list of files unknown to the VCS.""" -    raise NotImplementedError( -        "abstract method -- subclass %s must override" % self.__class__) - -  def CheckForUnknownFiles(self): -    """Show an "are you sure?" prompt if there are unknown files.""" -    unknown_files = self.GetUnknownFiles() -    if unknown_files: -      print "The following files are not added to version control:" -      for line in unknown_files: -        print line -      prompt = "Are you sure to continue?(y/N) " -      answer = raw_input(prompt).strip() -      if answer != "y": -        ErrorExit("User aborted") - -  def GetBaseFile(self, filename): -    """Get the content of the upstream version of a file. - -    Returns: -      A tuple (base_content, new_content, is_binary, status) -        base_content: The contents of the base file. -        new_content: For text files, this is empty.  For binary files, this is -          the contents of the new file, since the diff output won't contain -          information to reconstruct the current file. -        is_binary: True iff the file is binary. -        status: The status of the file. -    """ - -    raise NotImplementedError( -        "abstract method -- subclass %s must override" % self.__class__) - - -  def GetBaseFiles(self, diff): -    """Helper that calls GetBase file for each file in the patch. - -    Returns: -      A dictionary that maps from filename to GetBaseFile's tuple.  Filenames -      are retrieved based on lines that start with "Index:" or -      "Property changes on:". -    """ -    files = {} -    for line in diff.splitlines(True): -      if line.startswith('Index:') or line.startswith('Property changes on:'): -        unused, filename = line.split(':', 1) -        # On Windows if a file has property changes its filename uses '\' -        # instead of '/'. -        filename = filename.strip().replace('\\', '/') -        files[filename] = self.GetBaseFile(filename) -    return files - - -  def UploadBaseFiles(self, issue, rpc_server, patch_list, patchset, options, -                      files): -    """Uploads the base files (and if necessary, the current ones as well).""" - -    def UploadFile(filename, file_id, content, is_binary, status, is_base): -      """Uploads a file to the server.""" -      file_too_large = False -      if is_base: -        type = "base" -      else: -        type = "current" -      if len(content) > MAX_UPLOAD_SIZE: -        print ("Not uploading the %s file for %s because it's too large." % -               (type, filename)) -        file_too_large = True -        content = "" -      checksum = md5.new(content).hexdigest() -      if options.verbose > 0 and not file_too_large: -        print "Uploading %s file for %s" % (type, filename) -      url = "/%d/upload_content/%d/%d" % (int(issue), int(patchset), file_id) -      form_fields = [("filename", filename), -                     ("status", status), -                     ("checksum", checksum), -                     ("is_binary", str(is_binary)), -                     ("is_current", str(not is_base)), -                    ] -      if file_too_large: -        form_fields.append(("file_too_large", "1")) -      if options.email: -        form_fields.append(("user", options.email)) -      ctype, body = EncodeMultipartFormData(form_fields, -                                            [("data", filename, content)]) -      response_body = rpc_server.Send(url, body, -                                      content_type=ctype) -      if not response_body.startswith("OK"): -        StatusUpdate("  --> %s" % response_body) -        sys.exit(1) - -    patches = dict() -    [patches.setdefault(v, k) for k, v in patch_list] -    for filename in patches.keys(): -      base_content, new_content, is_binary, status = files[filename] -      file_id_str = patches.get(filename) -      if file_id_str.find("nobase") != -1: -        base_content = None -        file_id_str = file_id_str[file_id_str.rfind("_") + 1:] -      file_id = int(file_id_str) -      if base_content != None: -        UploadFile(filename, file_id, base_content, is_binary, status, True) -      if new_content != None: -        UploadFile(filename, file_id, new_content, is_binary, status, False) - -  def IsImage(self, filename): -    """Returns true if the filename has an image extension.""" -    mimetype =  mimetypes.guess_type(filename)[0] -    if not mimetype: -      return False -    return mimetype.startswith("image/") - - -class SubversionVCS(VersionControlSystem): -  """Implementation of the VersionControlSystem interface for Subversion.""" - -  def __init__(self, options): -    super(SubversionVCS, self).__init__(options) -    if self.options.revision: -      match = re.match(r"(\d+)(:(\d+))?", self.options.revision) -      if not match: -        ErrorExit("Invalid Subversion revision %s." % self.options.revision) -      self.rev_start = match.group(1) -      self.rev_end = match.group(3) -    else: -      self.rev_start = self.rev_end = None -    # Cache output from "svn list -r REVNO dirname". -    # Keys: dirname, Values: 2-tuple (ouput for start rev and end rev). -    self.svnls_cache = {} -    # SVN base URL is required to fetch files deleted in an older revision. -    # Result is cached to not guess it over and over again in GetBaseFile(). -    required = self.options.download_base or self.options.revision is not None -    self.svn_base = self._GuessBase(required) - -  def GuessBase(self, required): -    """Wrapper for _GuessBase.""" -    return self.svn_base - -  def _GuessBase(self, required): -    """Returns the SVN base URL. - -    Args: -      required: If true, exits if the url can't be guessed, otherwise None is -        returned. -    """ -    info = RunShell(["svn", "info"]) -    for line in info.splitlines(): -      words = line.split() -      if len(words) == 2 and words[0] == "URL:": -        url = words[1] -        scheme, netloc, path, params, query, fragment = urlparse.urlparse(url) -        username, netloc = urllib.splituser(netloc) -        if username: -          logging.info("Removed username from base URL") -        if netloc.endswith("svn.python.org"): -          if netloc == "svn.python.org": -            if path.startswith("/projects/"): -              path = path[9:] -          elif netloc != "pythondev@svn.python.org": -            ErrorExit("Unrecognized Python URL: %s" % url) -          base = "http://svn.python.org/view/*checkout*%s/" % path -          logging.info("Guessed Python base = %s", base) -        elif netloc.endswith("svn.collab.net"): -          if path.startswith("/repos/"): -            path = path[6:] -          base = "http://svn.collab.net/viewvc/*checkout*%s/" % path -          logging.info("Guessed CollabNet base = %s", base) -        elif netloc.endswith(".googlecode.com"): -          path = path + "/" -          base = urlparse.urlunparse(("http", netloc, path, params, -                                      query, fragment)) -          logging.info("Guessed Google Code base = %s", base) -        else: -          path = path + "/" -          base = urlparse.urlunparse((scheme, netloc, path, params, -                                      query, fragment)) -          logging.info("Guessed base = %s", base) -        return base -    if required: -      ErrorExit("Can't find URL in output from svn info") -    return None - -  def GenerateDiff(self, args): -    cmd = ["svn", "diff"] -    if self.options.revision: -      cmd += ["-r", self.options.revision] -    cmd.extend(args) -    data = RunShell(cmd) -    count = 0 -    for line in data.splitlines(): -      if line.startswith("Index:") or line.startswith("Property changes on:"): -        count += 1 -        logging.info(line) -    if not count: -      ErrorExit("No valid patches found in output from svn diff") -    return data - -  def _CollapseKeywords(self, content, keyword_str): -    """Collapses SVN keywords.""" -    # svn cat translates keywords but svn diff doesn't. As a result of this -    # behavior patching.PatchChunks() fails with a chunk mismatch error. -    # This part was originally written by the Review Board development team -    # who had the same problem (https://reviews.reviewboard.org/r/276/). -    # Mapping of keywords to known aliases -    svn_keywords = { -      # Standard keywords -      'Date':                ['Date', 'LastChangedDate'], -      'Revision':            ['Revision', 'LastChangedRevision', 'Rev'], -      'Author':              ['Author', 'LastChangedBy'], -      'HeadURL':             ['HeadURL', 'URL'], -      'Id':                  ['Id'], - -      # Aliases -      'LastChangedDate':     ['LastChangedDate', 'Date'], -      'LastChangedRevision': ['LastChangedRevision', 'Rev', 'Revision'], -      'LastChangedBy':       ['LastChangedBy', 'Author'], -      'URL':                 ['URL', 'HeadURL'], -    } - -    def repl(m): -       if m.group(2): -         return "$%s::%s$" % (m.group(1), " " * len(m.group(3))) -       return "$%s$" % m.group(1) -    keywords = [keyword -                for name in keyword_str.split(" ") -                for keyword in svn_keywords.get(name, [])] -    return re.sub(r"\$(%s):(:?)([^\$]+)\$" % '|'.join(keywords), repl, content) - -  def GetUnknownFiles(self): -    status = RunShell(["svn", "status", "--ignore-externals"], silent_ok=True) -    unknown_files = [] -    for line in status.split("\n"): -      if line and line[0] == "?": -        unknown_files.append(line) -    return unknown_files - -  def ReadFile(self, filename): -    """Returns the contents of a file.""" -    file = open(filename, 'rb') -    result = "" -    try: -      result = file.read() -    finally: -      file.close() -    return result - -  def GetStatus(self, filename): -    """Returns the status of a file.""" -    if not self.options.revision: -      status = RunShell(["svn", "status", "--ignore-externals", filename]) -      if not status: -        ErrorExit("svn status returned no output for %s" % filename) -      status_lines = status.splitlines() -      # If file is in a cl, the output will begin with -      # "\n--- Changelist 'cl_name':\n".  See -      # https://web.archive.org/web/20090918234815/svn.collab.net/repos/svn/trunk/notes/changelist-design.txt -      if (len(status_lines) == 3 and -          not status_lines[0] and -          status_lines[1].startswith("--- Changelist")): -        status = status_lines[2] -      else: -        status = status_lines[0] -    # If we have a revision to diff against we need to run "svn list" -    # for the old and the new revision and compare the results to get -    # the correct status for a file. -    else: -      dirname, relfilename = os.path.split(filename) -      if dirname not in self.svnls_cache: -        cmd = ["svn", "list", "-r", self.rev_start, dirname or "."] -        out, returncode = RunShellWithReturnCode(cmd) -        if returncode: -          ErrorExit("Failed to get status for %s." % filename) -        old_files = out.splitlines() -        args = ["svn", "list"] -        if self.rev_end: -          args += ["-r", self.rev_end] -        cmd = args + [dirname or "."] -        out, returncode = RunShellWithReturnCode(cmd) -        if returncode: -          ErrorExit("Failed to run command %s" % cmd) -        self.svnls_cache[dirname] = (old_files, out.splitlines()) -      old_files, new_files = self.svnls_cache[dirname] -      if relfilename in old_files and relfilename not in new_files: -        status = "D   " -      elif relfilename in old_files and relfilename in new_files: -        status = "M   " -      else: -        status = "A   " -    return status - -  def GetBaseFile(self, filename): -    status = self.GetStatus(filename) -    base_content = None -    new_content = None - -    # If a file is copied its status will be "A  +", which signifies -    # "addition-with-history".  See "svn st" for more information.  We need to -    # upload the original file or else diff parsing will fail if the file was -    # edited. -    if status[0] == "A" and status[3] != "+": -      # We'll need to upload the new content if we're adding a binary file -      # since diff's output won't contain it. -      mimetype = RunShell(["svn", "propget", "svn:mime-type", filename], -                          silent_ok=True) -      base_content = "" -      is_binary = mimetype and not mimetype.startswith("text/") -      if is_binary and self.IsImage(filename): -        new_content = self.ReadFile(filename) -    elif (status[0] in ("M", "D", "R") or -          (status[0] == "A" and status[3] == "+") or  # Copied file. -          (status[0] == " " and status[1] == "M")):  # Property change. -      args = [] -      if self.options.revision: -        url = "%s/%s@%s" % (self.svn_base, filename, self.rev_start) -      else: -        # Don't change filename, it's needed later. -        url = filename -        args += ["-r", "BASE"] -      cmd = ["svn"] + args + ["propget", "svn:mime-type", url] -      mimetype, returncode = RunShellWithReturnCode(cmd) -      if returncode: -        # File does not exist in the requested revision. -        # Reset mimetype, it contains an error message. -        mimetype = "" -      get_base = False -      is_binary = mimetype and not mimetype.startswith("text/") -      if status[0] == " ": -        # Empty base content just to force an upload. -        base_content = "" -      elif is_binary: -        if self.IsImage(filename): -          get_base = True -          if status[0] == "M": -            if not self.rev_end: -              new_content = self.ReadFile(filename) -            else: -              url = "%s/%s@%s" % (self.svn_base, filename, self.rev_end) -              new_content = RunShell(["svn", "cat", url], -                                     universal_newlines=True, silent_ok=True) -        else: -          base_content = "" -      else: -        get_base = True - -      if get_base: -        if is_binary: -          universal_newlines = False -        else: -          universal_newlines = True -        if self.rev_start: -          # "svn cat -r REV delete_file.txt" doesn't work. cat requires -          # the full URL with "@REV" appended instead of using "-r" option. -          url = "%s/%s@%s" % (self.svn_base, filename, self.rev_start) -          base_content = RunShell(["svn", "cat", url], -                                  universal_newlines=universal_newlines, -                                  silent_ok=True) -        else: -          base_content = RunShell(["svn", "cat", filename], -                                  universal_newlines=universal_newlines, -                                  silent_ok=True) -        if not is_binary: -          args = [] -          if self.rev_start: -            url = "%s/%s@%s" % (self.svn_base, filename, self.rev_start) -          else: -            url = filename -            args += ["-r", "BASE"] -          cmd = ["svn"] + args + ["propget", "svn:keywords", url] -          keywords, returncode = RunShellWithReturnCode(cmd) -          if keywords and not returncode: -            base_content = self._CollapseKeywords(base_content, keywords) -    else: -      StatusUpdate("svn status returned unexpected output: %s" % status) -      sys.exit(1) -    return base_content, new_content, is_binary, status[0:5] - - -class GitVCS(VersionControlSystem): -  """Implementation of the VersionControlSystem interface for Git.""" - -  def __init__(self, options): -    super(GitVCS, self).__init__(options) -    # Map of filename -> hash of base file. -    self.base_hashes = {} - -  def GenerateDiff(self, extra_args): -    # This is more complicated than svn's GenerateDiff because we must convert -    # the diff output to include an svn-style "Index:" line as well as record -    # the hashes of the base files, so we can upload them along with our diff. -    if self.options.revision: -      extra_args = [self.options.revision] + extra_args -    gitdiff = RunShell(["git", "diff", "--full-index"] + extra_args) -    svndiff = [] -    filecount = 0 -    filename = None -    for line in gitdiff.splitlines(): -      match = re.match(r"diff --git a/(.*) b/.*$", line) -      if match: -        filecount += 1 -        filename = match.group(1) -        svndiff.append("Index: %s\n" % filename) -      else: -        # The "index" line in a git diff looks like this (long hashes elided): -        #   index 82c0d44..b2cee3f 100755 -        # We want to save the left hash, as that identifies the base file. -        match = re.match(r"index (\w+)\.\.", line) -        if match: -          self.base_hashes[filename] = match.group(1) -      svndiff.append(line + "\n") -    if not filecount: -      ErrorExit("No valid patches found in output from git diff") -    return "".join(svndiff) - -  def GetUnknownFiles(self): -    status = RunShell(["git", "ls-files", "--exclude-standard", "--others"], -                      silent_ok=True) -    return status.splitlines() - -  def GetBaseFile(self, filename): -    hash = self.base_hashes[filename] -    base_content = None -    new_content = None -    is_binary = False -    if hash == "0" * 40:  # All-zero hash indicates no base file. -      status = "A" -      base_content = "" -    else: -      status = "M" -      base_content, returncode = RunShellWithReturnCode(["git", "show", hash]) -      if returncode: -        ErrorExit("Got error status from 'git show %s'" % hash) -    return (base_content, new_content, is_binary, status) - - -class MercurialVCS(VersionControlSystem): -  """Implementation of the VersionControlSystem interface for Mercurial.""" - -  def __init__(self, options, repo_dir): -    super(MercurialVCS, self).__init__(options) -    # Absolute path to repository (we can be in a subdir) -    self.repo_dir = os.path.normpath(repo_dir) -    # Compute the subdir -    cwd = os.path.normpath(os.getcwd()) -    assert cwd.startswith(self.repo_dir) -    self.subdir = cwd[len(self.repo_dir):].lstrip(r"\/") -    if self.options.revision: -      self.base_rev = self.options.revision -    else: -      self.base_rev = RunShell(["hg", "parent", "-q"]).split(':')[1].strip() - -  def _GetRelPath(self, filename): -    """Get relative path of a file according to the current directory, -    given its logical path in the repo.""" -    assert filename.startswith(self.subdir), filename -    return filename[len(self.subdir):].lstrip(r"\/") - -  def GenerateDiff(self, extra_args): -    # If no file specified, restrict to the current subdir -    extra_args = extra_args or ["."] -    cmd = ["hg", "diff", "--git", "-r", self.base_rev] + extra_args -    data = RunShell(cmd, silent_ok=True) -    svndiff = [] -    filecount = 0 -    for line in data.splitlines(): -      m = re.match("diff --git a/(\S+) b/(\S+)", line) -      if m: -        # Modify line to make it look like as it comes from svn diff. -        # With this modification no changes on the server side are required -        # to make upload.py work with Mercurial repos. -        # NOTE: for proper handling of moved/copied files, we have to use -        # the second filename. -        filename = m.group(2) -        svndiff.append("Index: %s" % filename) -        svndiff.append("=" * 67) -        filecount += 1 -        logging.info(line) -      else: -        svndiff.append(line) -    if not filecount: -      ErrorExit("No valid patches found in output from hg diff") -    return "\n".join(svndiff) + "\n" - -  def GetUnknownFiles(self): -    """Return a list of files unknown to the VCS.""" -    args = [] -    status = RunShell(["hg", "status", "--rev", self.base_rev, "-u", "."], -        silent_ok=True) -    unknown_files = [] -    for line in status.splitlines(): -      st, fn = line.split(" ", 1) -      if st == "?": -        unknown_files.append(fn) -    return unknown_files - -  def GetBaseFile(self, filename): -    # "hg status" and "hg cat" both take a path relative to the current subdir -    # rather than to the repo root, but "hg diff" has given us the full path -    # to the repo root. -    base_content = "" -    new_content = None -    is_binary = False -    oldrelpath = relpath = self._GetRelPath(filename) -    # "hg status -C" returns two lines for moved/copied files, one otherwise -    out = RunShell(["hg", "status", "-C", "--rev", self.base_rev, relpath]) -    out = out.splitlines() -    # HACK: strip error message about missing file/directory if it isn't in -    # the working copy -    if out[0].startswith('%s: ' % relpath): -      out = out[1:] -    if len(out) > 1: -      # Moved/copied => considered as modified, use old filename to -      # retrieve base contents -      oldrelpath = out[1].strip() -      status = "M" -    else: -      status, _ = out[0].split(' ', 1) -    if status != "A": -      base_content = RunShell(["hg", "cat", "-r", self.base_rev, oldrelpath], -        silent_ok=True) -      is_binary = "\0" in base_content  # Mercurial's heuristic -    if status != "R": -      new_content = open(relpath, "rb").read() -      is_binary = is_binary or "\0" in new_content -    if is_binary and base_content: -      # Fetch again without converting newlines -      base_content = RunShell(["hg", "cat", "-r", self.base_rev, oldrelpath], -        silent_ok=True, universal_newlines=False) -    if not is_binary or not self.IsImage(relpath): -      new_content = None -    return base_content, new_content, is_binary, status - - -# NOTE: The SplitPatch function is duplicated in engine.py, keep them in sync. -def SplitPatch(data): -  """Splits a patch into separate pieces for each file. - -  Args: -    data: A string containing the output of svn diff. - -  Returns: -    A list of 2-tuple (filename, text) where text is the svn diff output -      pertaining to filename. -  """ -  patches = [] -  filename = None -  diff = [] -  for line in data.splitlines(True): -    new_filename = None -    if line.startswith('Index:'): -      unused, new_filename = line.split(':', 1) -      new_filename = new_filename.strip() -    elif line.startswith('Property changes on:'): -      unused, temp_filename = line.split(':', 1) -      # When a file is modified, paths use '/' between directories, however -      # when a property is modified '\' is used on Windows.  Make them the same -      # otherwise the file shows up twice. -      temp_filename = temp_filename.strip().replace('\\', '/') -      if temp_filename != filename: -        # File has property changes but no modifications, create a new diff. -        new_filename = temp_filename -    if new_filename: -      if filename and diff: -        patches.append((filename, ''.join(diff))) -      filename = new_filename -      diff = [line] -      continue -    if diff is not None: -      diff.append(line) -  if filename and diff: -    patches.append((filename, ''.join(diff))) -  return patches - - -def UploadSeparatePatches(issue, rpc_server, patchset, data, options): -  """Uploads a separate patch for each file in the diff output. - -  Returns a list of [patch_key, filename] for each file. -  """ -  patches = SplitPatch(data) -  rv = [] -  for patch in patches: -    if len(patch[1]) > MAX_UPLOAD_SIZE: -      print ("Not uploading the patch for " + patch[0] + -             " because the file is too large.") -      continue -    form_fields = [("filename", patch[0])] -    if not options.download_base: -      form_fields.append(("content_upload", "1")) -    files = [("data", "data.diff", patch[1])] -    ctype, body = EncodeMultipartFormData(form_fields, files) -    url = "/%d/upload_patch/%d" % (int(issue), int(patchset)) -    print "Uploading patch for " + patch[0] -    response_body = rpc_server.Send(url, body, content_type=ctype) -    lines = response_body.splitlines() -    if not lines or lines[0] != "OK": -      StatusUpdate("  --> %s" % response_body) -      sys.exit(1) -    rv.append([lines[1], patch[0]]) -  return rv - - -def GuessVCS(options): -  """Helper to guess the version control system. - -  This examines the current directory, guesses which VersionControlSystem -  we're using, and returns an instance of the appropriate class.  Exit with an -  error if we can't figure it out. - -  Returns: -    A VersionControlSystem instance. Exits if the VCS can't be guessed. -  """ -  # Mercurial has a command to get the base directory of a repository -  # Try running it, but don't die if we don't have hg installed. -  # NOTE: we try Mercurial first as it can sit on top of an SVN working copy. -  try: -    out, returncode = RunShellWithReturnCode(["hg", "root"]) -    if returncode == 0: -      return MercurialVCS(options, out.strip()) -  except OSError, (errno, message): -    if errno != 2:  # ENOENT -- they don't have hg installed. -      raise - -  # Subversion has a .svn in all working directories. -  if os.path.isdir('.svn'): -    logging.info("Guessed VCS = Subversion") -    return SubversionVCS(options) - -  # Git has a command to test if you're in a git tree. -  # Try running it, but don't die if we don't have git installed. -  try: -    out, returncode = RunShellWithReturnCode(["git", "rev-parse", -                                              "--is-inside-work-tree"]) -    if returncode == 0: -      return GitVCS(options) -  except OSError, (errno, message): -    if errno != 2:  # ENOENT -- they don't have git installed. -      raise - -  ErrorExit(("Could not guess version control system. " -             "Are you in a working copy directory?")) - - -def RealMain(argv, data=None): -  """The real main function. - -  Args: -    argv: Command line arguments. -    data: Diff contents. If None (default) the diff is generated by -      the VersionControlSystem implementation returned by GuessVCS(). - -  Returns: -    A 2-tuple (issue id, patchset id). -    The patchset id is None if the base files are not uploaded by this -    script (applies only to SVN checkouts). -  """ -  logging.basicConfig(format=("%(asctime).19s %(levelname)s %(filename)s:" -                              "%(lineno)s %(message)s ")) -  os.environ['LC_ALL'] = 'C' -  options, args = parser.parse_args(argv[1:]) -  global verbosity -  verbosity = options.verbose -  if verbosity >= 3: -    logging.getLogger().setLevel(logging.DEBUG) -  elif verbosity >= 2: -    logging.getLogger().setLevel(logging.INFO) -  vcs = GuessVCS(options) -  if isinstance(vcs, SubversionVCS): -    # base field is only allowed for Subversion. -    # Note: Fetching base files may become deprecated in future releases. -    base = vcs.GuessBase(options.download_base) -  else: -    base = None -  if not base and options.download_base: -    options.download_base = True -    logging.info("Enabled upload of base file") -  if not options.assume_yes: -    vcs.CheckForUnknownFiles() -  if data is None: -    data = vcs.GenerateDiff(args) -  files = vcs.GetBaseFiles(data) -  if verbosity >= 1: -    print "Upload server:", options.server, "(change with -s/--server)" -  if options.issue: -    prompt = "Message describing this patch set: " -  else: -    prompt = "New issue subject: " -  message = options.message or raw_input(prompt).strip() -  if not message: -    ErrorExit("A non-empty message is required") -  rpc_server = GetRpcServer(options) -  form_fields = [("subject", message)] -  if base: -    form_fields.append(("base", base)) -  if options.issue: -    form_fields.append(("issue", str(options.issue))) -  if options.email: -    form_fields.append(("user", options.email)) -  if options.reviewers: -    for reviewer in options.reviewers.split(','): -      if "@" in reviewer and not reviewer.split("@")[1].count(".") == 1: -        ErrorExit("Invalid email address: %s" % reviewer) -    form_fields.append(("reviewers", options.reviewers)) -  if options.cc: -    for cc in options.cc.split(','): -      if "@" in cc and not cc.split("@")[1].count(".") == 1: -        ErrorExit("Invalid email address: %s" % cc) -    form_fields.append(("cc", options.cc)) -  description = options.description -  if options.description_file: -    if options.description: -      ErrorExit("Can't specify description and description_file") -    file = open(options.description_file, 'r') -    description = file.read() -    file.close() -  if description: -    form_fields.append(("description", description)) -  # Send a hash of all the base file so the server can determine if a copy -  # already exists in an earlier patchset. -  base_hashes = "" -  for file, info in files.iteritems(): -    if not info[0] is None: -      checksum = md5.new(info[0]).hexdigest() -      if base_hashes: -        base_hashes += "|" -      base_hashes += checksum + ":" + file -  form_fields.append(("base_hashes", base_hashes)) -  # If we're uploading base files, don't send the email before the uploads, so -  # that it contains the file status. -  if options.send_mail and options.download_base: -    form_fields.append(("send_mail", "1")) -  if not options.download_base: -    form_fields.append(("content_upload", "1")) -  if len(data) > MAX_UPLOAD_SIZE: -    print "Patch is large, so uploading file patches separately." -    uploaded_diff_file = [] -    form_fields.append(("separate_patches", "1")) -  else: -    uploaded_diff_file = [("data", "data.diff", data)] -  ctype, body = EncodeMultipartFormData(form_fields, uploaded_diff_file) -  response_body = rpc_server.Send("/upload", body, content_type=ctype) -  patchset = None -  if not options.download_base or not uploaded_diff_file: -    lines = response_body.splitlines() -    if len(lines) >= 2: -      msg = lines[0] -      patchset = lines[1].strip() -      patches = [x.split(" ", 1) for x in lines[2:]] -    else: -      msg = response_body -  else: -    msg = response_body -  StatusUpdate(msg) -  if not response_body.startswith("Issue created.") and \ -  not response_body.startswith("Issue updated."): -    sys.exit(0) -  issue = msg[msg.rfind("/")+1:] - -  if not uploaded_diff_file: -    result = UploadSeparatePatches(issue, rpc_server, patchset, data, options) -    if not options.download_base: -      patches = result - -  if not options.download_base: -    vcs.UploadBaseFiles(issue, rpc_server, patches, patchset, options, files) -    if options.send_mail: -      rpc_server.Send("/" + issue + "/mail", payload="") -  return issue, patchset - - -def main(): -  try: -    RealMain(sys.argv) -  except KeyboardInterrupt: -    print -    StatusUpdate("Interrupted.") -    sys.exit(1) - - -if __name__ == "__main__": -  main() diff --git a/googlemock/scripts/upload_gmock.py b/googlemock/scripts/upload_gmock.py deleted file mode 100755 index 5dc484b3..00000000 --- a/googlemock/scripts/upload_gmock.py +++ /dev/null @@ -1,78 +0,0 @@ -#!/usr/bin/env python -# -# Copyright 2009, Google Inc. -# All rights reserved. -# -# Redistribution and use in source and binary forms, with or without -# modification, are permitted provided that the following conditions are -# met: -# -#     * Redistributions of source code must retain the above copyright -# notice, this list of conditions and the following disclaimer. -#     * Redistributions in binary form must reproduce the above -# copyright notice, this list of conditions and the following disclaimer -# in the documentation and/or other materials provided with the -# distribution. -#     * Neither the name of Google Inc. nor the names of its -# contributors may be used to endorse or promote products derived from -# this software without specific prior written permission. -# -# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS -# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT -# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR -# A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT -# OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, -# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT -# LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, -# DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY -# THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT -# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE -# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. - -"""upload_gmock.py v0.1.0 -- uploads a Google Mock patch for review. - -This simple wrapper passes all command line flags and ---cc=googlemock@googlegroups.com to upload.py. - -USAGE: upload_gmock.py [options for upload.py] -""" - -__author__ = 'wan@google.com (Zhanyong Wan)' - -import os -import sys - -CC_FLAG = '--cc=' -GMOCK_GROUP = 'googlemock@googlegroups.com' - - -def main(): -  # Finds the path to upload.py, assuming it is in the same directory -  # as this file. -  my_dir = os.path.dirname(os.path.abspath(__file__)) -  upload_py_path = os.path.join(my_dir, 'upload.py') - -  # Adds Google Mock discussion group to the cc line if it's not there -  # already. -  upload_py_argv = [upload_py_path] -  found_cc_flag = False -  for arg in sys.argv[1:]: -    if arg.startswith(CC_FLAG): -      found_cc_flag = True -      cc_line = arg[len(CC_FLAG):] -      cc_list = [addr for addr in cc_line.split(',') if addr] -      if GMOCK_GROUP not in cc_list: -        cc_list.append(GMOCK_GROUP) -      upload_py_argv.append(CC_FLAG + ','.join(cc_list)) -    else: -      upload_py_argv.append(arg) - -  if not found_cc_flag: -    upload_py_argv.append(CC_FLAG + GMOCK_GROUP) - -  # Invokes upload.py with the modified command line flags. -  os.execv(upload_py_path, upload_py_argv) - - -if __name__ == '__main__': -  main() diff --git a/googlemock/src/gmock_main.cc b/googlemock/src/gmock_main.cc index 98611b93..18c500f6 100644 --- a/googlemock/src/gmock_main.cc +++ b/googlemock/src/gmock_main.cc @@ -32,7 +32,10 @@  #include "gmock/gmock.h"  #include "gtest/gtest.h" -#ifdef ARDUINO +#if GTEST_OS_ESP8266 || GTEST_OS_ESP32 +#if GTEST_OS_ESP8266 +extern "C" { +#endif  void setup() {    // Since Google Mock depends on Google Test, InitGoogleMock() is    // also responsible for initializing Google Test.  Therefore there's @@ -40,6 +43,10 @@ void setup() {    testing::InitGoogleMock();  }  void loop() { RUN_ALL_TESTS(); } +#if GTEST_OS_ESP8266 +} +#endif +  #else  // MS C++ compiler/linker has a bug on Windows (not on Windows CE), which diff --git a/googlemock/test/gmock-actions_test.cc b/googlemock/test/gmock-actions_test.cc index f63c8c5a..ae4fa20e 100644 --- a/googlemock/test/gmock-actions_test.cc +++ b/googlemock/test/gmock-actions_test.cc @@ -46,6 +46,7 @@  #include <iterator>  #include <memory>  #include <string> +#include <type_traits>  #include "gmock/gmock.h"  #include "gmock/internal/gmock-port.h"  #include "gtest/gtest.h" @@ -73,6 +74,7 @@ using testing::Return;  using testing::ReturnNull;  using testing::ReturnRef;  using testing::ReturnRefOfCopy; +using testing::ReturnRoundRobin;  using testing::SetArgPointee;  using testing::SetArgumentPointee;  using testing::Unused; @@ -646,6 +648,41 @@ TEST(ReturnRefTest, IsCovariant) {    EXPECT_EQ(&derived, &a.Perform(std::make_tuple()));  } +template <typename T, typename = decltype(ReturnRef(std::declval<T&&>()))> +bool CanCallReturnRef(T&&) { return true; } +bool CanCallReturnRef(Unused) { return false; } + +// Tests that ReturnRef(v) is working with non-temporaries (T&) +TEST(ReturnRefTest, WorksForNonTemporary) { +  int scalar_value = 123; +  EXPECT_TRUE(CanCallReturnRef(scalar_value)); + +  std::string non_scalar_value("ABC"); +  EXPECT_TRUE(CanCallReturnRef(non_scalar_value)); + +  const int const_scalar_value{321}; +  EXPECT_TRUE(CanCallReturnRef(const_scalar_value)); + +  const std::string const_non_scalar_value("CBA"); +  EXPECT_TRUE(CanCallReturnRef(const_non_scalar_value)); +} + +// Tests that ReturnRef(v) is not working with temporaries (T&&) +TEST(ReturnRefTest, DoesNotWorkForTemporary) { +  auto scalar_value = []()  -> int { return 123; }; +  EXPECT_FALSE(CanCallReturnRef(scalar_value())); + +  auto non_scalar_value = []() -> std::string { return "ABC"; }; +  EXPECT_FALSE(CanCallReturnRef(non_scalar_value())); + +  // cannot use here callable returning "const scalar type", +  // because such const for scalar return type is ignored +  EXPECT_FALSE(CanCallReturnRef(static_cast<const int>(321))); + +  auto const_non_scalar_value = []() -> const std::string { return "CBA"; }; +  EXPECT_FALSE(CanCallReturnRef(const_non_scalar_value())); +} +  // Tests that ReturnRefOfCopy(v) works for reference types.  TEST(ReturnRefOfCopyTest, WorksForReference) {    int n = 42; @@ -670,6 +707,31 @@ TEST(ReturnRefOfCopyTest, IsCovariant) {    EXPECT_NE(&derived, &a.Perform(std::make_tuple()));  } +// Tests that ReturnRoundRobin(v) works with initializer lists +TEST(ReturnRoundRobinTest, WorksForInitList) { +  Action<int()> ret = ReturnRoundRobin({1, 2, 3}); + +  EXPECT_EQ(1, ret.Perform(std::make_tuple())); +  EXPECT_EQ(2, ret.Perform(std::make_tuple())); +  EXPECT_EQ(3, ret.Perform(std::make_tuple())); +  EXPECT_EQ(1, ret.Perform(std::make_tuple())); +  EXPECT_EQ(2, ret.Perform(std::make_tuple())); +  EXPECT_EQ(3, ret.Perform(std::make_tuple())); +} + +// Tests that ReturnRoundRobin(v) works with vectors +TEST(ReturnRoundRobinTest, WorksForVector) { +  std::vector<double> v = {4.4, 5.5, 6.6}; +  Action<double()> ret = ReturnRoundRobin(v); + +  EXPECT_EQ(4.4, ret.Perform(std::make_tuple())); +  EXPECT_EQ(5.5, ret.Perform(std::make_tuple())); +  EXPECT_EQ(6.6, ret.Perform(std::make_tuple())); +  EXPECT_EQ(4.4, ret.Perform(std::make_tuple())); +  EXPECT_EQ(5.5, ret.Perform(std::make_tuple())); +  EXPECT_EQ(6.6, ret.Perform(std::make_tuple())); +} +  // Tests that DoDefault() does the default action for the mock method.  class MockClass { diff --git a/googlemock/test/pump_test.py b/googlemock/test/pump_test.py new file mode 100755 index 00000000..eb5a1313 --- /dev/null +++ b/googlemock/test/pump_test.py @@ -0,0 +1,182 @@ +#!/usr/bin/env python +# +# Copyright 2010, Google Inc. +# All rights reserved. +# +# Redistribution and use in source and binary forms, with or without +# modification, are permitted provided that the following conditions are +# met: +# +#     * Redistributions of source code must retain the above copyright +# notice, this list of conditions and the following disclaimer. +#     * Redistributions in binary form must reproduce the above +# copyright notice, this list of conditions and the following disclaimer +# in the documentation and/or other materials provided with the +# distribution. +#     * Neither the name of Google Inc. nor the names of its +# contributors may be used to endorse or promote products derived from +# this software without specific prior written permission. +# +# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +# "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +# LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +# A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +# OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +# LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +# DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +# THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +"""Tests for the Pump meta-programming tool.""" + +from google3.testing.pybase import googletest +import google3.third_party.googletest.googlemock.scripts.pump + +pump = google3.third_party.googletest.googlemock.scripts.pump +Convert = pump.ConvertFromPumpSource +StripMetaComments = pump.StripMetaComments + + +class PumpTest(googletest.TestCase): + +  def testConvertsEmptyToEmpty(self): +    self.assertEquals('', Convert('').strip()) + +  def testConvertsPlainCodeToSame(self): +    self.assertEquals('#include <stdio.h>\n', +                      Convert('#include <stdio.h>\n')) + +  def testConvertsLongIWYUPragmaToSame(self): +    long_line = '// IWYU pragma: private, include "' + (80*'a') + '.h"\n' +    self.assertEquals(long_line, Convert(long_line)) + +  def testConvertsIWYUPragmaWithLeadingSpaceToSame(self): +    long_line = ' // IWYU pragma: private, include "' + (80*'a') + '.h"\n' +    self.assertEquals(long_line, Convert(long_line)) + +  def testConvertsIWYUPragmaWithSlashStarLeaderToSame(self): +    long_line = '/* IWYU pragma: private, include "' + (80*'a') + '.h"\n' +    self.assertEquals(long_line, Convert(long_line)) + +  def testConvertsIWYUPragmaWithSlashStarAndSpacesToSame(self): +    long_line = ' /* IWYU pragma: private, include "' + (80*'a') + '.h"\n' +    self.assertEquals(long_line, Convert(long_line)) + +  def testIgnoresMetaComment(self): +    self.assertEquals('', +                      Convert('$$ This is a Pump meta comment.\n').strip()) + +  def testSimpleVarDeclarationWorks(self): +    self.assertEquals('3\n', +                      Convert('$var m = 3\n' +                              '$m\n')) + +  def testVarDeclarationCanReferenceEarlierVar(self): +    self.assertEquals('43 != 3;\n', +                      Convert('$var a = 42\n' +                              '$var b = a + 1\n' +                              '$var c = (b - a)*3\n' +                              '$b != $c;\n')) + +  def testSimpleLoopWorks(self): +    self.assertEquals('1, 2, 3, 4, 5\n', +                      Convert('$var n = 5\n' +                              '$range i 1..n\n' +                              '$for i, [[$i]]\n')) + +  def testSimpleLoopWithCommentWorks(self): +    self.assertEquals('1, 2, 3, 4, 5\n', +                      Convert('$var n = 5    $$ This is comment 1.\n' +                              '$range i 1..n $$ This is comment 2.\n' +                              '$for i, [[$i]]\n')) + +  def testNonTrivialRangeExpressionsWork(self): +    self.assertEquals('1, 2, 3, 4\n', +                      Convert('$var n = 5\n' +                              '$range i (n/n)..(n - 1)\n' +                              '$for i, [[$i]]\n')) + +  def testLoopWithoutSeparatorWorks(self): +    self.assertEquals('a + 1 + 2 + 3;\n', +                      Convert('$range i 1..3\n' +                              'a$for i [[ + $i]];\n')) + +  def testCanGenerateDollarSign(self): +    self.assertEquals('$\n', Convert('$($)\n')) + +  def testCanIterpolateExpressions(self): +    self.assertEquals('a[2] = 3;\n', +                      Convert('$var i = 1\n' +                              'a[$(i + 1)] = $(i*4 - 1);\n')) + +  def testConditionalWithoutElseBranchWorks(self): +    self.assertEquals('true\n', +                      Convert('$var n = 5\n' +                              '$if n > 0 [[true]]\n')) + +  def testConditionalWithElseBranchWorks(self): +    self.assertEquals('true -- really false\n', +                      Convert('$var n = 5\n' +                              '$if n > 0 [[true]]\n' +                              '$else [[false]] -- \n' +                              '$if n > 10 [[really true]]\n' +                              '$else [[really false]]\n')) + +  def testConditionalWithCascadingElseBranchWorks(self): +    self.assertEquals('a\n', +                      Convert('$var n = 5\n' +                              '$if n > 0 [[a]]\n' +                              '$elif n > 10 [[b]]\n' +                              '$else [[c]]\n')) +    self.assertEquals('b\n', +                      Convert('$var n = 5\n' +                              '$if n > 10 [[a]]\n' +                              '$elif n > 0 [[b]]\n' +                              '$else [[c]]\n')) +    self.assertEquals('c\n', +                      Convert('$var n = 5\n' +                              '$if n > 10 [[a]]\n' +                              '$elif n > 8 [[b]]\n' +                              '$else [[c]]\n')) + +  def testNestedLexicalBlocksWork(self): +    self.assertEquals('a = 5;\n', +                      Convert('$var n = 5\n' +                              'a = [[$if n > 0 [[$n]]]];\n')) + + +class StripMetaCommentsTest(googletest.TestCase): + +  def testReturnsSameStringIfItContainsNoComment(self): +    self.assertEquals('', StripMetaComments('')) +    self.assertEquals(' blah ', StripMetaComments(' blah ')) +    self.assertEquals('A single $ is fine.', +                      StripMetaComments('A single $ is fine.')) +    self.assertEquals('multiple\nlines', +                      StripMetaComments('multiple\nlines')) + +  def testStripsSimpleComment(self): +    self.assertEquals('yes\n', StripMetaComments('yes $$ or no?\n')) + +  def testStripsSimpleCommentWithMissingNewline(self): +    self.assertEquals('yes', StripMetaComments('yes $$ or no?')) + +  def testStripsPureCommentLinesEntirely(self): +    self.assertEquals('yes\n', +                      StripMetaComments('$$ a pure comment line.\n' +                                        'yes $$ or no?\n' +                                        '    $$ another comment line.\n')) + +  def testStripsCommentsFromMultiLineText(self): +    self.assertEquals('multi-\n' +                      'line\n' +                      'text is fine.', +                      StripMetaComments('multi- $$ comment 1\n' +                                        'line\n' +                                        'text is fine. $$ comment 2')) + + +if __name__ == '__main__': +  googletest.main() | 
