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