将python编写的网站制作成docker镜像并上传到Github Packages上

news2025/1/8 5:42:57

文章目录

  • 前言
  • Docker
    • 安装docker
      • 注意事项
    • 创建Dockerfile
      • 注意事项
    • 构建 Docker 镜像
    • 运行 Docker 镜像
  • 发布到Github Packages
    • 坑坑到位
      • 申请token的坑
      • docker登录的坑
      • 给镜像添加标签的坑
      • docker推送的坑
  • 在Github Packages上查看
  • 总结

前言

还记得上一篇《借助ChatGPT使用Python搭建一个工具网站》总结中我利用ChatGPT写了一个网站,最终它运行良好,就在昨天我看到了Github Packages,不久前刚刚使用了GitHub Actions,我发现Github在被微软收购后,并没有变的更“闭源”,之前广大网友还在调侃,最大的闭源软件公司收购了最大的开源平台,看来一切还在向好的方向发展,简单介绍下前面提到的这两个都是什么东西。

GitHub Actions 是一个Github原生的持续集成和部署的工作流组件。通俗来说就是Github免费给你提供虚拟主机,由你编写工作流脚本来进行源码的检出,编译,测试,和发布。类似的我们可以想象成Github给每个仓库都免费绑定了一个Jenkins服务,编写pipeline脚本即可进行源码的集成和发布。

GitHub Packages 是一个和每一个代码仓库关联的软件包仓库。通俗来说就是代码仓库中存放的是源码,软件包仓库中存放的是编译输出后的可以被各个语言生态的依赖管理工具直接依赖的lib,类似的我们熟知的有maven中央仓库和nmp仓库。

今天我们只看GitHub Packages这部分,并且将范围缩小到制作一个Docker镜像并上传到GitHub Packages,用来制作的项目就是前文中我们在ChatGPT辅助下编写的网站,本文只写流程,对于其中的概念不会过多解释,大家可以借助网络来解决概念性问题,但是我在完成目标过程中遇到的坑会列举出来,希望能帮助有相同经历的人,同时自己记录一下防止忘记。

Docker

既然要制作docker镜像,那先简单了解下docker是什么

Docker 使用 Google 公司推出的 Go 语言 进行开发实现,基于 Linux 内核的 cgroup,namespace,以及 OverlayFS 类的 Union FS 等技术,对进程进行封装隔离,属于 操作系统层面的虚拟化技术。由于隔离的进程独立于宿主和其它的隔离的进程,因此也称其为容器。最初实现是基于 LXC,从 0.7 版本以后开始去除 LXC,转而使用自行开发的 libcontainer,从 1.11 版本开始,则进一步演进为使用 runC 和 containerd

Docker 可以通过定制应用镜像来实现持续集成、持续交付、部署。开发人员可以通过 Dockerfile 来进行镜像构建,并结合 持续集成(Continuous Integration) 系统进行集成测试,而运维人员则可以直接在生产环境中快速部署该镜像,甚至结合 持续部署(Continuous Delivery/Deployment) 系统进行自动部署。而且使用 Dockerfile 使镜像构建透明化,不仅仅开发团队可以理解应用运行环境,也方便运维团队理解应用运行所需条件,帮助更好的生产环境中部署该镜像。

以上我只是摘抄了一部分,想要看更详细的解释请跳转至 《Docker 技术入门与实战》,总的来说就是能解决环境一致性问题,并且可以方便部署的,把你的软件及运行环境打个包,在目标机器上直接运行,最大程度的去除环境差异,被打成的这个包就是docker镜像,我们可以把这个镜像丢到“任何”一台机器上去执行,不用再操心安装繁琐的运行环境。

但是无论是制作docker镜像还是运行docker镜像都需要先安装docker环境,相当于它来做软件和机器之间的代理,是不是听起来像虚拟机,他俩常常被拿来比较,而docker更加轻量级,目前应用广泛,我们来列举下安装步骤,事先说明操作系统是Ubuntu20.04,项目是minimalist-tool-web,未特殊说明时则是在项目根目录下执行命令。

安装docker

  • 更新apt包列表

    sudo apt update
    
  • 安装依赖包,以允许apt通过HTTPS使用存储库

    sudo apt install -y apt-transport-https ca-certificates curl gnupg2 software-properties-common
    
  • 添加Docker的官方GPG密钥

    curl -fsSL https://download.docker.com/linux/ubuntu/gpg | sudo apt-key add -
    
  • 添加Docker的APT存储库

    sudo add-apt-repository "deb [arch=amd64] https://download.docker.com/linux/ubuntu $(lsb_release -cs) stable"
    
  • 再次更新apt包列表

    sudo apt update
    
  • 安装Docker

    sudo apt install -y docker-ce
    
  • 验证Docker是否安装成功:

    sudo docker --version
    

注意事项

  1. 添加GPG密钥的时候有可能报错 curl: (35) OpenSSL SSL_connect: SSL_ERROR_SYSCALL in connection to download.docker.com:443,相同命令多尝试几次就能看见 OK

  2. 我在 sudo apt update 的时候报错 Failed to fetch https://dl.google.com/linux/chrome/deb/dists/stable/InRelease The following signatures couldn't be verified because the public key is not available: NO_PUBKEY E88979FB9B30ACF2,通过执行 wget -q -O - https://dl.google.com/linux/linux_signing_key.pub | sudo apt-key add - 修复好的,不过这可能是个例,你们不一定遇到

创建Dockerfile

在项目根目录下创建一个名为 Dockerfile 的文件,并编写 Docker 镜像的构建指令,以下是一个示例的 Dockerfile 文件,它是在初始命令模板下精简而成,目的是缩小镜像大小,调整命令后镜像从1.1G缩小到65M

# 使用 Alpine Linux 作为基础镜像
FROM python:3.8-alpine

# 设置工作目录
WORKDIR /app

# 将当前目录下的所有文件复制到工作目录
COPY . .

# 安装项目依赖
RUN pip install --no-cache-dir -r requirements.txt

# 清理不必要的缓存
RUN rm -rf /var/cache/apk/*

# 暴露应用端口
EXPOSE 9206

# 运行应用
CMD ["python3", "main.py"]

注意事项

  • WORKDIR /app 是指镜像中的目录,在当前操作系统中可能并不存在
  • COPY . . 很奇怪的写法,但是两个.含义是不同的,第一个只当前目录,第二个指镜像中的当前目录,也就是 /app

构建 Docker 镜像

$ sudo docker build -t minimalist-tool-web .

其中 -t minimalist-tool-web 指定了镜像的名称为 minimalist-tool-web,构建完成后,可以查看当前镜像

$ sudo docker images
REPOSITORY            TAG       IMAGE ID       CREATED          SIZE
minimalist-web-tool   latest    f11497961379   5 minutes ago    65.6MB
hello-world           latest    d2c94e258dcb   10 months ago    13.3kB

运行 Docker 镜像

docker run -d -p 9206:9205 minimalist-tool-web

其中 -d参数表示在后台运行容器,-p 9206:9205 将容器内部的 9205 端口映射到宿主机的 9206 端口,这样就可以通过 http://localhost:9206 访问应用程序,至此,我的项目就已经制作成了 Docker 镜像,并且可以在 Docker 容器中运行了。

发布到Github Packages

这部分确实花了我不少时间,我先把正确的流程描述完,然后再说一下我遇到的问题。

  1. 到Developer Settings (classic) 申请一个token,记得至少勾选 read:packageswrite:packages 权限

  2. 登录docker,使用上一步申请的token

sudo docker login ghcr.io --username your-github-username --password-stdin your-personal-access-token
  1. 给镜像添加标签
sudo docker tag your-image-id ghcr.io/your-github-username/your-image-name:1.0.0
  1. 将docker镜像推送到Github Packages上
sudo docker push ghcr.io/your-github-username/your-image-name:1.0.0

坑坑到位

申请token的坑

许多教程上申请token截图还是只有一个页面,但是现在(2024-3-8)的Github上申请的token的页面已经分成了两类,具体导航路径在『Github头像』->『Settings』->『Developer Settings』->『Personal access tokens』,到此处申请token页面分成了两个,分别是 Fine-grained tokensTokens (classic),最近几次其他的访问权限分配我都是用的前面这种Fine-grained tokens,虽然标记着 Beta,但这是官方推荐的形式。

但这一次我却遇到了 unauthorized: unauthenticated: User cannot be authenticated with the token provided. 这个问题,后来改成Tokens (classic)这种类型的token才解决,可以看到的是第一个Fine-grained tokens中我没有找到read:packageswrite:packages 权限,不知道是不是和这个有关,我还在尝试。

那么在申请 Tokens (classic) 时要记得勾选read:packageswrite:packages 权限,最好把 delete:packagesrepo 也勾选上,如果想通过token更新Github Actions,最好也要勾选上 workflow

Fine-grained tokens:这种token形如 github_pat_xxx,位数较长

Tokens (classic):这种token形如 ghp_xxx,位数较短

docker登录的坑

很多帖子可能是时间太久远了,里面提到的地址都是 docker.pkg.github.com,我这里倒是都显示登录成功,但是每次推送都会提示 no basic auth credentials,就像下面这样

$ cat ~/TOKEN.txt | docker login https://docker.pkg.github.com -u AlbertGithubHome --password-stdin
WARNING! Your password will be stored unencrypted in /home/shz/.docker/config.json.
Configure a credential helper to remove this warning. See
https://docs.docker.com/engine/reference/commandline/login/#credentials-store

Login Succeeded

$ sudo docker push docker.pkg.github.com/albertgithubhome/minimalist-tool-web/tools-web:1.0.0
The push refers to repository [docker.pkg.github.com/albertgithubhome/minimalist-tool-web/tools-web]
c0f047bfac29: Preparing
87174263225b: Preparing
5462d39b06ad: Preparing
2c604764da2f: Preparing
a469c5a79f90: Preparing
7402639a4746: Waiting
42672d5bf49e: Waiting
678cac8b069e: Waiting
d4fc045c9e3a: Waiting
no basic auth credentials

后来在 Stack Overflow 上找到一个问题,里面用的是 ghcr.io 修改之后发现才可以

给镜像添加标签的坑

这个需要注意的点是整个标签的字母必须都是小写的,而我的用户名是大写,所以我按照老版本的教程创建了类似 docker tag f11497961379 docker.pkg.github.com/albertgithubhome/minimalist-tool-web/tools-web:1.0.0 的标签,但是我的用户名是大写的,每次推送失败告诉我未验证时,我无法确定是哪类的问题:

  1. token类型有问题
  2. 权限有问题,登录没有用sudo
  3. 登录地址有问题,https://docker.pkg.github.com地址打不开
  4. 我的用户名有大写字母,与推送的镜像名中的用户名不匹配

反正费了九牛二虎之力吧,把这些原因一一排除掉,总算能推送上去了

还有一点,之前老版本的教程标签格式要求是 docker.pkg.github.com/your-github-username/your-repository/your-image-name:1.0.0,在新版的教程中改为了 ghcr.io/your-github-username/your-image-name:1.0.0,不要求添加库的名称了

docker推送的坑

这一步的坑都是前面几步积累得来的,因为这一步的失败根本不知道前边哪一步出错了,更离谱的是在登录这一步,无论用户名是什么,token是什么都显示登录成功,导致后面查找原因浪费了不少时间

推送时直接填写前面的标签名即可

$ sudo docker push ghcr.io/albertgithubhome/minimalist-tool-web:1.0.0
The push refers to repository [ghcr.io/albertgithubhome/minimalist-tool-web]
c0f047bfac29: Pushed
87174263225b: Pushed
5462d39b06ad: Pushed
2c604764da2f: Pushed
a469c5a79f90: Pushed
7402639a4746: Pushed
42672d5bf49e: Pushed
678cac8b069e: Pushed
d4fc045c9e3a: Pushed
1.0.0: digest: sha256:f749fd2eba021248c97c8a44374772165372354b9d48f4f6b845bbce46d02402 size: 2201

这些参数可以在 ghcr.io 查看,页面上有一个示例

ghcr.io

echo $PAT | docker login ghcr.io --username phanatic --password-stdin
docker tag app ghcr.io/phanatic/app:1.0.0
docker push ghcr.io/phanatic/app:1.0.0

在Github Packages上查看

打开你账号下的Packages页签就能看到了,点击进去可以将这个 package 关联到指定项目,这样在项目的首页右侧的packages区域就能看到这个package,在package的介绍页面会教你怎么使用这个镜像

docker pull ghcr.io/albertgithubhome/minimalist-tool-web:1.0.0

minimalist-tool-web

总结

  • GitHub Actions 是一个Github原生的持续集成和部署的工作流组件,相当免费绑定了一个Jenkins服务
  • GitHub Packages 是和每一个代码仓库关联的软件包仓库,类似我们熟知的maven中央仓库和nmp仓库
  • 运行docker命令 docker run -d -p 9206:9205 minimalist-tool-web,其中minimalist-tool-web是镜像名
  • 停止docker命令 sudo docker stop af98cf2f93bf,其中af98cf2f93bf是容器ID
  • 推送docker命令 sudo docker push ghcr.io/albertgithubhome/minimalist-tool-web:1.0.0
  • 拉取docker命令 sudo docker pull ghcr.io/albertgithubhome/minimalist-tool-web:1.0.0
  • 最终Fine-grained tokens应该会替换到Tokens (classic),但现在感觉还不能完全代替
  • 接下来会尝试用.yml编写一个GitHub Action控制项目更新时自动创建镜像和发布

==>> 反爬链接,请勿点击,原地爆炸,概不负责!<<==

从秦灭六国、焚书坑儒开始,从罢黜百家、独尊儒术开始,洗脑便已经登峰造极了~

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

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

相关文章

借助Aspose.html控件,在 Java 中将 URL 转换为 PDF

如果您正在寻找一种将实时 URL 中的网页另存为 PDF文档的方法&#xff0c;那么您来对地方了。在这篇博文中&#xff0c;我们将学习如何使用 Java 将 URL 转换为 PDF。从实时 URL转换HTML网页可以像任何其他文档一样保存所需的网页以供离线访问。将网页保存为 PDF 格式可以轻松突…

PandasAI—让AI做数据分析

安装 pip install pandasai !pip install --upgrade pandas pandasai 导入依赖项 import pandas as pdfrom pandasai import PandasAIfrom pandasai.llm.openai import OpenAI使用pandas创建一个数据框 df pd.DataFrame({"country": ["United States",…

MKS Type 631C 真空计说明详细内容查看图中目录

MKS Type 631C 真空计说明详细内容查看图中目录

【sw网络监控】通过snmp协议相关的snmp-exporter(收集交换机网络监控数据)+ promethus + grafana

本站以分享各种运维经验和运维所需要的技能为主 《python零基础入门》&#xff1a;python零基础入门学习 《python运维脚本》&#xff1a; python运维脚本实践 《shell》&#xff1a;shell学习 《terraform》持续更新中&#xff1a;terraform_Aws学习零基础入门到最佳实战 《k8…

C++异常处理机制

目录 一&#xff0c;传统C语言处理异常 二&#xff0c;C异常概念 三&#xff0c;使用 1. 自定义异常体系 2. 在函数调用链中异常栈展开匹配原则 3. 异常的重新抛出 四&#xff0c;异常规范 五&#xff0c;异常安全 六&#xff0c;C标准库的异常体系 七&#xff0c;异…

【开源】JAVA+Vue.js实现高校宿舍调配管理系统

目录 一、摘要1.1 项目介绍1.2 项目录屏 二、功能需求2.1 学生端2.2 宿管2.3 老师端 三、系统展示四、核心代码4.1 查询单条个人习惯4.2 查询我的室友4.3 查询宿舍4.4 查询指定性别全部宿舍4.5 初次分配宿舍 五、免责说明 一、摘要 1.1 项目介绍 基于JAVAVueSpringBootMySQL的…

liteIDE 解决go root报错 go: cannot find GOROOT directory: c:\go

liteIDE环境配置 我使用的liteIDE为 x36 5.9.5版本 。在查看–>选项 中可以看到 LiteEnv&#xff0c;双击LiteEnv &#xff0c;在右侧选择对应系统的env文件&#xff0c;我的是win64系统&#xff0c;所以文件名为win64.env 再双击 win64.env &#xff0c;关闭当前窗口&…

python淘宝网页爬虫数据保存到 csv和mysql(selenium)

数据库连接设置&#xff08;表和字段要提前在数据库中建好&#xff09; # 数据库中要插入的表 MYSQL_TABLE goods# MySQL 数据库连接配置,根据自己的本地数据库修改 db_config {host: localhost,port: 3306,user: root,password: ma*****6,database: may2024,charset: utf8mb…

TCP包头、TCP为什么安全可靠、UDP和TCP的区别、http协议

我要成为嵌入式高手之3月8日Linux高编第十八天&#xff01;&#xff01; __________________________________________________ 学习笔记 TPC包头 1、序号 发送端发送数据包的编号 2、确认号 已经确认接收到的数据的编号&#xff0c;只有当ACK为1时&#xff0c;该位才有用 …

简介maven核心:pom项目对象模型

Maven Maven 意思是知识的积累者&#xff0c;最初是为了简化 Jakarta Turbine 项目中的构建过程。有几个项目&#xff0c;每个项目都有自己的 Ant 构建文件&#xff0c;它们都略有不同。JAR 被检入 CVS。我们想要一种标准的方式来构建项目&#xff0c;清楚地定义项目的组成&am…

Jmeter 测试使用基本组件结构

JMeter简介 Apache组织开发的开源免费压测工具纯Java程序&#xff0c;跨平台性强源程序可以从网上下载高扩展性可对服务器、网络或对象模拟巨大的负载&#xff0c;进行压力测试可以用于接口测试支持分布式、多节点部署 JMeter安装 下载位置 官网https://jmeter.apache.org/ …

基于单片机的灭火机器人设计

目 录 摘 要 I Abstract II 引 言 1 1 系统方案设计 4 1.1 方案论证 4 1.2 灭火机器人系统工作原理 4 2 系统硬件设计 6 2.1 单片机 6 2.2 火焰探测系统设计 8 2.3 灭火系统设计 8 2.4 循迹模块设计 9 2.5 电机驱动模块 10 3 系统软件设计 12 3.1 系统软件开发环境 12 3.2 系统…

防御保护 IPSEC VPPN实验

实验背景&#xff1a;FW1和FW2是双机热备 主备备份模式。 实验要求&#xff1a;在FW5和FW3之间建立一条IPSEC通道&#xff0c;保证10.0.2.0/24网段可以正常访问到192.168.1.0/24 IPSEC VPPN实验配置&#xff08;由于是双机热备状态&#xff0c;所以FW1和FW2只需要配置FW1主设…

STM32标准库——(20)WDG看门狗

1.WDG简介 独立看门狗(IWDG)由专用的低速时钟(LSI)驱动&#xff0c;即使主时钟发生故障它也仍然有效。窗口看门狗由从APB1时钟分频后得到的时钟驱动&#xff0c;通过可配置的时间窗口来检测应用程序非正常的过迟或过早的操作 2.IWDG框图 它的结构和定时器是非常相似的&#xf…

打印机项目需求

打印机项目需求 工作情况如下 我拿着一张带有二维码的纸张对准“打印机的摄像头”“打印机的摄像头”解析我的二维码假如解析后的二维码内容是&#xff1a;24030924发送一个http请求&#xff1a;https://…/getMessage?code24030924外部服务器会返回一个json数据{‘地址’:‘…

Python编程实验六:面向对象应用

目录 一、实验目的与要求 二、实验内容 三、主要程序清单和程序运行结果 第1题 第2题 四、实验结果分析与体会 一、实验目的与要求 &#xff08;1&#xff09;通过本次实验&#xff0c;学生应掌握类的定义与对象的创建、类的继承与方法的覆盖&#xff1b; &#xff08;2…

高级语言讲义2016计专(仅高级语言部分)

1.斐波那契序列的第n项可以表示成以下形式&#xff0c;编写一个非递归函数&#xff0c;返回该数列的第n项的数值 #include <stdio.h>int func(int n) {if(n1||n2)return 1;int p1,q1,num;for(int i3; i<n; i) {numpq;qp;pnum;}return num; } 2.在MXN的二维数组A中&am…

鸿蒙Harmony应用开发—ArkTS声明式开发(基础手势:LoadingProgress)

用于显示加载动效的组件。 说明&#xff1a; 该组件从API Version 8开始支持。后续版本如有新增内容&#xff0c;则采用上角标单独标记该内容的起始版本。 子组件 无 接口 LoadingProgress() 创建加载进展组件。 从API version 9开始&#xff0c;该接口支持在ArkTS卡片中使…

k8s常用命令大全

k8s常用的命令 下面是一些常用的Kubernetes&#xff08;K8s&#xff09;命令&#xff0c;以及它们的简要说明。这些命令可以帮助您管理和操作Kubernetes集群中的资源。 集群管理命令&#xff1a; kubectl cluster-info: 显示集群的基本信息。 kubectl config use-context &l…

大语言模型如何充分理解人类自然语言指令

经过海量数据预训练后的语言模型虽然具备了大量的知识&#xff0c;但是由于其训练的目标仅是进行下一个词的预测&#xff0c;此时的模型还不能够理解并遵循人类自然语言的指令。指令微调(Instruction Tuning)&#xff0c;是指在已经训练好的语言模型的基础上&#xff0c;通过使…