Skip to main content

More Imports! - Delayed importing

Project description

More Imports! - Delayed importing

A few methods to make late importing cleaner

Branch Status
master Build Status
dev Build Status

Problem

Splitting code into modules is nice, but it can result in cyclic dependencies.

foos.py

from bars import bar

def foo():
    bar()

bars.py

from foos import foo

def bar():
    foo()

We are not concerned with the infinite recursion; this is only for demonstrating cyclic dependencies.

More Imports!

Solution #1: Use expect/export pattern

All your cyclic dependencies are covered with this one pattern: Break cycles by expecting a name in the first module, and let the second module export to the first when the value is available

foos.py

from mo_imports import expect

bar = expect("bar")

def foo():
    bar()

bars.py

from mo_imports import export
from foos import foo

def bar():
    foo()

export("bars", bar)

Benefits

  • every expect is verified to match with an export (and visa-versa)
  • using an expected variable before export raises an error
  • code is run only once, at module load time, not later
  • methods do not run import code
  • all "imports" are at the top of the file

Solution #2: Use delay_import

Provide a proxy which is responsible for import upon first use of the module variable.

foos.py

from mo_imports import delay_import

bar = delay_import("bars.bar")

def foo():
    bar()

bars.py

from foos import foo

def bar():
    foo()

Benefits

  • cleaner code
  • costly imports are delayed until first use

WARNING

Requires any of __call__, __getitem__, __getattr__ to be called to trigger the import. This means sentinals, placeholders, and default values can NOT be imported using delay_import()

Other solutions

If you do not use mo-imports your import cycles can be broken using one of the following common patterns:

Bad Solution: Keep in single file

You can declare yet-another-module that holds the cycles

foosbars.py

    def foo():
        bar()

    def bar():
        foo()

but this breaks the code modularity

Bad Solution: Use end-of-file imports

During import, setup of the first module is paused while it imports a second. A bottom-of-file import will ensure the first module is mostly setup to be used by the second.

foos.py

def foo():
    bar()

from bars import bar

bars.py

def bar():
    foo()

from foos import foo

Linters do not like this pattern: You may miss imports, since these are hiding at the bottom.

Bad Solution: Inline import

Import the name only when it is needed

foos.py

def foo():
    from bars import bar
    bar()

bars.py

def bar():
    from foos import foo
    foo()

This is fine for rarely run code, but there is an undesirable overhead because import is checked everytime the method is run. You may miss imports because they are hiding inline rather than at the top of the file.

Bad Solution: Use the _late_import() pattern

When other bad solutions do not work work, then importing late is the remaining option

foos.py

from bars import bar

def foo():
    bar()

bars.py

foo = None

def _late_import():
    global foo
    from foos import foo
    _ = foo

def bar():
    if not foo:
        _late_import()
    foo()

Placeholders variables are added, which linters complain about type. There is the added _late_import() method. You risk it is not run everywhere as needed. This has less overhead than an inline import, but there is still a check.

More on importing

Importing a complex modular library is still hard; the complexity comes from the the order other modules declare their imports; you have no control over which of your modules will be imported first. For example, one module may

from my_lib.bars import bar
from my_lib.foos import foo

another module may choose the opposite order

from my_lib.foos import foo
from my_lib.bars import bar

Ordering imports

With cyclic dependencies, ordering the imports can get tricky. Here are some rules

  • choose your principle modules and the order you want them imported.
  • your remaining modules are assumed to be imported in alphabetical order (as most linters prefer)
  • use top level __init__.py to control the order of imports
  • encourage third party modules to use this top level module. for example
    from my_lib import foo, bar
    
  • finally, use mo_imports to break cycles

Project details


Release history Release notifications | RSS feed

Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

mo_imports-7.685.25166.tar.gz (12.2 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

mo_imports-7.685.25166-py3-none-any.whl (11.8 kB view details)

Uploaded Python 3

File details

Details for the file mo_imports-7.685.25166.tar.gz.

File metadata

  • Download URL: mo_imports-7.685.25166.tar.gz
  • Upload date:
  • Size: 12.2 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/6.1.0 CPython/3.13.1

File hashes

Hashes for mo_imports-7.685.25166.tar.gz
Algorithm Hash digest
SHA256 d5c44eca0f5db99f7443548759f49094f1c1d5c348176826c460a8ba37429202
MD5 680cec1e15d7a77ef09798ff422de4f5
BLAKE2b-256 0c6338af1440ee68413e8b8965070e0939b9dbce68fced5a3a0073a4ffea64c9

See more details on using hashes here.

File details

Details for the file mo_imports-7.685.25166-py3-none-any.whl.

File metadata

File hashes

Hashes for mo_imports-7.685.25166-py3-none-any.whl
Algorithm Hash digest
SHA256 e98ecf92f88402fb2a1f9693955c8451419260c6e54ee9037c1a857ebc80b774
MD5 b080f3b42f9f95af3801544a0ac6084c
BLAKE2b-256 555defb6a6b25ccdd359d12ae34c71dc39a1edf267524eb77fa6a5e58aa88693

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page