This is incomplete; please consider contributing to the documentation effort.
In order for Hy and Python to work together as nicely as it does, Hy code needs to be able to import Python code, and vice versa.
This is done through Python’s import hooks. However, since the implementation and feature set available differs between the various versions of Python, its worth noting how the system works and the various limitations and quirks, so not to fall into certain pitfalls.
- State “DONE” from “TODO” [2018-03-31 Sat 22:45]
Python new import hooks, as specified in PEP 302, allows Hy to integrate with Python seamlessly, with Hy code able to import Python code and Python code able to import Hy code.
The new import hooks allows Hy to hook into the Python import hooks and customize them. When a Hy module is requested, Hy has a chance to look for its own modules, evaluate them, and load them into Python.
The implementation of the Python import system differ greatly between Python 3 and Python 2, and thus impacts Hy’s ability to integrate into the wider Python ecosystem.
Python 3’s import system is backed importlib
and most of its
import mechanism is pure Python code.
The Python 2 import system, in contrast, is backed by the deprecated
and lower level imp
library and a mix of exposed Python code and
built-in functionality.
https://docs.python.org/3/reference/import.html#finders-and-loaders
If the named module is not found in
sys.modules
, then Python’s import protocol is invoked to find and load the module. This protocol consists of two conceptual objects,finders
andloaders
. A finder’s job is to determine whether it can find the named module using whatever strategy it knows about. Objects that implement both of these interfaces are referred to as importers - they return themselves when they find that they can load the requested module.
Python always first tries to find a matching entry inside
sys.modules
, and if one is found, returns that.
Should the module not be found, then there are slight differences between the two platforms.
On Python 3, the process starts by looking at sys.meta_path
. This
variable contains an array of module loaders. Python tries them,
one at a time, until a loader returns successfully loads a
module. Typically contains three default entries:
- The
BuiltinImporter
which handles finding builtin modules, such asbuiltins
orsys
. - The
FrozenImporter
which handles finding frozen modules. - And the
PathFinder
which handles finding Python modules in the filesystem.
There may be a few more entries, depending on the setup, as
packages like six
and pkg_resources
hook into this mechanism to
implement some of their functionality. For example, six
hooks
into the import system to make Python 3 import paths work on
Python 2.
Hy injects a Hy module finder into the sys.meta_path
array right
before the builtin PathFinder
. The reasons for why Hy code must
be attempted first is elaborated on below.
Finders do not actually load modules. If they can find the named module, they return a module spec, an encapsulation of the module’s import-related information, which the import machinery then uses when loading the module.
The module finder then is split into three components:
- The
HyPathFinder
- The
HyFileFinder
- And the
HyLoader
The three parts provide a ModuleSpec
which Python is able to then
convert into a module.
Python 2 also goes through the sys.meta_path
list as the first
step of its import process, this time calling find_module
.
By default on Python 2, this list is empty, but Hy will hook itself into the interpreter by injecting an entry into this list.
- The Hy metaloader must be specified first, and can’t be fallen back on
- Valid Hy modules could be confused as valid Python namespace modules. This means that a Hy module could correctly import, but not contain any attributes.
- We can’t support namespace modules because otherwise valid Python modules start looking like Hy namespace modules (reverse of the problem above).
While its generally not a good idea to mix Hy and Python code in the same package, as it can lead to confusing behaviours, it technically works under Python 3, but not under Python 2.
From how I understand the reasoning behind namespace packages, they should not : having a
__init__.hy
in a folder should be enough for the package to NOT be a namespace. => HyPathFinder.find_spec should take care of that. See https://github.com/asmodehn/filefinder2/blob/master/filefinder2/_filefinder2.py#L142 for background on this.Goal here is that someone needing a
PathFinder
just inherits fromfilefinder2.machinery.PathFinder
and override what needs to be overridden.An example of an override (making a proper package based on content, not on
__init__.py
file presence ): https://github.com/pyros-dev/rosimport/blob/master/rosimport/_ros_directory_finder.py#L42The namespace logic implemented is pretty tricky, but this is suitable to have (to follow python, and not introduce hard to spot differences), and (intuitively) should be possible.