Пакеты Python? - PullRequest
       21

Пакеты Python?

9 голосов
/ 27 августа 2009

Хорошо, я думаю, что бы я ни делал неправильно, это, вероятно, ослепительно очевидно, но я не могу понять это. Я прочитал и перечитал раздел учебника по пакетам, и единственное, что я могу понять, это то, что это не сработает, потому что я выполняю его напрямую. Вот настройка каталога:

eulerproject/
  __init__.py
  euler1.py
  euler2.py
  ...
  eulern.py
  tests/
    __init__.py
    testeulern.py

Вот содержимое testeuler12.py (первый написанный мной тестовый модуль):

import unittest
from .. import euler12

class Euler12UnitTests(unittest.TestCase):


    def testtriangle(self):
        """
        Ensure that the triangle number generator returns the first 10
        triangle numbers.

        """
        self.seq = [1,3,6,10,15,21,28,36,45,55]
        self.generator = euler12.trianglegenerator()
        self.results = []
        while len(self.results) != 10:
            self.results.append(self.generator.next())
        self.assertEqual(self.seq, self.results)

    def testdivisors(self):
        """
        Ensure that the divisors function can properly factor the number 28.

        """
        self.number = 28
        self.answer = [1,2,4,7,14,28]
        self.assertEqual(self.answer, euler12.divisors(self.number))


if __name__ == '__main__':

    unittest.main()

Теперь, когда я выполняю это из IDLE и из командной строки, находясь в каталоге, я получаю следующую ошибку:

Traceback (most recent call last):
  File "C:\Documents and Settings\jbennet\My Documents\Python\eulerproject\tests\testeuler12.py", line 2, in <module>
    from .. import euler12
ValueError: Attempted relative import in non-package

Я думаю, что проблема в том, что, поскольку я запускаю его напрямую, я не могу выполнить относительный импорт (потому что __name__ меняется, и мое смутное понимание описания пакетов состоит в том, что __name__ является частью того, как он говорит в каком пакете он находится), но в таком случае, что вы, ребята, предлагаете, как импортировать производственный код, хранящийся на 1 уровень выше тестового кода?

Ответы [ 2 ]

10 голосов
/ 27 августа 2009

У меня была такая же проблема. Теперь я использую nose для запуска моих тестов, и относительный импорт корректно обрабатывается.

Да, весь этот относительный импорт очень запутан.

8 голосов
/ 27 августа 2009

Обычно у вас есть каталог, именем которого является имя вашего пакета, где-то на вашей PYTHONPATH. Например:

eulerproject/
    euler/
        __init__.py
        euler1.py
        ...
        tests/
            ...
    setup.py

Затем вы можете либо установить эту систему в масштабе всей системы, либо обязательно установить PYTHONPATH=/path/to/eulerproject/:$PYTHONPATH при вызове сценария.

Тогда будет работать такой абсолютный импорт:

from euler import euler1

Редактировать :

Согласно документации по Python, «модули, предназначенные для использования в качестве основного модуля приложения Python, всегда должны использовать абсолютный импорт». ( Cite )

Таким образом, тестовый набор, подобный nose, упомянутый в другом ответе, работает, поскольку он импортирует пакеты, а не запускает их из командной строки.

Если вы хотите сделать что-то вручную, ваш исполняемый скрипт должен находиться вне иерархии пакетов, например:

eulerproject/
    runtests.py
    euler/
        __init__.py
        euler1.py
        ...
        tests/
            __init__.py
           testeulern.py

Теперь runtests.py можно сделать from euler.tests.testeulern import TestCase и testeulern.py можно сделать from .. import euler1

...