'Python Namespace Packages in Python3
The topic of namespace packages seems a bit confusing for the uninitiated, and it doesn't help that prior versions of Python have implemented it in a few different ways or that a lot of the Q&A on StackOverflow are dated. I am looking for a solution in Python 3.5
or later.
#The scenario: I'm in the process of refactoring a bunch of Python code into modules and submodules, and working to get each of these projects set up to operate independently of each other while sitting in the same namespace.
We're eventually going to be using an internal PyPi server, serving these packages to our internal network and don't want to confuse them with external (public) PyPi packages.
Example: I have 2 modules, and I would like to be able to perform the following:
from org.client.client1 import mod1
from org.common import config
The reflected modules would be separated as such:
Repository 1:
org_client_client1_mod1/
setup.py
mod1/
__init__.py
somefile.py
Repository 2:
org_common_config/
setup.py
config/
__init__.py
someotherfile.py
My Git repositories are already setup as org_client_client1_mod1
and org_common_config
, so I just need to perform the setup on the packaging and __init__.py
files, I believe.
Questions:
#1
With the
__init__.py
, which of these should I be using (if any)?:from pkgutil import extend_path __path__ = extend_path(__path__, __name__)
Or:
import pkg_resources pkg_resources.declare_namespace(__name__)
#2
With
setup.py
, do I still need to add thenamespace_modules
parameter, and if so, would I usenamespace_modules=['org.common']
, ornamespace_modules=['org', 'common']
?
#3
Could I forgo all of the above by just implementing this differently somehow? Perhaps something simpler or more "pythonic"?
Solution 1:[1]
This is a tough subject. All the -
's, _
's, and __init__.py
's everywhere don't exactly make it easy on us.
First, I'll answer your questions:
With the
__init__.py
, which of these should I be using (if any)?
__init__.py
can be completely empty, it just needs to be in the correct place. Namely (pun) they should be in any subpackage containing python code (excludingsetup.py
.) Follow those rules and you should be fine.
With setup.py, do I still need to add the
namespace_modules
parameter, and if so, would I usenamespace_modules=['org.common']
, ornamespace_modules=['org', 'common']
?
- Nope! Only
name=
andpackages=
. However, note the format of thepackages=
arg compared against the directory structure. - Here's the format of the
package=
arg: - Here's the corresponding directory structure:
Could I forgo all of the above by just implementing this differently somehow? Perhaps something simpler or more "pythonic"?
- If you want to be able to install multiple features individually, but under the same top-level namespace, you're on the right track.
I'll spend the rest of this answer re-implementing your namespace package in native format:
I'll put all helpful documentation I've been able to find at the bottom of the post.
K so I'm going to assume you want native namespace packages. First let's look at the current structure of your 2 repos:
org_client_client1_mod1/
setup.py
mod1/
__init__.py
somefile.py
&
org_common_config/
setup.py
config/
__init__.py
someotherfile.py
This^ would be too easy!!!
To get what you want:
My brain isn't elastic enough to know if we can go 3-levels deep with namespace packages, but to do what you want, here's what I'm pretty sure you'd want to do:
org-client/
setup.py
org/
client/
client1/
__init__.py
mod1/
__init__.py
somefile.py
&
org-common-but-also-note-this-name-doesnt-matter/
setup.py
org/
common/
__init__.py
config/
__init__.py
someotherfile.py
Basically then the key is going to be specifying the correct name=
& packages=
args to stuptools.setup()
inside of each setup.py
.
These are going to be:
name='org_client',
...
packages=['org.client']
&
name='org_common'
...
packages['org.common']
respectively.
Then just install each one with pip install .
inside each top-level dir.
Installing the first one will give you access to the somefile.py
module, and installing the second will give you access to someotherfile.py
. It also won't get confused about you trying to install 2 packages named org
in the same environment.
K so the most helpful section of the docs: https://packaging.python.org/guides/packaging-namespace-packages/#packaging-namespace-packages
And then here's how I actually came to understand this: https://github.com/pypa/sample-namespace-packages/tree/master/native
Sources
This article follows the attribution requirements of Stack Overflow and is licensed under CC BY-SA 3.0.
Source: Stack Overflow
Solution | Source |
---|---|
Solution 1 |