linux下安装JDK的时候,Sun公司为JDK6的linux版本提供了一个shell的安装包,用起来特别的好用,基本上和在Windows下安装软件没什么两样,shell文件执行之后,几乎一切都系都设置好了,不用我们自己再动手设置PATH和JAVA_HOME,可是一个shell文件中是如何把二进制代码包含进来的呢?
再例如淘宝为linux开发的淘宝插件,其实也是一个shell文件,但是执行这个shell文件之后,会安装很多二进制的东西,同样的问题,Shell只是文本文件,其中的二进制是怎么来包含进来的呢?
从文本文件转换出可执行文件,通过编译器把源程序编程成可执行程序当然是可以的,但是前面提到的哪两种情况都不是这样做的。原因有两个,第一对于大的项目来说,编译需要的时间比较长,环境比较复杂;第二,更加重要的是,这样做其实和从源代码编译程序没什么两样,对于不想让用户看到自己的源码的商业软件来说,这显然是不可取的。
我们看看一般的程序是如何发布他们的二进制包的。
一般的程序,无非是把二进制和必要的文档压缩到一个压缩文件中,然后通过README文档的解释程序的运行依赖什么样的假设,然后,你就可以把程序自行的移动到你想移动的地方去了。
很多时候,我们把程序放到任何地方都是可以运行的,但是程序可以运行,并不是说我们已经完成了程序的安装,举个例子来说,如果我们解压JDK的二进制包之后,直接把程序移动到一个地方,然后把对应的bin目录添加到PATH中就可以执行JDK提供的一系列工具了。
但是如果我们安装其他的依赖与JDK的程序的时候,比如TOMCAT,那么就会有问题。因为我们只是在PATH中加入了JDK的bin目录而没有制定JAVA_HOME这个环境变量,所以TOMCAT很可能会不能运行。
再比如我们使用man命令来查看一个程序的手册,一般情况下二进制包中也会包含man文档的,但是如果我们只是把解压的二进制包的路径添加到了PATH中,还不能在man中找到对应的文档。
也正是因为这样的原因吧,所以很多的二进制包的发布是使用deb或者rpm包来发布。安装的时候少了很多的烦恼。要制做deb或者rpm包当然是需要学习成本的,而且deb和rpm也只能在对应的linux发行版中使用,如果想要为所有的Linux发行版都提供一个安装文件,那么使用shell文件来做无疑是最好的办法。
Shell的学习成本低,而且对linux平台来说有通用,那么是如何做到的呢?
想想我们在手动安装的情形,无非是把压缩的二进制包解压,移动到特定的目录下,在PATH变量中添加二进制包的可执行文件的路径等等工作。首先我们把二进制包压缩文件和shell文件分开。这样一来,shell中只要完成解压,然后把解压后的目录移动到指定的目录中去,设置各种各样的环境变量然后就完成了工作了。
但是我们在如何把压缩文件和这个shell解压之后要执行的命令的shell文件一起放到一个shell文件中呢?
要做到这一点,首先这个shell文件中,要有可以解压的二进制内容,其次,这个shell要做的工作就是,把二进制内容,解压,然后把原来手动做的工作在这个shell中用命令完成。再用shell写个脚本完成一些手动完成的工作,这个任务比较容易,所以制作shell安装包的难点就是如何在其中包含二进制内容了。
如何在一个文本文件中记录二进制的内容呢?
这个问题早就被解决了。答案就是使用Base64编码。在linux下就有base64 这个命令程序就是来做这个工作的。base64可以把文件进行base64的编码,输出的标准输出中去或者把文件中的Base64编码的内容解码。命令base64除了可以对文件的内容做Base64的编解码外,也可以对标准输入中的数据进行Base64编辑码。
有了这些预备的知识,那么我们就可以看看具体的如何来做shell的二进制发布包了。
首先假设我们要发布的文件都放在名为test的当前文档中。
-
把要发布的文件打包
tar zcf test.tar ./test
-
对打好的二进制包做Base64编码
base64 ./test.tar > test_base64.txt
-
准备安装文件的shell文件
test_base64="";#test_base64中的所有内容 echo $test_base64|base64 -d >test.tar tar zxf test.tar rm test.tar # 其他的安装代码
写到这里,我们已经把道理将明白了。但是可不可以写一个shell程序,专门来生成这样的发布包软件呢?当然可以。下面是我写的这个打包程序的源代码。
function mkpackage(){
target_dir=$1
felow_install_shell_command_file=$2
echo "tar -zcf ._test_dir.tar.gz $target_dir"
tar -zcf ._test_dir.tar.gz $target_dir
base64 ._test_dir.tar.gz >._base64
rm ._test_dir.tar.gz
printf "test_base64=/"">install.sh
while IFS='' read -r line || [[ -n "$line" ]]; do
printf "$line//" >>install.sh
printf "n" >>install.sh
done <./._base64
rm ._base64
echo "/"" >>install.sh
echo 'printf $test_base64|base64 -d >._temp.tar.gz;'>>install.sh
echo 'tar zxf ._temp.tar.gz' >> install.sh
echo 'rm ._temp.tar.gz' >>install.sh
if [[ -e $fellow_install_shell_command_file ]]; then
cat $fellow_install_shell_command_file >>install.sh
fi
chmod +x install.sh
}
function usage(){
echo "usage:"
echo " $1 test_dir [the_command.sh]"
}
if [[ $# != 0 ]]
then
mkpackage $1 $2
else
usage $0
fi
这个程序接受两个输入参数,第一个表示要打的程序包目录,第二个是解压后,安装动作的shell脚本文件。最后这个程序生成一个名为 install.sh 的文件。执行 install.sh 之后,会首先得到加压的程序包内容,然后执行第二个参数指定的 shell 脚本的内容。这样我们就做了用Shell发布二进制文件的一个打包程序。当然了,在最后发布之前,还需要把这个shell压缩一下。以便消除因为Base64编码而带来的文件长度的变大的影响。
原创文章,作者:ItWorker,如若转载,请注明出处:https://blog.ytso.com/tech/linux/41480.html