当前位置: 首页 > news >正文

python发包

Python 中我们经常会用到第三方的包,默认情况下,用到的第三方工具包基本都是从 Pypi.org 里面下载。这些第三方的包都是开发者们发布的自己的库。我们有自己的想法,或者有一些常用的方法想要分享出去,就可以发布自己的库,也就是我们常说的造轮子。

PyPI (Python Package Index) 是 python 官方的第三方库的仓库,所有人都可以下载第三方库或上传自己开发的库到PyPI。PyPI 推荐使用 pip 包管理器来下载第三方库。截至目前,PyPI 已经有 574,662 个项目,很多知名项目都发布在上面。

造轮子的步骤:

  • 包源代码开发
  • git 版本管理
  • 编写setup.py
  • 编写说明文档
  • 发布到 Pypi
  • 后续维护升级

1 包源代码开发

包的功能可以使各种各样的,关于包的源代码编写就不过多阐述,这里就只是写了一个件简单的例子,用来输出Hello Pypi。项目工程地址:https://github.com/Taot-chen/hellopypi/tree/main

1.1 创建项目必须文件
touch README.md hellopypi.py setup.py

文件结构:

hellopypi/
├── hellopypi.py
├── README.md
└── setup.py
1.2 创建 git 仓库

现在我们已经创建了项目结构,下面将初始化一个 GitHub 存储库来托管代码:

git init
git add *
git commit -m "init repo"
git branch -M main
git remote set-url origin https://<your_token>github.com/<USERNAME>/hellopypi.git
git push -u origin main

也可以通过在 github 手动建好仓库之后,再通过 clone 建好的仓库,之后再往仓库添加文件的方式。

1.3 包源代码开发

这里的主程序就是前面的hellopypi.py,里面的内容很简单:

__version__ = '0.1.0'def hello_pypi():print("Hello Pypi!")def main():hello_pypi()if __name__ == '__main__':main()
1.4 编写setup.py

setup.py是每个能从 PyPi 上能下载到的库都有的文件,它是发布的关键所在。

kennethreitz 大神编写了一个 for human 的setup.py模板,项目地址:传送门,只需要把它复制过来,修改自己项目需要的地方即可,不需要额外的编写setup.cfg等其他文件。

我这里修改完的内容如下:

#!/usr/bin/env python
# -*- coding: utf-8 -*-# Note: To use the 'upload' functionality of this file, you must:
#   $ pipenv install twine --devimport io
import os
import sys
from shutil import rmtreefrom setuptools import find_packages, setup, Command# Package meta-data.
NAME = 'hellopypi'
DESCRIPTION = 'print Hello Pypi in terminal.'
URL = 'https://github.com/Taot-chen/hellopypi'
EMAIL = 'oehuosi@foxmail.com'
AUTHOR = 'oehuosi'
REQUIRES_PYTHON = '>=3.6.0'
VERSION = '0.1.0'# What packages are required for this module to be executed?
REQUIRED = [# 'requests', 'maya', 'records',
]# What packages are optional?
EXTRAS = {# 'fancy feature': ['django'],
}# The rest you shouldn't have to touch too much :)
# ------------------------------------------------
# Except, perhaps the License and Trove Classifiers!
# If you do change the License, remember to change the Trove Classifier for that!here = os.path.abspath(os.path.dirname(__file__))# Import the README and use it as the long-description.
# Note: this will only work if 'README.md' is present in your MANIFEST.in file!
try:with io.open(os.path.join(here, 'README.md'), encoding='utf-8') as f:long_description = '\n' + f.read()
except FileNotFoundError:long_description = DESCRIPTION# Load the package's __version__.py module as a dictionary.
about = {}
if not VERSION:project_slug = NAME.lower().replace("-", "_").replace(" ", "_")with open(os.path.join(here, project_slug, '__version__.py')) as f:exec(f.read(), about)
else:about['__version__'] = VERSIONclass UploadCommand(Command):"""Support setup.py upload."""description = 'Build and publish the package.'user_options = []@staticmethoddef status(s):"""Prints things in bold."""print('\033[1m{0}\033[0m'.format(s))def initialize_options(self):passdef finalize_options(self):passdef run(self):try:self.status('Removing previous builds…')rmtree(os.path.join(here, 'dist'))except OSError:passself.status('Building Source and Wheel (universal) distribution…')os.system('{0} setup.py sdist bdist_wheel --universal'.format(sys.executable))self.status('Uploading the package to PyPI via Twine…')os.system('twine upload dist/*')self.status('Pushing git tags…')os.system('git tag v{0}'.format(about['__version__']))os.system('git push --tags')sys.exit()# Where the magic happens:
setup(name=NAME,version=about['__version__'],description=DESCRIPTION,long_description=long_description,long_description_content_type='text/markdown',author=AUTHOR,author_email=EMAIL,python_requires=REQUIRES_PYTHON,url=URL,# packages=find_packages(exclude=["tests", "*.tests", "*.tests.*", "tests.*"]),# If your package is a single module, use this instead of 'packages':py_modules=['hellopypi'],entry_points={'console_scripts': ['hellopypi=hello:main'], },install_requires=REQUIRED,extras_require=EXTRAS,include_package_data=True,license='Apache-2.0',classifiers=[# Trove classifiers# Full list: https://pypi.python.org/pypi?%3Aaction=list_classifiers'License :: OSI Approved :: Apache-2.0 license','Programming Language :: Python','Programming Language :: Python :: 3','Programming Language :: Python :: 3.6','Programming Language :: Python :: Implementation :: CPython','Programming Language :: Python :: Implementation :: PyPy'],# $ setup.py publish support.cmdclass={'upload': UploadCommand,},
)

配置信息说明:

  • 项目的配置信息:
# Package meta-data.
NAME = 'mypackage'
DESCRIPTION = '填写你的项目简短描述.'
URL = 'https://github.com/你的github账户/mypackage'
EMAIL = 'me@example.com'    # 你的邮箱
AUTHOR = 'Awesome Soul'     # 你的名字
REQUIRES_PYTHON = '>=3.6.0' # 项目支持的python版本
VERSION = '0.1.0'           # 项目版本号
  • 项目的依赖库(没有就不填):
# What packages are required for this module to be executed?
REQUIRED = [# 'requests', 'maya', 'records',
]
  • setup部分:

这里大部分内容都不用填,只有以下几个注意点

  • long_description这里默认是项目的README.md文件
  • 注释掉的entry_points部分是用来生成命令行工具或者GUI工具的(理论上是跨平台的),这里我生成了一个hellopypi的命令来代替hello.pymain函数,安装成功以后就可以直接使用hellopypi命令:
    entry_points={ 'console_scripts': ['hellopypi=hello:main'], },
  • 如果你的项目文件夹下只有一个py文件来实现你的功能的话,需要将packages=find_packages(exclude=["tests", "*.tests", "*.tests.*", "tests.*"]),注释掉,然后取消py_modules的注释并进行相应修改。
setup(name=NAME,version=about['__version__'],description=DESCRIPTION,long_description=long_description,long_description_content_type='text/markdown',author=AUTHOR,author_email=EMAIL,python_requires=REQUIRES_PYTHON,url=URL,# packages=find_packages(exclude=["tests", "*.tests", "*.tests.*", "tests.*"]),# If your package is a single module, use this instead of 'packages':py_modules=['hellopypi'],entry_points={'console_scripts': ['hellopypi=hello:main'], },install_requires=REQUIRED,extras_require=EXTRAS,include_package_data=True,license='Apache-2.0',classifiers=[# Trove classifiers# Full list: https://pypi.python.org/pypi?%3Aaction=list_classifiers'License :: OSI Approved :: Apache-2.0 license','Programming Language :: Python','Programming Language :: Python :: 3','Programming Language :: Python :: 3.6','Programming Language :: Python :: Implementation :: CPython','Programming Language :: Python :: Implementation :: PyPy'],# $ setup.py publish support.cmdclass={'upload': UploadCommand,},
)
1.5 编写说明文档

一个好的项目,需要有一个条理清晰的文档的,在 README.md 对项目进行详尽的说明。

2 发布到 Pypi

2.1 生成分发档案

为包生成分发包。这些是上传到包索引的档案,可以通过pip安装。

  • 确保有setuptools, wheel 安装了最新版本:
python3 -m pip install --user --upgrade setuptools wheel
  • 检查setup.py是否有错误:
    运行python setup.py check,如果没报错误,则输出一般是running check;如果有错误,就根据报错信息来修一下。

准备好上面的步骤, 一个包就基本完整了, 剩下的就是打包了。

2.1.1 生成 tar.gz 包
python3 setup.py sdist build

在当前目录的 dist 文件夹下, 就会多出一个tar.gz结尾的包了。

2.1.2 也可以打包一个 wheel 格式的包
python3 setup.py bdist_wheel --universal

在 dist 文件夹下面生成一个whl文件.

也可以一次性生成tar.gz包和whl包:

python3 setup.py sdist bdist_wheel

会在dist目录下生成一个tar.gz的源码包和一个.whl的 Wheel 包。

2.2 发布包到 Pypi

先去pypi注册账号,记住账号和密码,后面上传包会使用。

注册号账号之后,接下来就是上传包。

上传的时候会用到twine,需要先安装twine(用 twine上传分发包,并且只有 twine> = 1.11.0 才能将元数据正确发送到 Pypi上)。

pip install twine

前面编写的setup.py具备上传包的功能:

python3 setup.py upload

不出意外的话,到这里,我们自己的包就发布完成了。但是这里可能会遇到这样的报错:The user 'xxx' isn't allowed to upload to project 'xxx'. See https://pypi.org/help/#project-name for more information.

这个是由于软件包名字是PyPI用以区分的唯一标识,因此必须全球唯一,此时表明可能已经存在了相同名字的包了,那么换个不重复的名字即可。我这里就遇到了这个问题,因此我把名字改成了hellopypi_oh就可以了。

2.3 验证发布 PYPI 成功

上传完成了会显示 success, 我们直接可以在 PyPI 上看到。

可以使用pip来安装包并验证它是否有效:

pip install hellopypi_oh

安装成功之后,直接在 Terminal 中执行 hellopypi_oh命令,看到输出

Hello Pypi!

表明发布成功。

3 后续维护升级

  • 有更新升级之后,首先删除旧版本打包文件,然后生成新文件:
python3 setup.py sdist bdist_wheel
  • 输入以下命令,上传新版本即可:
python setup.py upload

这个命令还会自动把代码改动更新到 github 仓库。

http://www.lryc.cn/news/457000.html

相关文章:

  • 农行1面:说说 final,finally,finalize的区别
  • ChatGPT实时语音将于本周向免费用户推出:OpenAI DevDay 2024详细解读
  • 一个月学会Java 第7天 字符串与键盘输入
  • Java后端面试题(day16)
  • Auto-Animate:是一款零配置、即插即用的动画工具,可以为您的 Web 应用添加流畅的过渡效果
  • k8s之ingress-nginx-controller安装
  • 力扣SQL仅数据库(1098~1132)
  • 优惠点餐api接口对接的具体步骤是什么?
  • 【韩顺平Java笔记】第8章:面向对象编程(中级部分)【297-313】
  • 快递批量查询物流追踪只揽收无物流信息的单号
  • 【动态网站资源保存下载】
  • Selenium自动化测试中如何处理数据驱动?
  • 淘宝API接口系列有哪些内容?
  • 华为OD机试 - 冠亚军排名(Java 2024 E卷 100分)
  • VmWare中安装CenterOs(内网服务器)
  • JS 数组去重 — 各类场景适合方法大全
  • 【Java 问题】集合——List
  • xss 跨站脚本攻击
  • 5.toString()、构造方法、垃圾回收、静态变量与静态方法、单例设计模式、内部类
  • Fiddler配合wireshark解密ssl
  • 【UI】将 naive ui 的 message 封装进axios 中,关于naiveui的message相关的用法
  • IC卡批量加密快速写入
  • 软件测试学习笔记丨tcpdump 与 wireshark
  • Redis:分布式 - 哨兵
  • 开源城市运动预约的工具类小程序源码
  • SldWorks问题 2. 矩阵相关接口使用上的失误
  • 2024十月超全大模型常见面试题(附答案)
  • Java 的数据结构整理(整合版)
  • 如何让信息学奥赛学习“边玩边学”?——趣味编程让枯燥学习变得有趣
  • 【艾思科蓝】C++游戏开发探秘:打造高性能游戏世界的钥匙