如何制作 RPM 包:源 RPM

在上一篇文章中,我们研究了 RPM 包是什么。 它们是包含文件和元数据的档案。 此元数据告诉 RPM 在安装或卸载 RPM 时从何处创建或删除文件。 元数据还包含有关“依赖项”的信息,您将在上一篇文章中记住这些信息,可以是“运行时”或“构建时间”。

作为一个 example,我们来看看fpaste。 您可以使用 dnf 下载 RPM。 这将下载最新版本的 fpaste Fedora 存储库。 在 Fedora 30,目前是0.3.9.2:

$ dnf download fpaste

...
fpaste-0.3.9.2-2.fc30.noarch.rpm

由于这是构建的 RPM,它只包含使用 fpaste 所需的文件:

$ rpm -qpl ./fpaste-0.3.9.2-2.fc30.noarch.rpm
/usr/bin/fpaste
/usr/share/doc/fpaste
/usr/share/doc/fpaste/README.rst
/usr/share/doc/fpaste/TODO
/usr/share/licenses/fpaste
/usr/share/licenses/fpaste/COPYING
/usr/share/man/man1/fpaste.1.gz

源 RPM

链中的下一个环节是源 RPM。 里面的所有软件 Fedora 必须从其源代码构建。 我们不包括预构建的二进制文件。 因此,要制作 RPM 文件,RPM(工具)需要:

  • 鉴于必须安装的文件,
  • 告诉如何生成这些文件,如果它们要被编译, example,
  • 告诉这些文件必须安装在哪里,
  • 这个特定软件需要哪些其他依赖项才能正常工作。

源 RPM 包含所有这些信息。 源 RPM 与 RPM 类似,但顾名思义,它们不是保存构建的二进制文件,而是包含一个软件的源文件。 让我们下载 fpaste 的源 RPM:

$ dnf download fpaste --source
...
fpaste-0.3.9.2-2.fc30.src.rpm

注意文件如何以“src.rpm”结尾。 所有 RPM 都是从源 RPM 构建的。 您也可以使用 dnf 轻松检查“二进制”RPM 来自哪个源 RPM:

$ dnf repoquery --qf "%{SOURCERPM}" fpaste
fpaste-0.3.9.2-2.fc30.src.rpm

此外,由于这是源 RPM,它不包含构建文件。 相反,它包含有关如何从中构建 RPM 的源代码和说明:

$ rpm -qpl ./fpaste-0.3.9.2-2.fc30.src.rpm
fpaste-0.3.9.2.tar.gz
fpaste.spec

在这里,第一个文件只是 fpaste 的源代码。 第二个是“spec”文件。 规范文件是告诉 RPM(工具)如何使用源 RPM 中包含的源创建 RPM(存档)的配方——包含 RPM(工具)构建 RPM(存档)所需的所有信息在规范文件中。 当我们打包维护者添加软件到 Fedora,我们大部分时间都花在编写和完善各个规范文件上。 当软件包需要更新时,我们会返回并调整规范文件。 您可以在中查看所有包的规范文件 Fedora 在我们的源代码库中 https://src.fedoraproject.org/browse/projects/

请注意,一个源 RPM 可能包含构建多个 RPM 的说明。 fpaste 是一款非常简单的软件,其中一个源 RPM 生成一个“二进制”RPM。 另一方面,Python 更复杂。 虽然只有一个源 RPM,但它会生成多个二进制 RPM:

$ sudo dnf repoquery --qf "%{SOURCERPM}" python3
python3-3.7.3-1.fc30.src.rpm
python3-3.7.4-1.fc30.src.rpm

$ sudo dnf repoquery --qf "%{SOURCERPM}" python3-devel
python3-3.7.3-1.fc30.src.rpm
python3-3.7.4-1.fc30.src.rpm

$ sudo dnf repoquery --qf "%{SOURCERPM}" python3-libs
python3-3.7.3-1.fc30.src.rpm
python3-3.7.4-1.fc30.src.rpm

$ sudo dnf repoquery --qf "%{SOURCERPM}" python3-idle
python3-3.7.3-1.fc30.src.rpm
python3-3.7.4-1.fc30.src.rpm

$ sudo dnf repoquery --qf "%{SOURCERPM}" python3-tkinter
python3-3.7.3-1.fc30.src.rpm
python3-3.7.4-1.fc30.src.rpm

在 RPM 术语中,“python3”是“主包”,因此规范文件将被称为“python3.spec”。 所有其他包都是“子包”。 您可以下载 python3 的源 RPM 并查看其中的内容。 (提示:补丁也是源代码的一部分):

$ dnf download --source python3
python3-3.7.4-1.fc30.src.rpm

$ rpm -qpl ./python3-3.7.4-1.fc30.src.rpm
00001-rpath.patch
00102-lib64.patch
00111-no-static-lib.patch
00155-avoid-ctypes-thunks.patch
00170-gc-assertions.patch
00178-dont-duplicate-flags-in-sysconfig.patch
00189-use-rpm-wheels.patch
00205-make-libpl-respect-lib64.patch
00251-change-user-install-location.patch
00274-fix-arch-names.patch
00316-mark-bdist_wininst-unsupported.patch
Python-3.7.4.tar.xz
check-pyc-timestamps.py
idle3.appdata.xml
idle3.desktop
python3.spec

从源 RPM 构建 RPM

现在我们有了源 RPM,并且知道其中有什么,我们可以从中重建我们的 RPM。 不过,在我们这样做之前,我们应该设置我们的系统来构建 RPM。 首先,我们安装所需的工具:

$ sudo dnf install fedora-packager

这将安装 rpmbuild 工具。 rpmbuild 需要一个默认布局,以便它知道源 rpm 的每个必需组件的位置。 让我们看看它们是什么:

# Where should the spec file go?
$ rpm -E %{_specdir}
/home/asinha/rpmbuild/SPECS

# Where should the sources go?
$ rpm -E %{_sourcedir}
/home/asinha/rpmbuild/SOURCES

# Where is temporary build directory?
$ rpm -E %{_builddir}
/home/asinha/rpmbuild/BUILD

# Where is the buildroot?
$ rpm -E %{_buildrootdir}
/home/asinha/rpmbuild/BUILDROOT

# Where will the source rpms be?
$ rpm -E %{_srcrpmdir}
/home/asinha/rpmbuild/SRPMS

# Where will the built rpms be?
$ rpm -E %{_rpmdir}
/home/asinha/rpmbuild/RPMS

我已经在我的系统上设置了所有这些:

$ cd
$ tree -L 1 rpmbuild/
rpmbuild/
├── BUILD
├── BUILDROOT
├── RPMS
├── SOURCES
├── SPECS
└── SRPMS

6 directories, 0 files

RPM 提供了一个工具,也可以为您设置所有内容:

$ rpmdev-setuptree

然后我们确保我们已经安装了 fpaste 的所有构建依赖项:

sudo dnf builddep fpaste-0.3.9.2-3.fc30.src.rpm

对于 fpaste,您只需要 Python 并且必须已经安装在您的系统上(dnf 也使用 Python)。 builddep 命令也可以提供一个规范文件而不是源 RPM。 在手册页中阅读更多信息:

$ man dnf.plugin.builddep

现在我们已经拥有了所需的一切,从源 RPM 构建 RPM 非常简单:

$ rpmbuild --rebuild fpaste-0.3.9.2-3.fc30.src.rpm
..
..

$ tree ~/rpmbuild/RPMS/noarch/
/home/asinha/rpmbuild/RPMS/noarch/
└── fpaste-0.3.9.2-3.fc30.noarch.rpm

0 directories, 1 file

rpmbuild 将安装源 RPM 并从中构建您的 RPM。 您现在可以安装 RPM 以像您一样使用它——使用 dnf。 当然,如前所述,如果您想更改 RPM 中的任何内容,则必须修改规范文件——我们将在下一篇文章中介绍规范文件。

概括

将这篇文章总结为两点:

  • 我们通常安装以使用软件的 RPM 是包含软件构建版本的“二进制”RPM
  • 这些是从源 RPM 构建的,其中包括生成二进制 RPM 所需的源代码和规范文件。

如果您想开始构建 RPM,并帮助 Fedora 社区维护着我们提供的海量软件,您可以从这里开始: https://fedoraproject.org/wiki/Join_the_package_collection_maintainers

如有任何疑问,请发布到 Fedora 开发者邮件列表——我们总是乐于提供帮助!