Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(217)

Unified Diff: testing_support/super_mox.py

Issue 939523004: Copied files from depot_tools. (Closed) Base URL: https://chromium.googlesource.com/infra/testing/testing_support.git@master
Patch Set: Created 5 years, 10 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
Index: testing_support/super_mox.py
diff --git a/testing_support/super_mox.py b/testing_support/super_mox.py
new file mode 100644
index 0000000000000000000000000000000000000000..3322719991abc2c777466f08f6fe9c3571b1f27f
--- /dev/null
+++ b/testing_support/super_mox.py
@@ -0,0 +1,155 @@
+# Copyright (c) 2011 The Chromium Authors. All rights reserved.
+# Use of this source code is governed by a BSD-style license that can be
+# found in the LICENSE file.
+
+"""Simplify unit tests based on pymox."""
+
+import os
+import random
+import shutil
+import string
+import StringIO
+import subprocess
+import sys
+
+sys.path.append(os.path.dirname(os.path.dirname(__file__)))
iannucci 2015/02/19 00:22:39 do we still need this? Can we just guarantee that
pgervais 2015/02/19 00:45:57 Turns out we don't need the file at all...
+from third_party.pymox import mox
+
+
+class IsOneOf(mox.Comparator):
+ def __init__(self, keys):
+ self._keys = keys
+
+ def equals(self, rhs):
+ return rhs in self._keys
+
+ def __repr__(self):
+ return '<sequence or map containing \'%s\'>' % str(self._keys)
+
+
+class TestCaseUtils(object):
+ """Base class with some additional functionalities. People will usually want
+ to use SuperMoxTestBase instead."""
+ # Backup the separator in case it gets mocked
+ _OS_SEP = os.sep
+ _RANDOM_CHOICE = random.choice
+ _RANDOM_RANDINT = random.randint
+ _STRING_LETTERS = string.letters
+
+ ## Some utilities for generating arbitrary arguments.
+ def String(self, max_length):
+ return ''.join([self._RANDOM_CHOICE(self._STRING_LETTERS)
+ for _ in xrange(self._RANDOM_RANDINT(1, max_length))])
+
+ def Strings(self, max_arg_count, max_arg_length):
+ return [self.String(max_arg_length) for _ in xrange(max_arg_count)]
+
+ def Args(self, max_arg_count=8, max_arg_length=16):
+ return self.Strings(max_arg_count,
+ self._RANDOM_RANDINT(1, max_arg_length))
+
+ def _DirElts(self, max_elt_count=4, max_elt_length=8):
+ return self._OS_SEP.join(self.Strings(max_elt_count, max_elt_length))
+
+ def Dir(self, max_elt_count=4, max_elt_length=8):
+ return (self._RANDOM_CHOICE((self._OS_SEP, '')) +
+ self._DirElts(max_elt_count, max_elt_length))
+
+ def SvnUrl(self, max_elt_count=4, max_elt_length=8):
+ return ('svn://random_host:port/a' +
+ self._DirElts(max_elt_count, max_elt_length
+ ).replace(self._OS_SEP, '/'))
+
+ def RootDir(self, max_elt_count=4, max_elt_length=8):
+ return self._OS_SEP + self._DirElts(max_elt_count, max_elt_length)
+
+ def compareMembers(self, obj, members):
+ """If you add a member, be sure to add the relevant test!"""
+ # Skip over members starting with '_' since they are usually not meant to
+ # be for public use.
+ actual_members = [x for x in sorted(dir(obj))
+ if not x.startswith('_')]
+ expected_members = sorted(members)
+ if actual_members != expected_members:
+ diff = ([i for i in actual_members if i not in expected_members] +
+ [i for i in expected_members if i not in actual_members])
+ print >> sys.stderr, diff
+ # pylint: disable=E1101
+ self.assertEqual(actual_members, expected_members)
+
+ def setUp(self):
+ self.root_dir = self.Dir()
+ self.args = self.Args()
+ self.relpath = self.String(200)
+
+ def tearDown(self):
+ pass
+
+
+class StdoutCheck(object):
+ def setUp(self):
+ # Override the mock with a StringIO, it's much less painful to test.
+ self._old_stdout = sys.stdout
+ stdout = StringIO.StringIO()
+ stdout.flush = lambda: None
+ sys.stdout = stdout
+
+ def tearDown(self):
+ try:
+ # If sys.stdout was used, self.checkstdout() must be called.
+ # pylint: disable=E1101
+ if not sys.stdout.closed:
+ self.assertEquals('', sys.stdout.getvalue())
+ except AttributeError:
+ pass
+ sys.stdout = self._old_stdout
+
+ def checkstdout(self, expected):
+ value = sys.stdout.getvalue()
+ sys.stdout.close()
+ # pylint: disable=E1101
+ self.assertEquals(expected, value)
+
+
+class SuperMoxTestBase(TestCaseUtils, StdoutCheck, mox.MoxTestBase):
+ def setUp(self):
+ """Patch a few functions with know side-effects."""
+ TestCaseUtils.setUp(self)
+ mox.MoxTestBase.setUp(self)
+ os_to_mock = ('chdir', 'chown', 'close', 'closerange', 'dup', 'dup2',
+ 'fchdir', 'fchmod', 'fchown', 'fdopen', 'getcwd', 'lseek',
+ 'makedirs', 'mkdir', 'open', 'popen', 'popen2', 'popen3', 'popen4',
+ 'read', 'remove', 'removedirs', 'rename', 'renames', 'rmdir', 'symlink',
+ 'system', 'tmpfile', 'walk', 'write')
+ self.MockList(os, os_to_mock)
+ os_path_to_mock = ('abspath', 'exists', 'getsize', 'isdir', 'isfile',
+ 'islink', 'ismount', 'lexists', 'realpath', 'samefile', 'walk')
+ self.MockList(os.path, os_path_to_mock)
+ self.MockList(shutil, ('rmtree'))
+ self.MockList(subprocess, ('call', 'Popen'))
+ # Don't mock stderr since it confuses unittests.
+ self.MockList(sys, ('stdin'))
+ StdoutCheck.setUp(self)
+
+ def tearDown(self):
+ StdoutCheck.tearDown(self)
+ TestCaseUtils.tearDown(self)
+ mox.MoxTestBase.tearDown(self)
+
+ def MockList(self, parent, items_to_mock):
+ for item in items_to_mock:
+ # Skip over items not present because of OS-specific implementation,
+ # implemented only in later python version, etc.
+ if hasattr(parent, item):
+ try:
+ self.mox.StubOutWithMock(parent, item)
+ except TypeError, e:
+ raise TypeError(
+ 'Couldn\'t mock %s in %s: %s' % (item, parent.__name__, e))
+
+ def UnMock(self, obj, name):
+ """Restore an object inside a test."""
+ for (parent, old_child, child_name) in self.mox.stubs.cache:
+ if parent == obj and child_name == name:
+ setattr(parent, child_name, old_child)
+ break

Powered by Google App Engine
This is Rietveld 408576698