用 pytest 正确导入

我刚刚设置了在 Python 2.6中使用 pytest。到目前为止,除了处理“ import”语句之外,它工作得很好: 我似乎无法让 pytest 像我的程序那样响应导入。

我的目录结构如下:

src/
main.py
util.py
test/
test_util.py
geom/
vector.py
region.py
test/
test_vector.py
test_region.py

要运行,我从 src/调用 python main.py

在 main.py 中,我用

from geom.region import Region
from geom.vector import Vector

在 vector.py 中,我用

from geom.region import Region

当我以标准运行方式运行代码时,这些都可以很好地工作。但是,当我从 src/调用“ py.test”时,它始终存在导入错误。


几个问题及解决的尝试

我的第一个问题是,当运行“ test/test _ foo.py”时,py.test 不能直接“ import foo.py”。我用“小恶魔”工具解决了这个问题。在“ test _ util.py”中:

import imp
util = imp.load_source("util", "util.py")

这对于许多文件都非常有效。它似乎还暗示,当 pytest 运行“ path/test/test _ foo.py”来测试“ path/foo.py”时,它基于目录“ path”。

但是,这对于“ test _ vector.py”是失败的。Pytest 可以查找和导入 vector模块,但是它可以定位 vector的任何导入。使用 pytest 时,以下导入(来自“ vector.py”)都会失败:

from geom.region import *
from region import *

这两者都给出了形式上的错误

ImportError: No module named [geom.region / region]

我不知道接下来要做什么来解决这个问题; 我对 Python 中导入的理解有限。

在使用 pytest 时,处理进口的正确方法是什么?


编辑: 极端拙劣的解决方案

vector.py中,我将 import 语句从

from geom.region import Region

变得简单

from region import Region

这使得导入相对于“ vector.py”目录。

接下来,在“ test/test _ vector.py”中,我将“ vector.py”目录添加到路径中,如下所示:

import sys, os
sys.path.append(os.path.realpath(os.path.dirname(__file__)+"/.."))

这使 Python 能够从“ geom/test/test _ vector.py”中找到“ . ./region.py”。

这可以工作,但是它似乎非常有问题,因为我正在向路径添加大量新目录。我要找的是

1)与 pytest 兼容的导入策略,或

2) pytest 中的一个选项,使其与我的导入策略兼容

所以我把这个问题留给这些类型的答案。

74106 次浏览

import looks in the following directories to find a module:

  1. The home directory of the program. This is the directory of your root script. When you are running pytest your home directory is where it is installed (/usr/local/bin probably). No matter that you are running it from your src directory because the location of your pytest determines your home directory. That is the reason why it doesn't find the modules.
  2. PYTHONPATH. This is an environment variable. You can set it from the command line of your operating system. In Linux/Unix systems you can do this by executing: 'export PYTHONPATH=/your/custom/path' If you wanted Python to find your modules from the test directory you should include the src path in this variable.
  3. The standard libraries directory. This is the directory where all your libraries are installed.
  4. There is a less common option using a pth file.

sys.path is the result of combining the home directory, PYTHONPATH and the standard libraries directory. What you are doing, modifying sys.path is correct. It is something I do regularly. You could try using PYTHONPATH if you don't like messing with sys.path

I was wondering what to do about this problem too. After reading this post, and playing around a bit, I figured out an elegant solution. I created a file called "test_setup.py" and put the following code in it:

import sys, os
sys.path.append(os.path.dirname(os.path.abspath(__file__)))

I put this file in the top-level directory (such as src). When pytest is run from the top-level directory, it will run all test files including this one since the file is prefixed with "test". There are no tests in the file, but it is still run since it begins with "test".

The code will append the current directory name of the test_setup.py file to the system path within the test environment. This will be done only once, so there are not a bunch of things added to the path.

Then, from within any test function, you can import modules relative to that top-level folder (such as import geom.region) and it knows where to find it since the src directory was added to the path.

If you want to run a single test file (such as test_util.py) instead of all the files, you would use:

pytest test_setup.py test\test_util.py

This runs both the test_setup and test_util code so that the test_setup code can still be used.

The issue here is that Pytest walks the filesystem to discover files that contain tests, but then needs to generate a module name that will cause import to load that file. (Remember, files are not modules.)

Pytest comes up with this test package name by finding the first directory at or above the level of the file that does not include an __init__.py file and declaring that the "basedir" for the module tree containing a module generated from this file. It then adds the basedir to sys.path and imports using the module name that will find that file relative to the basedir.

There are some implications of this of which you should beware:

  1. The basepath may not match your intended basepath in which case the module will have a name that doesn't match what you would normally use. E.g., what you think of as geom.test.test_vector will actually be named just test_vector during the Pytest run because it found no __init__.py in src/geom/test/ and so added that directory to sys.path.

  2. You may run into module naming collisions if two files in different directories have the same name. For example, lacking __init__.py files anywhere, adding geom/test/test_util.py will conflict with test/test_util.py because both are loaded as import test_util.py, with both test/ and geom/test/ in the path.

The system you're using here, without explicit __init__.py modules, is having Python create implicit namespace packages for your directories. (A package is a module with submodules.) Ideally we'd configure Pytest with a path from which it would also generate this, but it doesn't seem to know how to do that.

The easiest solution here is simply to add empty __init__.py files to all of the subdirectories under src/; this will cause Pytest to import everything using package/module names that start with directory names under src/.

The question How do I Pytest a project using PEP 420 namespace packages? discusses other solutions to this.

If you include an __init__.py file inside your tests directory, then when the program is looking to set a home directory it will walk 'upwards' until it finds one that does not contain an init file. In this case src/.

From here you can import by saying :

from geom.region import *

you must also make sure that you have an init file in any other subdirectories, such as the other nested test directory

Are so late to answer that question but usining python 3.9 or 3.10 u just need to add __init__.py folder in tests folders.

When u add this file python interprets this folders as a module. Wold be like this

src/
main.py
util.py
test/
__init__.py
test_util.py
geom/
vector.py
region.py
test/
__init__.py
test_vector.py
test_region.py

so u just run pytest.

Sorry my poor english

Not the best solution, but maybe the fastest one:

cd path/python_folder


python -m pytest python_file.py