1.rpm概述
内容基本源自https://rpm-packaging-guide.github.io/。
1.1.为什么需要 RPM 来打包呢
RPM 是运行在 Redhat CentOS 和 Fedora 上的包管理系统。RPM 帮助你更简单的分发,管理和更新软件。很多软件供应商通过传统的压缩包来分发软件。将软件打包到 RPM 中有以下几个优点:
RPM可以安装,重新安装,删除,升级和验证包:用户可以使用标准的包管理工具(例如 Yum 或者 PackageKit)来安装,重新安装,移除升级和验证你的 RPM 包。
使用已安装软件包的数据库来查询和验证已安装软件包:因为 RPM 维护已安装软件包及其文件的数据库,因此用户可以轻松查询和验证其系统上的软件包。
使用元数据来描述包,安装说明等:每个 RPM 软件包都包含描述软件包组件,版本,发行版,大小,项目 url,安装说明等的元数据。
将原始软件打包为源包和二进制包:RPM允许您获取原始软件源并将其打包为用户的源和二进制包。 在源包中,您拥有原始源以及所使用的任何修补程序以及完整的构建说明。 随着软件的新版本发布,此设计可以简化软件包的维护。
将包添加到 Yum 仓库:你可以将软件包添加到 yum 仓库,是客户端可以轻松查找和部署软件。
对包进行数字签名:使用 GPG 签名秘钥,你可以对包进行数字签名,以便用户能够验证包的真实性
1.2.rpm依赖工具
如果要使用rpm打包,我们需要预先安装一些rpm的依赖工具。例如gcc、rpm-build、rpm-devel、rpmlint、make、python、bash、coreutils、diffutils、patch、rpmdevtools等。在Fedora、CentOS8和RHEL8中通过“dnf install ${softwares}”进行安装,而CentOS7和RHEL7则通过“yum install ${softwares}”进行安装。
1.3.rpmbuild打包目录
rpm打包目录有一些严格的层次上的要求。
rpm的版本<=4.4.x,rpmbuid工具其默认的工作路径是 /usr/src/redhat因为权限的问题,普通用户不能制作rpm包,制作rpm软件包时必须切换到 root 身份才可以。
rpm从4.5.x版本开始,将rpmbuid的默认工作路径移动到用户家目录下的rpmbuild目录里,即 $HOME/rpmbuild ,并且推荐用户在制作rpm软件包时尽量不要以root身份进行操作。
rpmbuild默认工作路径的确定,通常由在/usr/lib/rpm/macros这个文件里的一个叫做 %_topdir 的宏变量来定义。在%_topdir目录下一般需要建立6个目录,如下表所示:
在构建rpm包之前,最好清理一下SOURCES目录。
1.4.SPEC文件说明
spec文件就是软件包描述文件。这个文件中包含了软件包的诸多信息,如:软件包的名字、版本、类别、说明摘要、创建时要执行什么指令、安装时要执行什么操作、以及软件包所要包含的文件列表等等。
spec文件内容由两部分组成:文件头和内容体。
1.4.1.spec文件头信息
涉及的条目如下表所示:
每条指令占一行内容。
1.4.2.spec文件内容
文件内容主要包含下表中的指令。
每条内容指令都表示一个阶段,决定着安装和卸载过程的具体操作。
在指令内容的编辑过程中,可能会遇到很多固有的属性,叫做macro,所有的属性都在文件 /usr/lib/rpm/macros中定义。常见的macro如下表所示:
可以通过命令rpm命令查看macros的具体值。例如
rpm --eval "some text printed on %{_arch}”
rpm --eval %{_lib}
1.5.RPM编译
rpm编译的代码分为两类:本地编译代码和解释语言代码。
1.5.1.本地编译代码
Natively compiled本地编译是把编程语言编译成机器代码,生成二进制可执行文件。这种软件可以独立运行。
以这种方式构建的 RPM 包是适用于特定体系结构的。这意味着您在使用 64 位(x86_64)AMD 或 Intel 处理器的计算机上编译此类软件,则无法在 32 位(x86)AMD 或 Intel 处理器上执行。生成的包将在其名称中指定体系结构。
1.5.2.解释语言代码
某些编程语言(如 bash 或 python)无法编译为机器代码。相反,它们程序的源代码是由语言解释器或语言虚拟机逐步执行的,无需事先转换。
完全用解释型语言编写的软件不是特定于体系结构的。因此,生成的 RPM 包名称中将包含字符串 noarch。
解释型语言是可以 byte-compiled 或者 raw-interpreted (字节编译或者原始解释)。这两种类型在程序构建过程和包装过程中有所不同。
(1)Raw-interpreted programs:原始解释语言完全不需要编译,它们由解释器直接执行。
(2)Byte-compiled programs:字节编译语言需要编译成字节代码,然后由语言虚拟机执行。
2.SPEC文件案例
下面是一个入门案例。
Name: hello-world
Version: 1
Release: 1
Summary: Most simple RPM package
License: FIXME
%description
This is my first RPM package, which does nothing.
%prep
# we have no source, so nothing here
%build
cat > hello-world.sh <<EOF
#!/usr/bin/bash
echo Hello world
EOF
%install
mkdir -p %{buildroot}/usr/bin/
install -m 755 hello-world.sh %{buildroot}/usr/bin/hello-world.sh
%files
/usr/bin/hello-world.sh
%changelog
# let's skip this for now
创建好文件后,可通过“rpmbuild -ba hello-world.spec”命令进行打包。
rpm打包成功后,通过“rpm -ivh hello-world-1-1.x86_64.rpm”命令进行软件安装。
3.RPM操作
RPM命令使用场景
3.1.rpm打包流程
打包流程有以下六个步骤:
(1)安装 rpmbuild
(2)构建rpm的编译目录结构: rpmbuild/ ├ BUILD ├ RPMS ├ SOURCES ├ SPECS ├ SRPMS
(3)将源码放到到 rpmbuild/SOURCES
(4)在rpmbuild/SPECS目录创建spec文件
(5)在rpmbuild/SPECS目录下执行打包编译: rpmbuild -bb xxxxx.spec
(6)最终生成的rpm包
3.2.rpm命令行操作
查看rpm包中有什么:rpm2cpio kafka_.rpm | cpio -t | grep connect-runtime
提取rpm包中的文件: rpm2cpio kafka.rpm | cpio -id ./usr/local/kafka/libs/connect-runtime-1.0.0.jar
3.3.查看lib.so文件属于哪个rpm包
命令为:rpm -qf {lib.so path}