私はPythonのパッケージに取り組んでいます。私はvirtualenvを使用しています。 virtualenvの.pthパスでモジュールのルートへのパスを設定し、コードの開発中にパッケージのモジュールをインポートしてテストを実行できるようにします(質問1:良い方法ですか?)。これは正常に動作します(ここに例を示します、これは私が望む動作です):
(VEnvTestRc) zz@zz:~/Desktop/GitFolders/rc$ python
Python 2.7.12 (default, Jul 1 2016, 15:12:24)
[GCC 5.4.0 20160609] on linux2
Type "help", "copyright", "credits" or "license" for more information.
>>> from rc import ns
>>> exit()
(VEnvTestRc) zz@zz:~/Desktop/GitFolders/rc$ python tests/test_ns.py
issued command: echo hello
command output: hello
ただし、PyTestを使用しようとすると、インポートエラーメッセージが表示されます。
(VEnvTestRc) zz@zz:~/Desktop/GitFolders/rc$ pytest
=========================================== test session starts ============================================
platform linux2 -- Python 2.7.12, pytest-3.0.5, py-1.4.31, pluggy-0.4.0
rootdir: /home/zz/Desktop/GitFolders/rc, inifile:
collected 0 items / 1 errors
================================================== ERRORS ==================================================
________________________________ ERROR collecting tests/test_ns.py ________________________________
ImportError while importing test module '/home/zz/Desktop/GitFolders/rc/tests/test_ns.py'.
Hint: make sure your test modules/packages have valid Python names.
Traceback:
tests/test_ns.py:2: in <module>
from rc import ns
E ImportError: cannot import name ns
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! Interrupted: 1 errors during collection !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
========================================= 1 error in 0.09 seconds ==========================================
(VEnvTestRc) zz@zz:~/Desktop/GitFolders/rc$ which pytest
/home/zz/Desktop/VirtualEnvs/VEnvTestRc/bin/pytest
私は少し困惑していますが、これはインポートエラーを示しているように見えますが、Pythonはそれで問題ないので、なぜPyTestに特に問題があるのですか?理由/救済策への提案(質問2)? PyTestの 'ImportError:cannot import'エラーをGoogleで検索してスタックオーバーフローしましたが、取得したヒットはpythonパスの欠落に関連しており、ここでは問題にならないようです。助言がありますか?
答えを見つけました:
Pytestを使用する予定がある場合は、__init__.py
ファイルをTESTSを含むフォルダーに入れないでください。そのようなファイルが1つあり、それを削除することで問題が解決しました。
これは実際には pytest 'ImportError:No module named YadaYadaYada'のPATHの問題 の2番目の答えに対するコメントに埋もれていたので、私はそれを見なかったので、ここでもっと見やすくなることを願っています。
なぜこれが機能するのか理解できないとは言えませんが、python -m pytest
を実行すると同じ問題が発生し、テストは正常に機能します。
私はvirtualenvで、pytestもグローバルに利用できます:
(proj)tom@neon ~/dev/proj$ type -a python
python is /home/tom/.virtualenvs/proj/bin/python
python is /usr/bin/python
(proj)tom@neon ~/dev/proj$ python -V
Python 3.5.2
(proj)tom@neon ~/dev/proj$ type -a pytest
pytest is /home/tom/.virtualenvs/proj/bin/pytest
pytest is /usr/bin/pytest
(proj)tom@neon ~/dev/proj$ pytest --version
This is pytest version 3.5.0, imported from /home/tom/.virtualenvs/proj/lib/python3.5/site-packages/pytest.py
私はプロジェクトのルートから__init__.pyを削除することでこれを解決しました:
.
├── __init__.py <--- removed
├── models
│ ├── __init__.py
│ ├── address.py
│ ├── appointment.py
│ └── client.py
├── requirements.txt
├── setup.cfg
├── tests
│ ├── __init__.py
│ ├── models
│ │ ├── __init__.py
│ │ ├── appointment_test.py
│ │ └── client_test.py
│ └── other_test.py
└── script.py
私は同じ問題を抱えていましたが、言及したものとは別の理由で:
パッケージが仮想環境にインストールされている間にpy.testをグローバルにインストールしました。
解決策は、pytest
を仮想環境にインストールすることでした。 (Bashと同様にシェルが実行可能ファイルをハッシュする場合は、hash -r
を使用するか、py.test
へのフルパスを使用します)
この問題は、tests.py
ファイルとtests/__init__.py
のテストフォルダーがある場合に発生します。
収集中にpytestはフォルダーを見つけますが、フォルダーからテストファイルをインポートしようとすると、tests.py
ファイルによってインポートの問題が発生します。
修正するには、tests.py
ファイルを削除して、すべてのテストをtests/
フォルダー内に配置します。
特定のケースでは、修正は正確に次のようになります。
/home/zz/Desktop/GitFolders/rc/tests.py
/home/zz/Desktop/GitFolders/rc/tests/__init__.py
が存在することを確認してくださいパッケージを仮想環境にインストールします。
次に、新しいシェルを起動し、仮想環境を再度ソースします。
私の場合、パッケージが同じ名前の別のパッケージ/ディレクトリを指しており、そのパスが実際に必要なフォルダの1レベル上にあるため、インポートエラーが発生しました。これは、一部の人が_init_.pyを削除する必要がある一方で、他の人が元に戻す必要がある理由も説明していると思います。
python
コンソールとpytest
スクリプトの両方にprint(the_root_package.__path__)
(import the_root_package
の後)を置いて、differenceを比較します
ボトムライン:python
を実行すると、インポートするパッケージがpytest
を実行したときのパッケージと異なる場合があります。
同様の問題があり、まったく同じエラーが発生しましたが、原因は異なります。テストコードは正常に実行されましたが、モジュールの古いバージョンに対して実行されました。私のコードの以前のバージョンでは、1つのクラスが存在しましたが、他のクラスは存在しませんでした。コードを更新したら、次を実行してインストールする必要があります。
Sudo pip install ./ --upgrade
更新されたモジュールをインストールすると、pytestを実行すると正しい結果が生成されました(正しいコードベースを使用していたため)。
上記の答えは私にはうまくいきません。 sys.path
(テストモジュール)の上部にあるtest_xxx.py
に見つからなかったモジュールの絶対パスを追加することで解決しました。
import sys
sys.path.append('path')
PytestはPythonモジュールとしてパッケージを読み取っていない可能性がありますが、Pythonは(パスの問題が原因である可能性があります)。 pytestスクリプトのディレクトリを変更するか、PYTHONPATHにモジュールを明示的に追加してください。
または、マシンにPythonの2つのバージョンがインストールされている可能性があります。 Pythonソースのpytestおよび実行するpythonシェルを確認してください。それらが異なる場合(つまりPython 2 vs 3)、source activate
を使用して、モジュールがインストールされているpythonと同じpytestを実行していることを確認します。
同様の問題があり、testsディレクトリにinit。pyファイルを追加したときに機能しました。
元々python 2.7で開発され、現在python 3.xに移行されたpythonコードに関連する場合、問題はおそらくインポートの問題に関連しています。
例えばファイルからオブジェクトをインポートするとき:base
は同じディレクトリにあり、python 2.xで機能します:
from base import MyClass
python 3.xでは、base
フルパスまたは.base
に置き換える必要があります。そうしないと、上記の問題が発生します。だから試してください:
from .base import MyClass
既に.pycファイルがある場合は、削除してみてください。
今日、私はこの問題に遭遇しました、ここで何が起こったのですか:
最初にMacでpytestを実行し(これによりpycファイルが生成されます)、次にプロジェクトdirをマウントした状態でdockerコンテナー(osはAlpine)を起動し、コンテナーでpytestを実行しようとするとImportErrorが発生します。すべてのpycファイルをクリーニングした後、エラーはもうありません。
これが役立つことを願っています。
すべてを試してもエラーが発生する場合は、回避策があります。
pytestがインストールされているフォルダーで、pytest-envフォルダーに移動します。
pyvenv.cfgファイルを開きます。
ファイルの変更include-system-site-packagesからfalseからtrueに変更します。
home = /usr/bin
include-system-site-packages = true
version = 3.6.6
うまくいけば、投票することを忘れないでください。
別の特別なケース:
Toxの使用で問題が発生しました。したがって、私のプログラムは正常に実行されましたが、toxを介した単体テストでは不満が続きました。パッケージをインストールした後(プログラムに必要)、tox.iniのunittestsで使用されるパッケージを追加で指定する必要があります
[testenv]
deps =
package1
package2
...