Rev 5440: Update the import sections in code-style.txt in file:///home/vila/src/bzr/experimental/imports/

Vincent Ladeuil v.ladeuil+lp at free.fr
Wed Sep 22 16:36:02 BST 2010


At file:///home/vila/src/bzr/experimental/imports/

------------------------------------------------------------
revno: 5440
revision-id: v.ladeuil+lp at free.fr-20100922153602-6dfrhhek82fred3p
parent: pqm at pqm.ubuntu.com-20100921104823-0jks3g5o1bahesyq
committer: Vincent Ladeuil <v.ladeuil+lp at free.fr>
branch nick: imports
timestamp: Wed 2010-09-22 17:36:02 +0200
message:
  Update the import sections in code-style.txt
-------------- next part --------------
=== modified file 'doc/developers/code-style.txt'
--- a/doc/developers/code-style.txt	2010-09-13 08:20:57 +0000
+++ b/doc/developers/code-style.txt	2010-09-22 15:36:02 +0000
@@ -141,8 +141,119 @@
   function runs.  Import statements have a cost, so try to make sure
   they don't run inside hot functions.
 
-* Module names should always be given fully-qualified,
-  i.e. ``bzrlib.hashcache`` not just ``hashcache``.
+* Symbols should not be imported from modules. Use ``import <module>`` then
+  ``<module>.symbol`` instead of ``from <module> import symbol`` then
+  ``symbol``. While the former style is often seen in other python
+  projects, it has some drawbaks that led to obscure bugs on several
+  occasions in bzr. 
+
+  The bad points of the ``from <module> import symbol`` style includes:
+
+  * The symbol list for each import is harder to keep up to date.
+
+  * The imported symbols can be imported by other modules without realizing
+    the aliasing that is occurring.
+
+  * It forbids overriding the imported symbol for test purposes. Or it
+    makes it harder and can lead to very obscure failures.
+
+  * Our `Lazy Imports`_ implementation can fail.
+
+  The good points of the ``<module>.symbol`` style includes:
+
+  * The import list at the top-level of the file is shorter hence easier to
+    read and provides a better overview of the relationship of the module
+    with the imported modules.
+
+  * The code refers directly to the symbol used, avoiding a trip to the
+    top-level to know where the symbol is coming from. This applies both to
+    the casual reader and to the reviewers when a patch doesn't modify the
+    imports (which it the vast majority). In essence it makes the code
+    easier to understand while requiring typing more (the module prefix),
+    which is a Good Thing because the code is written once but read many
+    times.
+
+  * It makes it easier to move code from one module to another. You don't
+    need to sort out which symbols need to be copied across the new module
+    or deleted in the old, you only have to care about modules (once).
+
+  There are a few exceptions to this rule though:
+
+  * If symbols are *exported* on purpose to provide a clearer API. The
+    ``bzrlib.tests`` module exports ``TestCase`` and ``TestSuite`` from the
+    ``bzrlib.tests.TestUtils`` for example, the rationale being that it's
+    easier to get these objects from the former.
+
+  * We use::
+      from bzrlib.lazy_import import lazy_import
+      lazy_import(globals(), """
+
+    not::
+      from bzrlib import lazy_import
+      lazy_import.lazy_import(globals(), """
+
+    Arguably the function ``lazy_import`` being named like its module
+    doesn't help here.
+
+* Also, when importing a module into the local namespace, which is likely
+  to clash with variable names (``transport``, ``revision``, etc), it is
+  recommended to prefix it as ``_mod_<module>``. This makes it clearer that
+  the variable is a module, and these objects should be hidden anyway,
+  since they shouldn't be imported into other namespaces.
+
+* Moving from the ``from <module> import symbol`` style is a work in
+  progress, submissions should avoid using it for new code but should not
+  either includes huge cleanups that obscure the purpose of the
+  proposal. When in doubt, use the ``<module>.symbol`` without modifying
+  the ``from <module> import symbol`` part.
+
+Lazy Imports
+============
+
+To make startup time faster, we use the ``bzrlib.lazy_import`` module to
+delay importing modules until they are actually used. ``lazy_import`` uses
+the same syntax as regular python imports. So to import a few modules in a
+lazy fashion do::
+
+  from bzrlib.lazy_import import lazy_import
+  lazy_import(globals(), """
+  import os
+  import subprocess
+  import sys
+  import time
+
+  from bzrlib import (
+     errors,
+     transport,
+     revision as _mod_revision,
+     )
+  """)
+
+At this point, all of these exist as a ``ImportReplacer`` object, ready to
+be imported once a member is accessed
+
+While it is possible for ``lazy_import()`` to import members of a module
+when using the ``from module import member`` syntax, it is recommended to
+only use that syntax to load sub modules ``from module import submodule``.
+This is because variables and classes can frequently be used without
+needing a sub-member for example::
+
+  lazy_import(globals(), """
+  from module import MyClass
+  """)
+
+  def test(x):
+      return isinstance(x, MyClass)
+
+This will incorrectly fail, because ``MyClass`` is a ``ImportReplacer``
+object, rather than the real class.
+
+It also is incorrect to assign ``ImportReplacer`` objects to other variables.
+Because the replacer only knows about the original name, it is unable to
+replace other variables. The ``ImportReplacer`` class will raise an
+``IllegalUseOfScopeReplacer`` exception if it can figure out that this
+happened. But it requires accessing a member more than once from the new
+variable, so some bugs are not detected right away.
 
 
 Naming
@@ -273,61 +384,6 @@
 right one to run is selected by asking each class, in reverse order of
 registration, whether it ``.is_compatible`` with the relevant objects.
 
-Lazy Imports
-============
-
-To make startup time faster, we use the ``bzrlib.lazy_import`` module to
-delay importing modules until they are actually used. ``lazy_import`` uses
-the same syntax as regular python imports. So to import a few modules in a
-lazy fashion do::
-
-  from bzrlib.lazy_import import lazy_import
-  lazy_import(globals(), """
-  import os
-  import subprocess
-  import sys
-  import time
-
-  from bzrlib import (
-     errors,
-     transport,
-     revision as _mod_revision,
-     )
-  import bzrlib.transport
-  import bzrlib.xml5
-  """)
-
-At this point, all of these exist as a ``ImportReplacer`` object, ready to
-be imported once a member is accessed. Also, when importing a module into
-the local namespace, which is likely to clash with variable names, it is
-recommended to prefix it as ``_mod_<module>``. This makes it clearer that
-the variable is a module, and these object should be hidden anyway, since
-they shouldn't be imported into other namespaces.
-
-While it is possible for ``lazy_import()`` to import members of a module
-when using the ``from module import member`` syntax, it is recommended to
-only use that syntax to load sub modules ``from module import submodule``.
-This is because variables and classes can frequently be used without
-needing a sub-member for example::
-
-  lazy_import(globals(), """
-  from module import MyClass
-  """)
-
-  def test(x):
-      return isinstance(x, MyClass)
-
-This will incorrectly fail, because ``MyClass`` is a ``ImportReplacer``
-object, rather than the real class.
-
-It also is incorrect to assign ``ImportReplacer`` objects to other variables.
-Because the replacer only knows about the original name, it is unable to
-replace other variables. The ``ImportReplacer`` class will raise an
-``IllegalUseOfScopeReplacer`` exception if it can figure out that this
-happened. But it requires accessing a member more than once from the new
-variable, so some bugs are not detected right away.
-
-
 The Null revision
 =================
 



More information about the bazaar-commits mailing list