如何在python安装的软件包中包含文本文件?

时间:2018-11-24 00:00:42

标签: python include install text-files setup.py

我创建了一个看起来像这样的python包:

/command
    /command
        module.py
        __main__.py
    README.md
    setup.py
    file.txt

要安装我,请运行:

sudo python setup.py install

现在我打电话给

$ command

它显示此错误:

FileNotFoundError: [Errno 2] No such file or directory: '../file.txt'

大约包含模块 setup.py __ main __。py module.py

setup.py

import setuptools

setuptools.setup(
    name='command',
    ...
    entry_points={
        'console_scripts': [
            'command = command.__main__:main'
        ]
    }
)

__ main __。py

from . import module

def main():
    module.run('arg')

module.py

import shutil

# copy file.txt from command project into the directory where it is running
def run(arg):
    shutil.copyfile('../file.txt', './file.txt')

通过以下方式安装此软件包后:

sudo python setup.py install

然后在命令行中调用程序

$ command

我收到下面的错误

FileNotFoundError: [Errno 2] No such file or directory: '../file.txt'

如何能够查看和使用属于已安装软件包的文件,但是我想在运行该程序的环境中使用它?

编辑:

This a simplification of the problem you can download and test:

https://github.com/mctrjalloh/project_initializer

2 个答案:

答案 0 :(得分:0)

因此,经过大量研究,我找到了解决方案以及它的工作方式。这有点令人困惑,其他stackoverflow答案都没有那么解释。我想在这里尝试:

我已经为此目的创建了一个示例项目,以演示和测试解决方案。我提出了两种解决方案:一种使用setup()函数的 data_files 参数,另一种使用我最喜欢的 package_data 参数。

Here is the link to the github repo you can download and test

要在安装后使用它,请运行

proj init <some-name>

但简单来说,每种方法都有最重要的模块。

使用数据文件 =参数方法:

项目结构:

project_initializer
    project_initializer
        __init__.py
        __main__.py
        init.py
    README.md
    setup.py

setup.py

import setuptools
import os
import sys


PROJECT_NAME = "project_initializer"
DATA_DIR = os.path.join(
    sys.prefix, "local/lib/python3.6/dist-packages", PROJECT_NAME)


setuptools.setup(
    name='project_initializer',
    version='0.1.0',
    packages=setuptools.find_packages(),
    install_requires=[
        'docopt'
    ],
    data_files=[         # is the important part
        (DATA_DIR, [
            "README.md",
            ".gitignore"
        ])               
    ],
    entry_points={
        'console_scripts': [
            'proj = project_initializer.__main__:main'
        ]
    }
)

init.py

import subprocess
import os
import shutil
import sys

"""Create a new project and initialize it with a .gitignore file
@params project: name of a project to be initialized
effects:
    /project
        README.md
    README.md in the created project directory must be the same as the README.md in THIS directory 
"""

PROJECT_NAME = "project_initializer"
DATA_DIR = os.path.join(
    sys.prefix, "local/lib/python3.6/dist-packages", PROJECT_NAME)


def run(project):
    os.mkdir(project)
    shutil.copyfile(os.path.join(DATA_DIR, "README.md"),
                    f"{project}/README.md")  # problem solved


if __name__ == '__main__':
    run("hello-world")

使用 package_data = 参数方法(我更喜欢)

项目结构:

project_initializer
    project_initializer
        data/
            README.md  # the file we want to copy
        __init__.py
        __main__.py
        init.py
    README.md
    setup.py

setup.py

import setuptools


setuptools.setup(
    name='project_initializer',
    version='0.1.0',
    packages=setuptools.find_packages(),
    package_dir={'project_initializer': 'project_initializer'}, # are the ... 
    package_data={'project_initializer': [ # ... important parameters
        'data/README.md', 'data/.gitignore']},
    install_requires=[
        'docopt'
    ],
    entry_points={
        'console_scripts': [
            'proj = project_initializer.__main__:main'
        ]
    }
)

init.py

import subprocess
import os
import shutil
import sys

PROJECT_DIR = os.path.dirname(__file__)

"""Create a new project and initialize it with a .gitignore file
@params project: name of a project to be initialized
effects:
    /project
        .gitignore
    .gitignore in the created project directory must be the same as the gitignore in THIS directory 
"""


def run(project):
    os.mkdir(project)
    shutil.copyfile(os.path.join(PROJECT_DIR, 'data/README.md'),
                    f"{project}/README.md")  # problem solved


if __name__ == '__main__':
    run("hello-world")

之所以我选择最后一种方法,是因为您不必在setup.py模块中导入任何内容,这可能是一种不好的做法。我想什么都不应该导入setup.py文件中,因为它是主程序包的外部文件。

要详细了解两个参数之间的区别,请查看python文档

Using data_files= argument

Using package_data= argument

答案 1 :(得分:-1)

默认情况下,软件包中仅包含python文件。

要包含更多内容,请添加MANIFEST.in并列出文件。 For example

Here is a comprehensive tutorial