RPM包是如何從源RPM制作的
在上一篇文章中,我們研究了什么是 RPM 軟件包。它們是包含文件和元數(shù)據(jù)的檔案文件。當(dāng)安裝或卸載 RPM 時(shí),此元數(shù)據(jù)告訴 RPM 在哪里創(chuàng)建或刪除文件。正如你將在上一篇文章中記住的,元數(shù)據(jù)還包含有關(guān)“依賴項(xiàng)”的信息,它可以是“運(yùn)行時(shí)”或“構(gòu)建時(shí)”的依賴信息。
例如,讓我們來看看 fpaste
。你可以使用 dnf
下載該 RPM。這將下載 Fedora 存儲(chǔ)庫中可用的 fpaste
最新版本。在 Fedora 30 上,當(dāng)前版本為 0.3.9.2:
$ dnf download fpaste
...
fpaste-0.3.9.2-2.fc30.noarch.rpm
由于這是個(gè)構(gòu)建 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
在此鏈條中的下一個(gè)環(huán)節(jié)是源 RPM。Fedora 中的所有軟件都必須從其源代碼構(gòu)建。我們不包括預(yù)構(gòu)建的二進(jìn)制文件。因此,要制作一個(gè) RPM 文件,RPM(工具)需要:
- 給出必須要安裝的文件,
- 例如,如果要編譯出這些文件,則告訴它們?nèi)绾紊蛇@些文件,
- 告知必須在何處安裝這些文件,
- 該特定軟件需要其他哪些依賴才能正常工作。
源 RPM 擁有所有這些信息。源 RPM 與構(gòu)建 RPM 相似,但顧名思義,它們不包含已構(gòu)建的二進(jìn)制文件,而是包含某個(gè)軟件的源文件。讓我們下載 fpaste
的源 RPM:
$ dnf download fpaste --source
...
fpaste-0.3.9.2-2.fc30.src.rpm
注意文件的結(jié)尾是 src.rpm
。所有的 RPM 都是從源 RPM 構(gòu)建的。你也可以使用 dnf
輕松檢查“二進(jìn)制” RPM 的源 RPM:
$ dnf repoquery --qf "%{SOURCERPM}" fpaste
fpaste-0.3.9.2-2.fc30.src.rpm
另外,由于這是源 RPM,因此它不包含構(gòu)建的文件。相反,它包含有關(guān)如何從中構(gòu)建 RPM 的源代碼和指令:
$ rpm -qpl ./fpaste-0.3.9.2-2.fc30.src.rpm
fpaste-0.3.9.2.tar.gz
fpaste.spec
這里,第一個(gè)文件只是 fpaste
的源代碼。第二個(gè)是 spec 文件。spec 文件是個(gè)配方,可告訴 RPM(工具)如何使用源 RPM 中包含的源代碼創(chuàng)建 RPM(檔案文件)— 它包含 RPM(工具)構(gòu)建 RPM(檔案文件)所需的所有信息。在 spec 文件中。當(dāng)我們軟件包維護(hù)人員添加軟件到 Fedora 中時(shí),我們大部分時(shí)間都花在編寫和完善 spec 文件上。當(dāng)軟件包需要更新時(shí),我們會(huì)回過頭來調(diào)整 spec 文件。你可以在 https://src.fedoraproject.org/browse/projects/ 的源代碼存儲(chǔ)庫中查看 Fedora 中所有軟件包的 spec 文件。
請(qǐng)注意,一個(gè)源 RPM 可能包含構(gòu)建多個(gè) RPM 的說明。fpaste
是一款非常簡單的軟件,一個(gè)源 RPM 生成一個(gè)“二進(jìn)制” RPM。而 Python 則更復(fù)雜。雖然只有一個(gè)源 RPM,但它會(huì)生成多個(gè)二進(jìn)制 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” 是“主包”,因此該 spec 文件將稱為 python3.spec
。所有其他軟件包均為“子軟件包”。你可以下載 python3 的源 RPM,并查看其中的內(nèi)容。(提示:補(bǔ)丁也是源代碼的一部分):
$ 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 構(gòu)建 RPM
現(xiàn)在我們有了源 RPM,并且其中有什么內(nèi)容,我們可以從中重建 RPM。但是,在執(zhí)行此操作之前,我們應(yīng)該設(shè)置系統(tǒng)以構(gòu)建 RPM。首先,我們安裝必需的工具:
$ sudo dnf install fedora-packager
這將安裝 rpmbuild
工具。rpmbuild
需要一個(gè)默認(rèn)布局,以便它知道源 RPM 中每個(gè)必需組件的位置。讓我們看看它們是什么:
# spec 文件將出現(xiàn)在哪里?
$ rpm -E %{_specdir}
/home/asinha/rpmbuild/SPECS
# 源代碼將出現(xiàn)在哪里?
$ rpm -E %{_sourcedir}
/home/asinha/rpmbuild/SOURCES
# 臨時(shí)構(gòu)建目錄是哪里?
$ rpm -E %{_builddir}
/home/asinha/rpmbuild/BUILD
# 構(gòu)建根目錄是哪里?
$ rpm -E %{_buildrootdir}
/home/asinha/rpmbuild/BUILDROOT
# 源 RPM 將放在哪里?
$ rpm -E %{_srcrpmdir}
/home/asinha/rpmbuild/SRPMS
# 構(gòu)建的 RPM 將放在哪里?
$ rpm -E %{_rpmdir}
/home/asinha/rpmbuild/RPMS
我已經(jīng)在系統(tǒng)上設(shè)置了所有這些目錄:
$ cd
$ tree -L 1 rpmbuild/
rpmbuild/
├── BUILD
├── BUILDROOT
├── RPMS
├── SOURCES
├── SPECS
└── SRPMS
6 directories, 0 files
RPM 還提供了一個(gè)為你全部設(shè)置好的工具:
$ rpmdev-setuptree
然后,確保已安裝 fpaste
的所有構(gòu)建依賴項(xiàng):
sudo dnf builddep fpaste-0.3.9.2-3.fc30.src.rpm
對(duì)于 fpaste
,你只需要 Python,并且它肯定已經(jīng)安裝在你的系統(tǒng)上(dnf
也使用 Python)。還可以給 builddep
命令一個(gè) spec 文件,而不是源 RPM。在手冊(cè)頁中了解更多信息:
$ man dnf.plugin.builddep
現(xiàn)在我們有了所需的一切,從源 RPM 構(gòu)建一個(gè) 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 并從中構(gòu)建你的 RPM?,F(xiàn)在,你可以使用 dnf
安裝 RPM 以使用它。當(dāng)然,如前所述,如果你想在 RPM 中進(jìn)行任何更改,則必須修改 spec 文件,我們將在下一篇文章中介紹 spec 文件。
總結(jié)
總結(jié)一下這篇文章有兩點(diǎn):
- 我們通常安裝使用的 RPM 是包含軟件的構(gòu)建版本的 “二進(jìn)制” RPM
- 構(gòu)建 RPM 來自于源 RPM,源 RPM 包括用于生成二進(jìn)制 RPM 所需的源代碼和規(guī)范文件。
如果你想開始構(gòu)建 RPM,并幫助 Fedora 社區(qū)維護(hù)我們提供的大量軟件,則可以從這里開始: https://fedoraproject.org/wiki/Join_the_package_collection_maintainers
如有任何疑問,請(qǐng)發(fā)郵件到 Fedora 開發(fā)人員郵件列表,我們隨時(shí)樂意為你提供幫助!