当前位置: 首页 > news >正文

基于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)

效果

在这里插入图片描述

相关文章:

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

import argparse import os import shutil import sys from shutil import copy2from loguru import loggerclass GoBuild:"""一个用于构建跨平台执行文件的类。初始化函数,设置构建的主文件、生成的执行文件名称以及目标平台。:param f: 需要构建的…...

远程桌面连接

电脑A:使用机 电脑B:被控制的另一个 方法1: 在电脑B上操作 ①winr输入cmd进入命令行窗口,输入ipconfig查询本机地址 ②我的电脑/此电脑 右键点击“属性” ③选择屏幕右边“远程桌面” ④打开“启用远程桌面” ⑤打开设置&am…...

网络地址转换NAT

NAT(Network Address Translation) 方法于1994年提出。需要在专用网连接到因特网的路由器上安装NAT软件。装有NAT软件的路由器叫做NAT路由器,它至少有一个有效的外部全球地址IPG。 所有使用本地地址的主机在和外界通信时都要在NAT路由器上将其本地地址转换成外部全球…...

什么是CRM管理软件?CRM的基本概念、功能、选择标准、应用场景

什么是CRM管理软件? 嘿,大家好!今天咱们聊聊一个在现代企业管理中非常重要的工具——CRM管理软件。CRM是Customer Relationship Management(客户关系管理)的缩写,简单来说,它就是一个帮助企业和…...

Python编程常用的19个经典案例

Python 的简洁和强大使其成为许多开发者的首选语言。本文将介绍36个常用的Python经典代码案例。这些示例覆盖了基础语法、常见任务、以及一些高级功能。 1. 列表推导式 fizz_buzz_list ["FizzBuzz" if i % 15 0 else "Fizz" if i % 3 0 else "Buzz…...

【Unity基础】AudioSource 常用方法总结

在 Unity 中,AudioSource 组件用于控制音频的播放和管理。以下是常用的 AudioSource 控制方法及其说明。 1. 播放和暂停音频 Play():开始播放音频,如果是从暂停的地方继续播放,可以直接调用。Pause():暂停当前播放的…...

CSS系列(25)-- 滚动优化详解

前端技术探索系列:CSS 滚动优化详解 📜 致读者:探索流畅滚动的艺术 👋 前端开发者们, 今天我们将深入探讨 CSS 滚动优化,学习如何创建流畅、高性能的滚动体验。 平滑滚动 🚀 基础设置 /* …...

CST天线设计的六大核心特点:为天线分析提供完整解决方案!

CST Studio Suite 为天线设计提供了从最初的概念评估到最终的合规性测试所需的所有功能,确保天线设计在各种环境下实现稳定通信。这一套工具覆盖了所有重要的设计阶段,帮助设计师顺利完成从概念到成品的全过程。 下面我们来看一看CST电磁仿真中天线设计…...

Ubuntu下C语言操作kafka示例

目录 安装kafka: 安装librdkafka consumer Producer 测试运行 安装kafka: Ubuntu下Kafka安装及使用_ubuntu安装kafka-CSDN博客 安装librdkafka github地址:GitHub - confluentinc/librdkafka: The Apache Kafka C/C library $ apt in…...

怎么将pdf中的某一个提取出来?介绍几种提取PDF中页面的方法

怎么将pdf中的某一个提取出来?传统上,我们可能通过手动截取屏幕或使用PDF阅读器的复制功能来提取信息,但这种方法往往不够精确,且无法保留原文档的排版和格式。此外,很多时候我们需要提取的内容可能涉及多个页面、多个…...

HTTP接口报错详解与解决 200,500,403,408,404

前言: 仅做学习记录,侵删 背景 当后端编写接口时,经常需要对接口使用ApiFox或者PostMan进行测试,此时就会出现各种各样的报错,一般都会包括报错编码:200,400,401等。这个状态码一般是服务器所返回的包含…...

监控IP频繁登录服务器脚本

该脚本的作用是监控IP登录失败次数,如果某个IP的登录失败次数超过设定的最大次数,则阻止该IP的进一步登录尝试。通过iptables防火墙阻止连接,当一个IP尝试登录次数超过5次时,iptables会阻止来自该IP的所有连接 #!/bin/bashfuncti…...

分布式链路追踪-03-Jaeger、Zipkin、skywalking 中的 span 是如何设计的?

开源项目 auto-log 自动日志输出 Jaeger、Zipkin 中的 spanId 是如何生成的? 在 Jaeger 和 Zipkin 这两个分布式跟踪系统中,Span ID 是通过不同的方法生成的。 下面分别介绍它们的生成方式: Jaeger 中的 Span ID 生成: 在 Ja…...

【达梦数据库】获取对象DDL

目录 背景获取表的DDL其他 背景 在排查问题时总会遇到获取对象DDL的问题,因此做以下总结。 获取表的DDL 设置disql工具中显示LONG类型数据的最大长度,避免截断: SET LONG 9999获取DDL SELECT DBMS_METADATA.GET_DDL(TABLE,表名,模式名) …...

InnoDB和MyISAM引擎优缺点和区别

nnoDB和MyISAM是MySQL数据库中常用的两种存储引擎。它们各自具有不同的特性和优势,适用于不同的应用场景。 一、InnoDB引擎: 1、它有如下特性: 1)、支持事务(ACID) 2)、支持外键约束(FOREIGN KEY const…...

文件上传知识点汇总

归纳总结一下文件上传(其实是懒得写wp) 基于Dream ZHO师傅的CTF show 文件上传篇(web151-170,看这一篇就够啦)-CSDN博客 和dota_st 师傅的ctfshow-Web1000题系列修炼(一) | dota_st 做一篇自己的总结 目录 一、什么…...

计算机网络技术基础:5.数据通信系统

一、数据通信的基本概念 1.信息 信息是对客观事物的运动状态和存在形式的反映,可以是客观事实的形态、大小、结构、性能等描述,也可以是客观事物与外部之间的联系。信息的载体可以是数字、文字、语音、图形和图像等。计算机及其外围设备产生和交换的信息…...

光谱相机在农业的应用

一、作物生长监测1、营养状况评估 原理:不同的营养元素在植物体内的含量变化会导致植物叶片或其他组织的光谱反射率特性发生改变。例如,氮元素是植物叶绿素的重要组成部分,植物缺氮时,叶绿素含量下降,其在可见光波段&a…...

高考志愿填报:如何制定合理的志愿梯度?

高考志愿填报中常见的避雷行为,深入分析了专业选择、招生政策了解、学校选择、备选方案准备以及防诈骗等方面的关键问题,并提出了针对性的建议与策略。旨在为考生和家长提供实用的指导,助力考生科学合理地填报高考志愿,避免陷入各…...

Android基于Path的addRoundRect,Canvas剪切clipPath简洁的圆角矩形实现,Kotlin(1)

Android基于Path的addRoundRect&#xff0c;Canvas剪切clipPath简洁的圆角矩形实现&#xff0c;Kotlin&#xff08;1&#xff09; <?xml version"1.0" encoding"utf-8"?> <LinearLayout xmlns:android"http://schemas.android.com/apk/res…...

synchronized 学习

学习源&#xff1a; https://www.bilibili.com/video/BV1aJ411V763?spm_id_from333.788.videopod.episodes&vd_source32e1c41a9370911ab06d12fbc36c4ebc 1.应用场景 不超卖&#xff0c;也要考虑性能问题&#xff08;场景&#xff09; 2.常见面试问题&#xff1a; sync出…...

Python爬虫实战:研究feedparser库相关技术

1. 引言 1.1 研究背景与意义 在当今信息爆炸的时代,互联网上存在着海量的信息资源。RSS(Really Simple Syndication)作为一种标准化的信息聚合技术,被广泛用于网站内容的发布和订阅。通过 RSS,用户可以方便地获取网站更新的内容,而无需频繁访问各个网站。 然而,互联网…...

【解密LSTM、GRU如何解决传统RNN梯度消失问题】

解密LSTM与GRU&#xff1a;如何让RNN变得更聪明&#xff1f; 在深度学习的世界里&#xff0c;循环神经网络&#xff08;RNN&#xff09;以其卓越的序列数据处理能力广泛应用于自然语言处理、时间序列预测等领域。然而&#xff0c;传统RNN存在的一个严重问题——梯度消失&#…...

高等数学(下)题型笔记(八)空间解析几何与向量代数

目录 0 前言 1 向量的点乘 1.1 基本公式 1.2 例题 2 向量的叉乘 2.1 基础知识 2.2 例题 3 空间平面方程 3.1 基础知识 3.2 例题 4 空间直线方程 4.1 基础知识 4.2 例题 5 旋转曲面及其方程 5.1 基础知识 5.2 例题 6 空间曲面的法线与切平面 6.1 基础知识 6.2…...

ETLCloud可能遇到的问题有哪些?常见坑位解析

数据集成平台ETLCloud&#xff0c;主要用于支持数据的抽取&#xff08;Extract&#xff09;、转换&#xff08;Transform&#xff09;和加载&#xff08;Load&#xff09;过程。提供了一个简洁直观的界面&#xff0c;以便用户可以在不同的数据源之间轻松地进行数据迁移和转换。…...

Spring AI 入门:Java 开发者的生成式 AI 实践之路

一、Spring AI 简介 在人工智能技术快速迭代的今天&#xff0c;Spring AI 作为 Spring 生态系统的新生力量&#xff0c;正在成为 Java 开发者拥抱生成式 AI 的最佳选择。该框架通过模块化设计实现了与主流 AI 服务&#xff08;如 OpenAI、Anthropic&#xff09;的无缝对接&…...

select、poll、epoll 与 Reactor 模式

在高并发网络编程领域&#xff0c;高效处理大量连接和 I/O 事件是系统性能的关键。select、poll、epoll 作为 I/O 多路复用技术的代表&#xff0c;以及基于它们实现的 Reactor 模式&#xff0c;为开发者提供了强大的工具。本文将深入探讨这些技术的底层原理、优缺点。​ 一、I…...

Android 之 kotlin 语言学习笔记三(Kotlin-Java 互操作)

参考官方文档&#xff1a;https://developer.android.google.cn/kotlin/interop?hlzh-cn 一、Java&#xff08;供 Kotlin 使用&#xff09; 1、不得使用硬关键字 不要使用 Kotlin 的任何硬关键字作为方法的名称 或字段。允许使用 Kotlin 的软关键字、修饰符关键字和特殊标识…...

学校时钟系统,标准考场时钟系统,AI亮相2025高考,赛思时钟系统为教育公平筑起“精准防线”

2025年#高考 将在近日拉开帷幕&#xff0c;#AI 监考一度冲上热搜。当AI深度融入高考&#xff0c;#时间同步 不再是辅助功能&#xff0c;而是决定AI监考系统成败的“生命线”。 AI亮相2025高考&#xff0c;40种异常行为0.5秒精准识别 2025年高考即将拉开帷幕&#xff0c;江西、…...

sipsak:SIP瑞士军刀!全参数详细教程!Kali Linux教程!

简介 sipsak 是一个面向会话初始协议 (SIP) 应用程序开发人员和管理员的小型命令行工具。它可以用于对 SIP 应用程序和设备进行一些简单的测试。 sipsak 是一款 SIP 压力和诊断实用程序。它通过 sip-uri 向服务器发送 SIP 请求&#xff0c;并检查收到的响应。它以以下模式之一…...