基于Python3编写的Golang程序多平台交叉编译自动化脚本

import argparse
import os
import shutil
import sys
from shutil import copy2from loguru import loggerclass GoBuild:"""一个用于构建跨平台执行文件的类。初始化函数,设置构建的主文件、生成的执行文件名称以及目标平台。:param f: 需要构建的主文件(例如: main.go):param n: 生成的执行文件主名称(例如: install)"""def __init__(self, f, n=None):self.darwin = darwinself.go = "go"self.name = nself.arch_list = []self.os_list = []self.amd64 = Falseself.mips64 = Falseself.arm64 = Falseself.arm32 = Falseself.linux = Falseself.windows = Falseself.file = fself.basename = ""self.archs = "X86_64"self.os_type = ""self.exe = ""self.tmp = ""self.logger = loggerdef init(self):"""初始化函数,根据不同的架构和操作系统生成相应的架构和操作系统列表,并设置可执行文件的基础名称。"""# 检查并添加架构到架构列表if self.arm64:self.arch_list.append("arm64")if self.mips64:self.arch_list.append("mips64le")if self.amd64:self.arch_list.append("amd64")if self.arm32:self.arch_list.append("arm")# 检查并添加操作系统到操作系统列表if self.linux:self.os_list.append("linux")if self.windows:self.os_list.append("windows")if self.darwin:self.os_list.append("darwin")# 设置可执行文件的基础名称if self.name is None:self.basename = str(os.path.basename(self.file)).replace(".go", "")else:self.basename = self.namedef delete(self):"""开始删除生成的临时文件:return: None"""# 构造临时文件的完整路径tmp = os.path.join(os.getcwd(), self.tmp)try:# 尝试删除临时文件os.remove(path=self.tmp)# 删除成功后记录日志self.logger.debug(f"删除成功: {tmp}")except Exception as e:# 删除失败后记录错误日志self.logger.error(f"删除出错 - [{tmp} ] : {str(e)}")def copy(self):"""复制执行文件到目标目录,并根据当前环境调整文件路径。此方法首先构建目标文件路径,然后尝试从临时文件路径复制文件到目标路径,如果临时文件存在的话。如果复制成功,它将调用delete方法删除临时文件。如果临时文件不存在,它将记录一个警告消息。"""# 构建目标文件路径dst = os.path.join("client", self.exe)# 将目标路径与当前工作目录结合dst = os.path.join(os.getcwd(), dst)# 替换路径中的'amd64'为'X86_64'以适配不同架构dst = str(dst).replace("amd64", "X86_64")# 记录复制操作的开始self.logger.debug(f"开始复制: {dst}")# 检查临时文件是否存在if os.path.isfile(self.tmp):try:# 尝试复制文件到目标路径copy2(src=self.tmp, dst=dst)# 复制成功后,删除临时文件self.delete()except Exception as e:# 如果复制过程中发生异常,记录错误信息self.logger.error(f"复制失败: {str(e)}")else:# 如果临时文件不存在,记录警告信息self.logger.warning(f"文件不存在: {self.tmp}")def build(self):"""构建指定的Go文件,根据操作系统类型和架构进行编译,并处理构建结果。1. 根据架构类型转换变量`self.archs`,确保其符合预期的架构命名规范。2. 记录构建系统的操作系统类型和架构,用于调试和追踪。3. 根据操作系统和架构生成可执行文件名`self.exe`,并调整Windows系统下的文件扩展名。4. 构建Go文件,如果构建成功则调用`self.copy()`方法处理构建结果,否则记录错误并退出程序。"""# 根据架构类型转换变量self.archs,确保其符合预期的架构命名规范if self.archs == "amd64":self.archs = "X86_64"# 记录构建系统的操作系统类型和架构,用于调试和追踪self.logger.debug(f"构建系统: {self.os_type}", )self.logger.debug(f"构建架构: {self.archs}")# 根据操作系统和架构生成可执行文件名self.exe,并调整Windows系统下的文件扩展名self.exe = self.basename + "_" + self.os_type + "-" + self.archsself.tmp = str(os.path.basename(self.file)).replace(".go", "")if self.os_type == "windows":self.exe = self.exe + ".exe"self.tmp = self.tmp + ".exe"# 构建Go文件c = f"{self.go} build {self.file}"if os.system(c) == 0:# 如果构建成功则记录信息并调用self.copy()方法处理构建结果self.logger.info(f"构建成功,正在生成: {self.exe}")self.copy()else:# 如果构建失败则记录错误并退出程序self.logger.error(f"构建失败: {self.exe}")print(c)sys.exit(2)def ost(self, o):"""设置操作系统类型该方法主要用于设置GOOS环境变量,以模拟不同的操作系统环境这对于后续的编译过程特别重要,因为GOOS环境变量决定了编译输出的目标操作系统。参数:o (str): 要模拟的操作系统类型,例如"linux"、"windows"等。返回:无"""# 设置GOOS环境变量以模拟指定的操作系统os.environ['GOOS'] = o# 更新实例的os_type属性以存储当前设置的操作系统类型self.os_type = odef arch(self, arch):"""设置架构并触发构建过程该方法接收一个架构名称,根据特定规则转换后设置环境变量GOARCH,并将该架构名称保存以供后续使用。最后,调用build方法进行构建。参数:arch (str): 构架名称,可能需要转换以适配特定的命名约定。返回:无"""# 根据输入的架构名称进行条件判断,以确定是否需要转换架构名称if arch == "X86_64":arch = "amd64"# 设置环境变量GOARCH为转换后的架构名称,以便在后续的构建过程中使用os.environ['GOARCH'] = arch# 保存当前架构名称到实例变量,以便在类的其他方法中访问self.archs = arch# 调用实例的build方法,触发针对指定架构的构建过程self.build()def start(self, save):"""启动初始化和操作系统处理流程在这个方法中,首先进行初始化操作,然后根据`save`参数检查目录是否存在,如果不存在则创建目录.随后,遍历操作系统列表,对每个操作系统进行处理.对于Linux操作系统,进一步遍历其架构列表并进行处理;对于其他操作系统,则默认使用X86_64架构进行处理.参数:save (str): 保存路径字符串,用于检查是否存在以及创建目录"""# 初始化操作self.init()# 检查save是否为目录,如果不是则创建client目录if not os.path.isdir(save):os.mkdir("./client")# 遍历操作系统列表,对每个操作系统调用ost方法for i in self.os_list:self.ost(i)# 对Linux操作系统,记录架构列表并遍历每个架构if i == "linux":self.logger.debug(f"架构列表: {self.arch_list}")for a in self.arch_list:self.arch(arch=a)# 对其他操作系统,默认使用X86_64架构else:self.arch(arch="X86_64")# 主程序入口
if __name__ == "__main__":# 获取当前目录cwd = os.getcwd()# 解析命令传参parser = argparse.ArgumentParser()parser.add_argument("-f", "--file", help="源代码文件名", type=str, default="ssl.go")parser.add_argument("-n", "--name", help="项目名称", type=str, default="ssl")# 是否启用Linux平台, 默认启用parser.add_argument("-l", "--linux", help="是否启用Linux平台", action='store_true', default=True)parser.add_argument("--no-linux", help="禁用Linux平台", action='store_false', dest='linux')# 是否启用Darwin平台, 默认启用parser.add_argument("-d", "--darwin", help="是否启用Darwin平台", action='store_true', default=True)parser.add_argument("--no-darwin", help="禁用Darwin平台", action='store_false', dest='darwin')# 是否启用Windows平台, 默认启用parser.add_argument("-w", "--windows", help="是否启用Windows平台", action='store_true', default=True)parser.add_argument("--no-windows", help="禁用Windows平台", action='store_false', dest='windows')# 是否启用arm64平台, 默认启用parser.add_argument("-a", "--arm64", help="是否启用arm64平台", action='store_true', default=True)parser.add_argument("--no-arm64", help="禁用arm64平台", action='store_false', dest='arm64')# 是否启用arm32平台, 默认启用parser.add_argument("-r32", "--arm32", help="是否启用arm32平台", action='store_true', default=True)parser.add_argument("--no-arm32", help="禁用arm32平台", action='store_false', dest='arm32')# 是否启用mips64平台, 默认启用parser.add_argument("-m", "--mips64", help="是否启用mips64平台", action='store_true', default=True)parser.add_argument("--no-mips64", help="禁用mips64平台", action='store_false', dest='mips64')# 是否启用amd64/x86平台, 默认启用parser.add_argument("-x", "--x86", help="是否启用amd64/x86平台", action='store_true', default=True)parser.add_argument("--no-x86", help="禁用amd64/x86平台", action='store_false', dest='x86')# 设置保存目录,默认: ./clientclient_ = os.path.join(cwd, 'client')parser.add_argument("-o", "--output", help=f"保存目录->默认: {client_}", type=str, default=client_)# 显示版本号parser.add_argument("-v", "--version", action='version', version='%(prog)s 1.0')# 解析命令行参数args = parser.parse_args()# 提取命令行参数中的文件名file = args.file# 提取命令行参数中的项目名称name = args.name# 提取命令行参数中的MIPS64平台编译选项mips64 = args.mips64# 提取命令行参数中的Linux平台编译选项linux = args.linux# 提取命令行参数中的Darwin平台编译选项darwin = args.darwin# 提取命令行参数中的ARM64平台编译选项arm64 = args.arm64# 提取命令行参数中的X86平台编译选项x86 = args.x86# 提取命令行参数中的ARM32平台编译选项arm32 = args.arm32# 提取命令行参数中的Windows平台编译选项windows = args.windows# 打印所有命令行参数print(args)# 打印提取的文件名print(f"文件名: {file}")# 打印提取的项目名称# 打印启用的平台print(f"Linux: {args.linux}")print(f"Darwin: {args.darwin}")print(f"Windows: {args.windows}")print(f"ARM64: {args.arm64}")print(f"ARM32: {args.arm32}")print(f"MIPS64: {args.mips64}")print(f"X86: {args.x86}")# 创建GoBuild实例,传入文件名、项目名称及各平台编译选项up = GoBuild(f=file, n=name)# 设置ARM32架构的编译选项up.arm32 = arm32# 设置Windows系统的编译选项up.windows = windows# 设置ARM64架构的编译选项up.arm64 = arm64# 设置MIPS64架构的编译选项up.mips64 = mips64# 设置X86架构的编译选项up.x86 = x86# 设置Linux系统的编译选项up.linux = linux# 设置Darwin系统的编译选项up.darwin = darwin# 启动编译过程up.start(save=args.output)

效果

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/diannao/64613.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

初学stm32 --- 定时器中断

目录 时钟选择: 内部时钟选择​编辑 时钟计算方法: 计数器模式 向下计数模式(时钟分频因子1,ARR36) 向上计数模式(时钟分频因子1,ARR36) 中央对齐计数模式(时钟分频因…

Pytorch | 从零构建Vgg对CIFAR10进行分类

Pytorch | 从零构建Vgg对CIFAR10进行分类 CIFAR10数据集Vgg网络结构特点性能应用影响 Vgg结构代码详解结构代码代码详解特征提取层 _make_layers前向传播 forward 训练过程和测试结果代码汇总vgg.pytrain.pytest.py 前面文章我们构建了AlexNet对CIFAR10进行分类: Py…

Python从0到100(七十八):神经网络--从0开始搭建全连接网络和CNN网络

前言: 零基础学Python:Python从0到100最新最全教程。 想做这件事情很久了,这次我更新了自己所写过的所有博客,汇集成了Python从0到100,共一百节课,帮助大家一个月时间里从零基础到学习Python基础语法、Pyth…

SAP ABAP-日期格式问题 SAP内部错误,反序列化JSON字符串时发生异常 值 20241215 不是根据 ABAP 的 XML 格式的有效日期

SAP ABAP-日期格式问题 SAP内部错误,反序列化JSON字符串时发生异常 值 20241215 不是根据 ABAP 的 XML 格式的有效日期 在SAP内部用 YYYYMMDD没有问题 外部传入参数

使用docker compose安装gitlab

使用docker compose安装gitlab GitLab简介设置GITLAB_HOME路径创建docker挂载目录获取可用的GitLab版本编写docker-compose.yml文件启动docker基础配置GITLAB_OMNIBUS_CONFIG修改配置 中文设置数据库配置系统邮箱配置 GitLab简介 ‌GitLab是一个基于Git的开源项目,…

获取显示器(主/副屏)友好名称(FriendlyName)

在开发涉及多显示器的应用程序时,获取显示器的友好名称(Friendly Name)是一个常见需求。本文将深入探讨GetMonitorFriendlyName 方法,了解其实现细节和工作原理。 方法签名 public static string GetMonitorFriendlyName(bool i…

sql server索引优化语句

第一步 建一个测试表 --create table TestUsers --( -- Id int primary key identity(1,1), -- Username varchar(30) not null, -- Password varchar(10) not null, -- CreateDateTime datetime not null --)第二步 插入100w数据 大概1分钟执行时间 ----插入数据…

day-21 内核链表以及栈

1.昨日作业 1.删除指定节点 找到删除就完事了,双向可以停在删除处。 /***************************** 功能:删除指定结点(通过姓名)* 参数:phead;oldname; * 返回:成功0,失-1&…

我在广州学 Mysql 系列——有关 Mysql 函数的练习

ℹ️大家好,我是LXJ,通过Mysql各种函数的相关学习,本文将通过一些练习来巩固Mysql的函数~~ 明天就冬至啦,提前大家冬至快乐!!“差不多冬至,一早一晚还是有雨”~ 复习:👉《…

第3节 测试套件数据驱动

创建Excel、 CSV测试数据 1. 从主菜单中选择 File > New > Test Data。将显示新的测试数据对话框。输入测试数据的名称并选择数据类型作为Excel File/ CSV File 。单击OK。 2. 浏览到要导入Katalon Studio的Excel File, 选择Excel中的sheetName,或者CSV文件…

tomato靶场攻略

前提:kali和tomato的连接方式都为net模式 tomato的默认网络连接方式为桥接模式,导入前注意修改,将tomato.ova的镜像导入虚拟机中 出现此页面则表示导入成功,打开kali虚拟机终端,切换为root权限 arp-scan -l 浏览器访…

【数据安全】如何保证其安全

数据安全风险 数字经济时代,数据已成为重要的生产要素。智慧城市、智慧政务的建设,正以数据为核心,推动城市管理的智能化和公共服务的优化。然而,公共数据开放共享与隐私保护之间的矛盾日益凸显,如何在确保数据安全的…

go聊天系统项目6-服务端发送消息

一、前言 敬告:本文不讲解代码,只是把代码展示出来。 该代码之前的代码见 go 聊天系统项目-1 go聊天系统项目-2 redis 验证用户id和密码 go聊天系统项目-3 redis注册用户 go聊天项目4-显示用户列表 go 聊天系统项目-5 客户端发消息 注意:本文使用 go mo…

【深度学习-论文】通俗易懂的理解多标签识别

文章目录 1. 文章主要内容2. 通俗易懂的理解多标签分类到底是如何实现的通俗易懂的多标签分类实现介绍实现步骤为什么这么做?小结论文题目:Rada r emitter multi-la bel recognition based on residual network 基于残差网络的雷达发射机多标签识别1. 文章主要内容 《基于残…

day5,数据结构,单向,双向,循环链表

1】思维导图 2】完成单向循环链表的所有操作 【创建、判空、尾插、遍历、尾删、销毁】 创建: LooplinkPtr caerte() {LooplinkPtr h(LooplinkPtr)malloc(sizeof(Looplink));if(NULLh){printf("创建失败\n");return NULL;}h->len0;h->data0;h->…

CH340系列芯片驱动电路·CH340系列芯片驱动!!!

目录 CH340基础知识 CH340常见类型 CH340引脚功能讲解 CH340驱动电路 CH340系列芯片数据手册 编写不易,仅供学习,请勿搬运,感谢理解 常见元器件驱动电路文章专栏连接 LM7805系列降压芯片驱动电路降压芯片驱动电路详解-CSDN博客 ME62…

【机器学习案列】使用随机森林(RF)进行白葡萄酒质量预测

🧑 博主简介:曾任某智慧城市类企业算法总监,目前在美国市场的物流公司从事高级算法工程师一职,深耕人工智能领域,精通python数据挖掘、可视化、机器学习等,发表过AI相关的专利并多次在AI类比赛中获奖。CSDN…

PostgreSql-学习06-libpq之同步命令处理

目录 一、环境 二、介绍 三、函数 1、PQsetdbLogin (1)作用 (2)声明 (3)参数介绍 (4)检测成功与否 2、PQfinish (1)作用 (2&#xff0…

Reactor

文章目录 正确的理解发送double free问题 1.把我们的reactor进行拆分2.链接管理3.Reactor的理论 listensock只需要设置_recv_cb,而其他sock,读,写,异常 所以今天写nullptr其实就不太对,添加为空就没办法去响应事件 获…

Mac iOS、Android、Flutter、React Native开发环境配置

1.安装XCode https://apps.apple.com/cn/app/xcode/id497799835?mt12 2.安装Android Studio https://developer.android.google.cn/studio 3.安装brew 参考地址:https://www.jianshu.com/p/22122a1d4474 /bin/zsh -c "$(curl -fsSL https://gitee.com/cunk…