diff options
| -rw-r--r-- | src/leap/base/network.py | 2 | ||||
| -rw-r--r-- | src/leap/base/tests/test_checks.py | 7 | ||||
| -rw-r--r-- | src/leap/base/tests/test_providers.py | 6 | ||||
| -rw-r--r-- | src/leap/eip/openvpnconnection.py | 17 | ||||
| -rw-r--r-- | src/leap/eip/specs.py | 2 | ||||
| -rw-r--r-- | src/leap/eip/tests/data.py | 7 | ||||
| -rw-r--r-- | src/leap/eip/tests/test_checks.py | 37 | ||||
| -rw-r--r-- | src/leap/eip/tests/test_config.py | 19 | ||||
| -rw-r--r-- | src/leap/eip/tests/test_eipconnection.py | 12 | ||||
| -rw-r--r-- | src/leap/eip/tests/test_openvpnconnection.py | 10 | ||||
| -rw-r--r-- | src/leap/gui/__init__.py | 7 | ||||
| -rw-r--r-- | src/leap/gui/firstrun/__init__.py | 9 | ||||
| -rw-r--r-- | src/leap/gui/firstrun/regvalidation.py | 2 | ||||
| -rwxr-xr-x | src/leap/gui/firstrun/tests/integration/fake_provider.py | 10 | ||||
| -rw-r--r-- | src/leap/gui/test_mainwindow_rc.py | 9 | ||||
| -rw-r--r-- | src/leap/util/dicts.py | 44 | 
16 files changed, 135 insertions, 65 deletions
| diff --git a/src/leap/base/network.py b/src/leap/base/network.py index 3891b00a..3aba3f61 100644 --- a/src/leap/base/network.py +++ b/src/leap/base/network.py @@ -31,7 +31,7 @@ class NetworkCheckerThread(object):          # see in eip.config for function          # #718          self.checker = LeapNetworkChecker( -                        provider_gw = get_eip_gateway()) +            provider_gw=get_eip_gateway())      def start(self):          self.process_handle = self._launch_recurrent_network_checks( diff --git a/src/leap/base/tests/test_checks.py b/src/leap/base/tests/test_checks.py index bec09ce6..8d573b1e 100644 --- a/src/leap/base/tests/test_checks.py +++ b/src/leap/base/tests/test_checks.py @@ -40,7 +40,14 @@ class LeapNetworkCheckTest(BaseLeapTest):      def test_checker_should_actually_call_all_tests(self):          checker = checks.LeapNetworkChecker() +        mc = Mock() +        checker.run_all(checker=mc) +        self.assertTrue(mc.check_internet_connection.called, "not called") +        self.assertTrue(mc.check_tunnel_default_interface.called, "not called") +        self.assertTrue(mc.is_internet_up.called, "not called") +        # ping gateway only called if we pass provider_gw +        checker = checks.LeapNetworkChecker(provider_gw="0.0.0.0")          mc = Mock()          checker.run_all(checker=mc)          self.assertTrue(mc.check_internet_connection.called, "not called") diff --git a/src/leap/base/tests/test_providers.py b/src/leap/base/tests/test_providers.py index 8d3b8847..15c4ed58 100644 --- a/src/leap/base/tests/test_providers.py +++ b/src/leap/base/tests/test_providers.py @@ -30,7 +30,9 @@ EXPECTED_DEFAULT_CONFIG = {  class TestLeapProviderDefinition(BaseLeapTest):      def setUp(self): -        self.definition = providers.LeapProviderDefinition() +        self.domain = "testprovider.example.org" +        self.definition = providers.LeapProviderDefinition( +            domain=self.domain)          self.definition.save()          self.definition.load()          self.config = self.definition.config @@ -51,7 +53,7 @@ class TestLeapProviderDefinition(BaseLeapTest):              os.path.join(                  self.home,                  '.config', 'leap', 'providers', -                '%s' % BRANDING.get('provider_domain'), +                '%s' % self.domain,                  'provider.json'))          with self.assertRaises(AttributeError):              self.definition.slug = 23 diff --git a/src/leap/eip/openvpnconnection.py b/src/leap/eip/openvpnconnection.py index 34f1e18b..4104bd0e 100644 --- a/src/leap/eip/openvpnconnection.py +++ b/src/leap/eip/openvpnconnection.py @@ -233,8 +233,8 @@ to be triggered for each one of them.          #self.tn.read_until('ENTER PASSWORD:', 2)          #self.tn.write(self.password + '\n')          #self.tn.read_until('SUCCESS:', 2) - -        self._seek_to_eof() +        if self.tn: +            self._seek_to_eof()          return True      def _seek_to_eof(self): @@ -364,7 +364,8 @@ to be triggered for each one of them.          interface          """          logger.debug("disconnecting...") -        self._send_command("signal SIGTERM\n") +        if self.connected(): +            self._send_command("signal SIGTERM\n")          if self.subp:              return True @@ -373,9 +374,13 @@ to be triggered for each one of them.          #try patching in old openvpn host and trying again          process = self._get_openvpn_process()          if process: -            self.host = \ -                process.cmdline[process.cmdline.index("--management") + 1] -            self._send_command("signal SIGTERM\n") +            logger.debug('process :%s' % process) +            cmdline = process.cmdline + +            if isinstance(cmdline, list): +                _index = cmdline.index("--management") +                self.host = cmdline[_index + 1] +                self._send_command("signal SIGTERM\n")              #make sure the process was terminated              process = self._get_openvpn_process() diff --git a/src/leap/eip/specs.py b/src/leap/eip/specs.py index 84b2597d..57e7537b 100644 --- a/src/leap/eip/specs.py +++ b/src/leap/eip/specs.py @@ -8,7 +8,7 @@ from leap.base import config as baseconfig  PROVIDER_CA_CERT = __branding.get(      'provider_ca_file', -    'testprovider-ca-cert.pem') +    'cacert.pem')  provider_ca_path = lambda domain: str(os.path.join(      #baseconfig.get_default_provider_path(), diff --git a/src/leap/eip/tests/data.py b/src/leap/eip/tests/data.py index f1d3b0bc..cadf720e 100644 --- a/src/leap/eip/tests/data.py +++ b/src/leap/eip/tests/data.py @@ -1,11 +1,12 @@  from __future__ import unicode_literals  import os -from leap import __branding +#from leap import __branding  # sample data used in tests -PROVIDER = __branding.get('provider_domain') +#PROVIDER = __branding.get('provider_domain') +PROVIDER = "testprovider.example.org"  EIP_SAMPLE_CONFIG = {      "provider": "%s" % PROVIDER, @@ -15,7 +16,7 @@ EIP_SAMPLE_CONFIG = {      "openvpn_ca_certificate": os.path.expanduser(          "~/.config/leap/providers/"          "%s/" -        "keys/ca/testprovider-ca-cert.pem" % PROVIDER), +        "keys/ca/cacert.pem" % PROVIDER),      "openvpn_client_certificate": os.path.expanduser(          "~/.config/leap/providers/"          "%s/" diff --git a/src/leap/eip/tests/test_checks.py b/src/leap/eip/tests/test_checks.py index 58ce473f..1d7bfc17 100644 --- a/src/leap/eip/tests/test_checks.py +++ b/src/leap/eip/tests/test_checks.py @@ -39,6 +39,8 @@ class NoLogRequestHandler:  class EIPCheckTest(BaseLeapTest):      __name__ = "eip_check_tests" +    provider = "testprovider.example.org" +    maxDiff = None      def setUp(self):          pass @@ -49,7 +51,7 @@ class EIPCheckTest(BaseLeapTest):      # test methods are there, and can be called from run_all      def test_checker_should_implement_check_methods(self): -        checker = eipchecks.EIPConfigChecker() +        checker = eipchecks.EIPConfigChecker(domain=self.provider)          self.assertTrue(hasattr(checker, "check_default_eipconfig"),                          "missing meth") @@ -62,7 +64,7 @@ class EIPCheckTest(BaseLeapTest):                          "missing meth")      def test_checker_should_actually_call_all_tests(self): -        checker = eipchecks.EIPConfigChecker() +        checker = eipchecks.EIPConfigChecker(domain=self.provider)          mc = Mock()          checker.run_all(checker=mc) @@ -79,7 +81,7 @@ class EIPCheckTest(BaseLeapTest):      # test individual check methods      def test_check_default_eipconfig(self): -        checker = eipchecks.EIPConfigChecker() +        checker = eipchecks.EIPConfigChecker(domain=self.provider)          # no eip config (empty home)          eipconfig_path = checker.eipconfig.filename          self.assertFalse(os.path.isfile(eipconfig_path)) @@ -93,15 +95,15 @@ class EIPCheckTest(BaseLeapTest):          # small workaround for evaluating home dirs correctly          EIP_SAMPLE_CONFIG = copy.copy(testdata.EIP_SAMPLE_CONFIG)          EIP_SAMPLE_CONFIG['openvpn_client_certificate'] = \ -            eipspecs.client_cert_path() +            eipspecs.client_cert_path(self.provider)          EIP_SAMPLE_CONFIG['openvpn_ca_certificate'] = \ -            eipspecs.provider_ca_path() +            eipspecs.provider_ca_path(self.provider)          self.assertEqual(deserialized, EIP_SAMPLE_CONFIG)          # TODO: shold ALSO run validation methods.      def test_check_is_there_default_provider(self): -        checker = eipchecks.EIPConfigChecker() +        checker = eipchecks.EIPConfigChecker(domain=self.provider)          # we do dump a sample eip config, but lacking a          # default provider entry.          # This error will be possible catched in a different @@ -178,6 +180,7 @@ class EIPCheckTest(BaseLeapTest):  class ProviderCertCheckerTest(BaseLeapTest):      __name__ = "provider_cert_checker_tests" +    provider = "testprovider.example.org"      def setUp(self):          pass @@ -226,13 +229,20 @@ class ProviderCertCheckerTest(BaseLeapTest):      # test individual check methods +    @unittest.skip      def test_is_there_provider_ca(self): +        # XXX commenting out this test. +        # With the generic client this does not make sense, +        # we should dump one there. +        # or test conductor logic.          checker = eipchecks.ProviderCertChecker()          self.assertTrue(              checker.is_there_provider_ca())  class ProviderCertCheckerHTTPSTests(BaseHTTPSServerTestCase, BaseLeapTest): +    provider = "testprovider.example.org" +      class request_handler(NoLogRequestHandler, BaseHTTPRequestHandler):          responses = {              '/': ['OK', ''], @@ -292,12 +302,19 @@ class ProviderCertCheckerHTTPSTests(BaseHTTPSServerTestCase, BaseLeapTest):          # same, but get cacert from leap.custom          # XXX TODO! +    @unittest.skip      def test_download_new_client_cert(self): +        # FIXME +        # Magick srp decorator broken right now... +        # Have to mock the decorator and inject something that +        # can bypass the authentication +          uri = "https://%s/client.cert" % (self.get_server())          cacert = where_cert('cacert.pem') -        checker = eipchecks.ProviderCertChecker() +        checker = eipchecks.ProviderCertChecker(domain=self.provider) +        credentials = "testuser", "testpassword"          self.assertTrue(checker.download_new_client_cert( -                        uri=uri, verify=cacert)) +                        credentials=credentials, uri=uri, verify=cacert))          # now download a malformed cert          uri = "https://%s/badclient.cert" % (self.get_server()) @@ -305,7 +322,7 @@ class ProviderCertCheckerHTTPSTests(BaseHTTPSServerTestCase, BaseLeapTest):          checker = eipchecks.ProviderCertChecker()          with self.assertRaises(ValueError):              self.assertTrue(checker.download_new_client_cert( -                            uri=uri, verify=cacert)) +                            credentials=credentials, uri=uri, verify=cacert))          # did we write cert to its path?          clientcertfile = eipspecs.client_cert_path() @@ -339,7 +356,7 @@ class ProviderCertCheckerHTTPSTests(BaseHTTPSServerTestCase, BaseLeapTest):      def test_check_new_cert_needed(self):          # check: missing cert -        checker = eipchecks.ProviderCertChecker() +        checker = eipchecks.ProviderCertChecker(domain=self.provider)          self.assertTrue(checker.check_new_cert_needed(skip_download=True))          # TODO check: malformed cert          # TODO check: expired cert diff --git a/src/leap/eip/tests/test_config.py b/src/leap/eip/tests/test_config.py index 6759b522..50538240 100644 --- a/src/leap/eip/tests/test_config.py +++ b/src/leap/eip/tests/test_config.py @@ -18,13 +18,14 @@ from leap.util.fileutil import mkdir_p  _system = platform.system() -PROVIDER = BRANDING.get('provider_domain') -PROVIDER_SHORTNAME = BRANDING.get('short_name') +#PROVIDER = BRANDING.get('provider_domain') +#PROVIDER_SHORTNAME = BRANDING.get('short_name')  class EIPConfigTest(BaseLeapTest):      __name__ = "eip_config_tests" +    provider = "testprovider.example.org"      def setUp(self):          pass @@ -74,7 +75,8 @@ class EIPConfigTest(BaseLeapTest):          args.append('--persist-tun')          args.append('--persist-key')          args.append('--remote') -        args.append('%s' % eipconfig.get_eip_gateway()) +        args.append('%s' % eipconfig.get_eip_gateway( +            provider=self.provider))          # XXX get port!?          args.append('1194')          # XXX get proto @@ -103,23 +105,23 @@ class EIPConfigTest(BaseLeapTest):          args.append(os.path.join(              self.home,              '.config', 'leap', 'providers', -            '%s' % PROVIDER, +            '%s' % self.provider,              'keys', 'client',              'openvpn.pem'))          args.append('--key')          args.append(os.path.join(              self.home,              '.config', 'leap', 'providers', -            '%s' % PROVIDER, +            '%s' % self.provider,              'keys', 'client',              'openvpn.pem'))          args.append('--ca')          args.append(os.path.join(              self.home,              '.config', 'leap', 'providers', -            '%s' % PROVIDER, +            '%s' % self.provider,              'keys', 'ca', -            '%s-cacert.pem' % PROVIDER_SHORTNAME)) +            'cacert.pem'))          return args      # build command string @@ -141,7 +143,8 @@ class EIPConfigTest(BaseLeapTest):          print 'vpnbin = ', vpnbin          command, args = eipconfig.build_ovpn_command(              do_pkexec_check=False, vpnbin=vpnbin, -            socket_path="/tmp/test.socket") +            socket_path="/tmp/test.socket", +            provider=self.provider)          self.assertEqual(command, self.home + '/bin/openvpn')          self.assertEqual(args, self.get_expected_openvpn_args()) diff --git a/src/leap/eip/tests/test_eipconnection.py b/src/leap/eip/tests/test_eipconnection.py index bb643ae0..aefca36f 100644 --- a/src/leap/eip/tests/test_eipconnection.py +++ b/src/leap/eip/tests/test_eipconnection.py @@ -19,6 +19,8 @@ from leap.testing.basetest import BaseLeapTest  _system = platform.system() +PROVIDER = "testprovider.example.org" +  class NotImplementedError(Exception):      pass @@ -27,6 +29,7 @@ class NotImplementedError(Exception):  @patch('OpenVPNConnection._get_or_create_config')  @patch('OpenVPNConnection._set_ovpn_command')  class MockedEIPConnection(EIPConnection): +      def _set_ovpn_command(self):          self.command = "mock_command"          self.args = [1, 2, 3] @@ -35,6 +38,7 @@ class MockedEIPConnection(EIPConnection):  class EIPConductorTest(BaseLeapTest):      __name__ = "eip_conductor_tests" +    provider = PROVIDER      def setUp(self):          # XXX there's a conceptual/design @@ -51,8 +55,8 @@ class EIPConductorTest(BaseLeapTest):          # XXX change to keys_checker invocation          # (see config_checker) -        keyfiles = (eipspecs.provider_ca_path(), -                    eipspecs.client_cert_path()) +        keyfiles = (eipspecs.provider_ca_path(domain=self.provider), +                    eipspecs.client_cert_path(domain=self.provider))          for filepath in keyfiles:              self.touch(filepath)              self.chmod600(filepath) @@ -61,6 +65,7 @@ class EIPConductorTest(BaseLeapTest):          # some methods mocked          self.manager = Mock(name="openvpnmanager_mock")          self.con = MockedEIPConnection() +        self.con.provider = self.provider          self.con.run_openvpn_checks()      def tearDown(self): @@ -118,8 +123,9 @@ class EIPConductorTest(BaseLeapTest):                           self.con.status.CONNECTED)          # disconnect +        self.con.cleanup = Mock()          self.con.disconnect() -        self.con._disconnect.assert_called_once_with() +        self.con.cleanup.assert_called_once_with()          # new status should be disconnected          # XXX this should evolve and check no errors diff --git a/src/leap/eip/tests/test_openvpnconnection.py b/src/leap/eip/tests/test_openvpnconnection.py index 61769f04..0f27facf 100644 --- a/src/leap/eip/tests/test_openvpnconnection.py +++ b/src/leap/eip/tests/test_openvpnconnection.py @@ -76,13 +76,17 @@ class OpenVPNConnectionTest(BaseLeapTest):      #      def test_detect_vpn(self): +        # XXX review, not sure if captured all the logic +        # while fixing. kali.          openvpn_connection = openvpnconnection.OpenVPNConnection() +          with patch.object(psutil, "get_process_list") as mocked_psutil: +            mocked_process = Mock() +            mocked_process.name = "openvpn" +            mocked_psutil.return_value = [mocked_process]              with self.assertRaises(eipexceptions.OpenVPNAlreadyRunning): -                mocked_process = Mock() -                mocked_process.name = "openvpn" -                mocked_psutil.return_value = [mocked_process]                  openvpn_connection._check_if_running_instance() +          openvpn_connection._check_if_running_instance()      @unittest.skipIf(_system == "Windows", "lin/mac only") diff --git a/src/leap/gui/__init__.py b/src/leap/gui/__init__.py index 6ecd665f..9b8f8746 100644 --- a/src/leap/gui/__init__.py +++ b/src/leap/gui/__init__.py @@ -1,3 +1,10 @@ +try: +    import sip +    sip.setapi('QString', 2) +    sip.setapi('QVariant', 2) +except ValueError: +    pass +  import firstrun  __all__ = ['firstrun'] diff --git a/src/leap/gui/firstrun/__init__.py b/src/leap/gui/firstrun/__init__.py index 477e7269..8a70d90e 100644 --- a/src/leap/gui/firstrun/__init__.py +++ b/src/leap/gui/firstrun/__init__.py @@ -1,6 +1,9 @@ -import sip -sip.setapi('QString', 2) -sip.setapi('QVariant', 2) +try: +    import sip +    sip.setapi('QString', 2) +    sip.setapi('QVariant', 2) +except ValueError: +    pass  import connect  import intro diff --git a/src/leap/gui/firstrun/regvalidation.py b/src/leap/gui/firstrun/regvalidation.py index 6681b953..dbe30d3c 100644 --- a/src/leap/gui/firstrun/regvalidation.py +++ b/src/leap/gui/firstrun/regvalidation.py @@ -211,8 +211,6 @@ class RegisterUserValidationPage(ValidationPage):              wizard,              'start_eipconnection_signal', None) -        import pdb4qt; pdb4qt.set_trace() -          if conductor:              conductor.set_provider_domain(domain)              conductor.run_checks() diff --git a/src/leap/gui/firstrun/tests/integration/fake_provider.py b/src/leap/gui/firstrun/tests/integration/fake_provider.py index 09c6c468..33ee0ee6 100755 --- a/src/leap/gui/firstrun/tests/integration/fake_provider.py +++ b/src/leap/gui/firstrun/tests/integration/fake_provider.py @@ -12,8 +12,10 @@ and that you place the following files:  [ ] provider.json  [ ] eip-service.json -  """ +# XXX NOTE: intended for manual debug. +# I intend to include this as a regular test after 0.2.0 release +# (so we can add twisted as a dep there)  import binascii  import json  import os @@ -47,11 +49,13 @@ Testing the FAKE_API:  #####################   1) register an user - >> curl -d "user[login]=me" -d "user[password_salt]=foo" -d "user[password_verifier]=beef" http://localhost:8000/1/users.json + >> curl -d "user[login]=me" -d "user[password_salt]=foo" \ +         -d "user[password_verifier]=beef" http://localhost:8000/1/users.json   << {"errors": null}   2) check that if you try to register again, it will fail: - >> curl -d "user[login]=me" -d "user[password_salt]=foo" -d "user[password_verifier]=beef" http://localhost:8000/1/users.json + >> curl -d "user[login]=me" -d "user[password_salt]=foo" \ +         -d "user[password_verifier]=beef" http://localhost:8000/1/users.json   << {"errors": {"login": "already taken!"}}  """ diff --git a/src/leap/gui/test_mainwindow_rc.py b/src/leap/gui/test_mainwindow_rc.py index 88ae5854..c2fb3f78 100644 --- a/src/leap/gui/test_mainwindow_rc.py +++ b/src/leap/gui/test_mainwindow_rc.py @@ -1,8 +1,11 @@  import unittest  import hashlib -import sip -sip.setapi('QVariant', 2) +try: +    import sip +    sip.setapi('QVariant', 2) +except ValueError: +    pass  from leap.gui import mainwindow_rc @@ -23,4 +26,4 @@ class MainWindowResourcesTest(unittest.TestCase):      def test_mainwindow_resources_hash(self):          self.assertEqual(              hashlib.md5(mainwindow_rc.qt_resource_data).hexdigest(), -            'd74eb99247b9d5cd2f00b2f695ca6b59') +            'cc7f55e551df55e39c7dbedc1f7de4c2') diff --git a/src/leap/util/dicts.py b/src/leap/util/dicts.py index d8177973..001ca96b 100644 --- a/src/leap/util/dicts.py +++ b/src/leap/util/dicts.py @@ -1,4 +1,5 @@ -# Backport of OrderedDict() class that runs on Python 2.4, 2.5, 2.6, 2.7 and pypy. +# Backport of OrderedDict() class that runs +# on Python 2.4, 2.5, 2.6, 2.7 and pypy.  # Passes Python2.7's test suite and incorporates all the latest updates.  try: @@ -17,9 +18,11 @@ class OrderedDict(dict):      # An inherited dict maps keys to values.      # The inherited dict provides __getitem__, __len__, __contains__, and get.      # The remaining methods are order-aware. -    # Big-O running times for all methods are the same as for regular dictionaries. +    # Big-O running times for all methods are the same as for regular +    # dictionaries. -    # The internal self.__map dictionary maps keys to links in a doubly linked list. +    # The internal self.__map dictionary maps keys to links in a doubly +    # linked list.      # The circular doubly linked list starts and ends with a sentinel element.      # The sentinel element never gets deleted (this simplifies the algorithm).      # Each link is stored as a list of length three:  [PREV, NEXT, KEY]. @@ -42,8 +45,9 @@ class OrderedDict(dict):      def __setitem__(self, key, value, dict_setitem=dict.__setitem__):          'od.__setitem__(i, y) <==> od[i]=y' -        # Setting a new item creates a new link which goes at the end of the linked -        # list, and the inherited dictionary is updated with the new key/value pair. +        # Setting a new item creates a new link which goes at the end +        # of the linked list, and the inherited dictionary is updated +        # with the new key/value pair.          if key not in self:              root = self.__root              last = root[0] @@ -53,7 +57,8 @@ class OrderedDict(dict):      def __delitem__(self, key, dict_delitem=dict.__delitem__):          'od.__delitem__(y) <==> del od[y]'          # Deleting an existing item uses self.__map to find the link which is -        # then removed by updating the links in the predecessor and successor nodes. +        # then removed by updating the links in the predecessor and successor +        # nodes.          dict_delitem(self, key)          link_prev, link_next, key = self.__map.pop(key)          link_prev[1] = link_next @@ -89,8 +94,8 @@ class OrderedDict(dict):      def popitem(self, last=True):          '''od.popitem() -> (k, v), return and remove a (key, value) pair. -        Pairs are returned in LIFO order if last is true or FIFO order if false. - +        Pairs are returned in LIFO order if last is true or FIFO order if +        false.          '''          if not self:              raise KeyError('dictionary is empty') @@ -142,11 +147,13 @@ class OrderedDict(dict):          '''od.update(E, **F) -> None.  Update od from dict/iterable E and F.          If E is a dict instance, does:           for k in E: od[k] = E[k] -        If E has a .keys() method, does:         for k in E.keys(): od[k] = E[k] +        If E has a .keys() method, does:         for k in E.keys(): +                                                    od[k] = E[k]          Or if E is an iterable of items, does:   for k, v in E: od[k] = v -        In either case, this is followed by:     for k, v in F.items(): od[k] = v - +        In either case, this is followed by:     for k, v in F.items(): +                                                    od[k] = v          ''' +          if len(args) > 2:              raise TypeError('update() takes at most 2 positional '                              'arguments (%d given)' % (len(args),)) @@ -169,13 +176,16 @@ class OrderedDict(dict):          for key, value in kwds.items():              self[key] = value -    __update = update  # let subclasses override update without breaking __init__ +    __update = update  # let subclasses override update +                       # without breaking __init__      __marker = object()      def pop(self, key, default=__marker): -        '''od.pop(k[,d]) -> v, remove specified key and return the corresponding value. -        If key is not found, d is returned if given, otherwise KeyError is raised. +        '''od.pop(k[,d]) -> v +        remove specified key and return the corresponding value. +        If key is not found, d is returned if given, +        otherwise KeyError is raised.          '''          if key in self: @@ -232,12 +242,12 @@ class OrderedDict(dict):          return d      def __eq__(self, other): -        '''od.__eq__(y) <==> od==y.  Comparison to another OD is order-sensitive +        '''od.__eq__(y) <==> od==y. +        Comparison to another OD is order-sensitive          while comparison to a regular mapping is order-insensitive. -          '''          if isinstance(other, OrderedDict): -            return len(self)==len(other) and self.items() == other.items() +            return len(self) == len(other) and self.items() == other.items()          return dict.__eq__(self, other)      def __ne__(self, other): | 
