我的编程空间,编程开发者的网络收藏夹
学习永远不晚

从交叉编译的二进制文件 Go 应用 tarball 构建 RPM 包

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

从交叉编译的二进制文件 Go 应用 tarball 构建 RPM 包

对于一个Golang开发者来说,牢固扎实的基础是十分重要的,编程网就来带大家一点点的掌握基础知识点。今天本篇文章带大家了解《从交叉编译的二进制文件 Go 应用 tarball 构建 RPM 包》,主要介绍了,希望对大家的知识积累有所帮助,快点收藏起来吧,否则需要时就找不到了!

问题内容

我正在尝试创建我的第一个与 go 交叉编译的 rpm 包。

这是包含所有必需信息的 makefile:

appname?=helloworld
version?=v1.0.0

appandver := ${appname}-$(version)

# build flags
ldflags := -ldflags "-s -w -x=main.version=$(version)"

# temporary directory for common files when creating tarballs
releasetmpdir := $(shell mktemp -d -t ${appname}-${version}-release-xxxxxx)

# cross-compile to these cpus
# https://golang.org/doc/install/source#environment
linux_archs := amd64 arm arm64 ppc64 ppc64le

default: release-bin

main.go:
    @echo 'package main' > $@
    @echo 'import "fmt"' >> $@ 
    @echo '// replaced with version when building' >> $@ 
    @echo 'var version = "v0.0.0"' >> $@
    @echo 'func main() {' >> $@
    @echo '  fmt.println("hello, world!")' >> $@
    @echo '}' >> $@
    @go fmt $@

readme.md:
    @echo 'this is a readme for $(appname) $(version)' > $@
    @echo '$(appname) simply prints "hello, world!"' >> $@

license:
    @echo 'you can do what ever you want with $(appname).' > $@
    @echo 'you have all the responsibility!' >> $@

# build for all listed architectures
linux-build: main.go 
    @for arch in $(linux_archs); do \
      echo "gnu/linux build... $$arch"; \
      cgo_enabled=0 goos=linux goarch=$$arch go build $(ldflags) -v -o ./bin/linux-$$arch/${appname} . ; \
    done

# copy common files used in binary tarball releases
copycommon: readme.md license
    @echo "copying common files to temporary release directory '$(releasetmpdir)'.."
    @mkdir "$(releasetmpdir)/bin"
    @cp -v "license" "$(releasetmpdir)"
    @cp -v "readme.md" "$(releasetmpdir)"
    @mkdir --parents "$(pwd)/release/${version}"

# create binary release tarballs for each cpu architecture
compress-linux: linux-build
    @for arch in $(linux_archs); do \
      echo "gnu/linux tar... $$arch"; \
      cp -v "$(pwd)/bin/linux-$$arch/${appname}" "$(releasetmpdir)/bin"; \
      cd "$(releasetmpdir)"; \
      tar --numeric-owner --owner=0 --group=0 -zcvf "$(pwd)/release/${version}/$(appandver)-linux-$$arch.tar.gz" . ; \
      rm "$(releasetmpdir)/bin/${appname}"; \
    done

# move all to temporary directory and compress with common files
compress-everything: copycommon compress-linux
    @echo "$@ ..."
    rm -rf "$(releasetmpdir)/*"

# create tarballs which has common files and different bin/${appname} per cpu architecture
release-bin: linux-build compress-everything
    @echo "release done..."

# linux distributions
release-ldistros: ldistro-rpm
    @echo "linux distros release done..."

release/linux/rpm:
    @mkdir --parents ./release/linux/rpm

# rpm spec file (probably wrong)
release/linux/rpm/package.spec: release/linux/rpm
    @echo 'name: ${appname}' > $@
    @echo 'version: %{_version}' >> $@
    @echo 'release: 1%{?dist}' >> $@
    @echo 'summary: hello world' >> $@
    @echo 'url: https://example.org/${appandver}/' >> $@
    @echo 'group: applications/utilities' >> $@
    @echo 'license: apache-2.0' >> $@
    @echo '%description' >> $@
    @echo '${appname} is a command line program which prints "hello, world!"' >> $@
    @echo '%setup -q' >> $@
    @echo '%clean' >> $@
    @echo '%files' >> $@
    @echo '%license /usr/share/licenses/%{name}/license' >> $@
    @echo '%doc /usr/share/doc/%{name}/readme.md' >> $@
    @echo '/usr/bin/${appname}' >> $@
    @echo '%install' >> $@
    @echo 'install -dm755 "usr/bin/%{name}" -t "/usr/bin"' >> $@

# create rpm package for each cpu architecture from tarballs (probably wrong)
ldistro-rpm: "release/linux/rpm/package.spec
    @for arch in $(linux_archs); do \
      echo "generating rpm... $$arch"; \
      tempdir=$$(mktemp -d -t $(appandver)-rpm-xxxxxx) ; \
      echo "  >> using temporary directory $$tempdir" ; \
      cd "$$tempdir" ; \
      mkdir --parents {sources,rpms,specs,srpms,build,tmp} ; \
      cp "$(pwd)/release/linux/rpm/package.spec" "specs/${appname}" ; \
      cd "build"; \
      echo "  >> extracting source binary package.." ; \
      tar -xzf "$(pwd)/release/${version}/$(appandver)-linux-$$arch.tar.gz" . ; \
      echo "  >> generating directory structure in temp dir.." ; \
      mkdir --parents ./usr/bin/ ; \
      mv ./bin/${appname} ./usr/bin/ ; \
      rm -rf ./bin ; \
      mkdir --parents ./usr/share/licenses/${appname}/ ; \
      mv license ./usr/share/licenses/${appname} ; \
      mkdir --parents ./usr/share/doc/${appname}/ ; \
      mv readme.md ./usr/share/doc/${appname} ; \
      cd .. ; \
      echo "  >> building rpm package.." ; \
      sudo rpmbuild -vv --nosignature --nodebuginfo --dbpath "$$tempdir" --root "$$tempdir" --buildroot "./build" --target $$arch --define "_tmppath /tmp" --define "_topdir ." --define "_version ${version}" --define "_buildhost localhost" --define "_rpmfilename $(appandver)-$$arch.rpm" -bb "specs/${appname}" && \
      rpm -qlp --info "./rpms/$(appandver)-$$arch.rpm" && \
      cp "./rpms/$(appandver)-$$arch.rpm" "$(pwd)/release/${version}/" ; \
      echo "------------------------------------------------------------"; \
    done

创建 bin/$os-$cpuarch/$appname 二进制文件:

% make linux-build

创建二进制源 tarball 到 release/$version/$appname-$version-$os-$cpuarch.tar.gz

% make compress-everything

在 tarball 内创建文件树结构:

bin/$appname (different for each architecture)
license
readme.md

此结构被锁定,因为也有不同的操作系统 tarball。出于可读性原因,在此最小化示例中删除了不同的操作系统构建目标。

为每个 cpu 架构生成 rpm:

% make ldistro-rpm

当前的问题是,在 rpm 构建期间,实际的可执行文件被安装到正在运行的系统中。据我所知,这不应该发生。规范文件或 ldistro-rpm 目标中可能缺少什么?另外,一些 rpmbuild 示例似乎只使用 -bb 参数,但我找不到示例或弄清楚如何修改 spec 文件以便可以工作。 ldistro-rpm 目标似乎过于复杂。某些命令是否应该位于规范文件的 %install%prep 等中?您能否以某种方式使用 spec 文件中的 source0 并将其指向 tarball,而不是在 ldistro-rpm 目标中生成目录结构?


解决方案


@echo 'install -Dm755 "usr/bin/%{NAME}" -t "/usr/bin"'

是您要安装到 /usr/bin 的位置。您应该安装到 $rpm_build_root/usr/bin

以上就是本文的全部内容了,是否有顺利帮助你解决问题?若是能给你带来学习上的帮助,请大家多多支持编程网!更多关于Golang的相关知识,也可关注编程网公众号。

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

从交叉编译的二进制文件 Go 应用 tarball 构建 RPM 包

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

从交叉编译的二进制文件 Go 应用 tarball 构建 RPM 包

对于一个Golang开发者来说,牢固扎实的基础是十分重要的,编程网就来带大家一点点的掌握基础知识点。今天本篇文章带大家了解《从交叉编译的二进制文件 Go 应用 tarball 构建 RPM 包》,主要介绍了,希望对大家的知识积累有所帮助,快
从交叉编译的二进制文件 Go 应用 tarball 构建 RPM 包
2024-04-05

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录