口玉 發表於 2019-7-1 08:04:00

kubernetes实战篇之创建密钥自动拉取私服镜像

<blockquote>
<p>系列目录</p>
</blockquote>
<p>前面我们讲解了如何搭建nexus服务器,以及如何使用nexus搭建docker私有镜像仓库,示例中我们都是手动<code>docker login</code>登陆私服,然后通过命令拉取镜像然后运行容器.然而这种做法在kubernetes集群中是不可行的.第一,项目规模不同,每天产生的镜像数量也不同,如果每天产生大量仓库都要手动执行<code>docker pull</code>来拉取,非常麻烦也非常容易出错.第二,集群规模不同,节点数量也不同,少则三五个,多则成百上千甚至更多.我们一台台拉取显然非常麻烦,即便使用ansible脚本批量执行命令也同样存在问题:因为不同的容器分布在不同的节点上.把所有镜像都用批量脚本拉到所有服务器上显然会浪费非常多的资源,并且集群扩容或者缩容器都需要更改脚本,很容易出现错误.这时候我们更倾向使用kubernetes自身的强大管理功能.其实kubernetes可以把docker的登陆信息做成secrets,在容器编排时显式指定要用到的secret,kubernetes就会自动去拉取私服上编排时指定的镜像.大大方便我们自动化的流程.下面我们就讲解如何如何制作docker私仓拉取secrets以及如何在拉取镜像时使用.</p>
<p>我们在拉取公仓镜像的时候,是不需要输入账户密码的.但是私仓往往都是要登陆以后才能拉取里面的镜像.前面讲解nexus docker私服的时候我们也讲到过,所有操作的前提就是使用<code>docker login -u 用户名 -p 密码 服务器</code>首先登陆私服.我们登陆过一次后下次再次登陆其实不用指定用户名和密码,比如,我可以使用以下命令直接登陆:</p>
<pre><code class="language-bash"># docker login 192.168.124.43:8002
Authenticating with existing credentials...
WARNING! Your password will be stored unencrypted in /root/.docker/config.json.
Configure a credential helper to remove this warning. See
https://docs.docker.com/engine/reference/commandline/login/#credentials-store

Login Succeeded
#

</code></pre>
<p>可以看到没有指定用户名和密码我们也同样登陆了.实际上,我们首次登陆成功后,docker便把登陆信息存在了<code>~/.docker/config.json</code>这个文件里,我们可以看一下</p>
<pre><code class="language-json"># .
{
      "auths": {
                "192.168.124.43:8002": {
                        "auth": "YWRtaW46YWRtaW4xMjM="
                }
      },
      "HttpHeaders": {
                "User-Agent": "Docker-Client/18.09.4 (linux)"
      }
}#

</code></pre>
<h2 id="基于现有docker登陆信息创建kubernetes密钥">基于现有docker登陆信息创建kubernetes密钥</h2>
<p>命令如下</p>
<pre><code class="language-bash">kubectl create secret generic regcred \
    --from-file=.dockerconfigjson= ~/.docker/config.json \
    --type=kubernetes.io/dockerconfigjson
</code></pre>
<p>以上方法同其它创建kubernetes密钥的方法并没有区别,这里的关键是.docker目录下的config.json文件.</p>
<p>当然,如果你想对生成出来的secrets有更多的控制,比如指定这个secrets的名称空间,可以使用yml文件来声明式创建.步骤与上面略有不同,如下</p>
<ul>
<li>
<p>data字段的名称必须设置为<code>.dockerconfigjson</code></p>
</li>
<li>
<p>对config.json文件进行base64编码,然后把编码后的内容复制到<code>.dockerconfigjson</code>字段</p>
</li>
<li>
<p>类型设置为<code>kubernetes.io/dockerconfigjson</code></p>
</li>
</ul>
<p>示例:</p>
<pre><code class="language-yml">apiVersion: v1
kind: Secret
metadata:
name: myregistrykey
namespace: awesomeapps
data:
.dockerconfigjson: UmVhbGx5IHJlYWxseSByZWVlZWVlZWVlZWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWFhYWxsbGxsbGxsbGxsbGxsbGxsbGxsbGxsbGxsbGxsbGx5eXl5eXl5eXl5eXl5eXl5eXl5eSBsbGxsbGxsbGxsbGxsbG9vb29vb29vb29vb29vb29vb29vb29vb29vb25ubm5ubm5ubm5ubm5ubm5ubm5ubm5ubmdnZ2dnZ2dnZ2dnZ2dnZ2dnZ2cgYXV0aCBrZXlzCg==
type: kubernetes.io/dockerconfigjson
</code></pre>
<p>常用报错解析:</p>
<ul>
<li>
<p><code>error: no objects passed to create</code>这意味着base64编码信息无效</p>
</li>
<li>
<p><code>Secret "myregistrykey" is invalid: data[.dockerconfigjson]: invalid value ...</code> 这意味着base64编码成功,但是不能解码为<code>.docker/config.json</code></p>
</li>
</ul>
<h2 id="直接从命令行创建secret">直接从命令行创建secret</h2>
<p>上面是使用docker存储的已经登陆的信息创建的secret,如果没有使用docker登陆,也可以直接通过命令行来创建secret,命令格式如下:</p>
<pre><code>kubectl create secret docker-registry regcred --docker-server=&lt;your-registry-server&gt; --docker-username=&lt;your-name&gt; --docker-password=&lt;your-pword&gt; --docker-email=&lt;your-email&gt;
</code></pre>
<p>下面对上面字段做简要描述:</p>
<ul>
<li>
<p><your-registry-server> docker私服地址</your-registry-server></p>
</li>
<li>
<p><your-name> 登陆名</your-name></p>
</li>
<li>
<p><your-pword> 登陆密码</your-pword></p>
</li>
<li>
<p><your-email> 邮箱,此字段为非必填项</your-email></p>
</li>
</ul>
<p>这里仍然基于前面搭建的nexus docker私仓为例:</p>
<pre><code class="language-bash">kubectl create secret docker-registry regcred --docker-server=192.168.124.43:8002 --docker-username=admin --docker-password=admin123
</code></pre>
<p>便可以创建secret了.</p>
<h2 id="查看密钥信息">查看密钥信息</h2>
<p>可以使用以下命令查看刚创建的密钥:</p>
<pre><code>kubectl get secret regcred --output=yaml
</code></pre>
<pre><code class="language-bash">$ kubectl get secret regcred --output=yaml
apiVersion: v1
data:
.dockerconfigjson: eyJhdXRocyI6eyIxOTIuMTY4LjEyNC40Mzo4MDAyIjp7IlVzZXJuYW1lIjoiYWRtaW4iLCJQYXNzd29yZCI6ImFkbWluMTIzIiwiRW1haWwiOiIifX19
kind: Secret
metadata:
creationTimestamp: "2019-04-12T05:53:19Z"
name: regcred
namespace: default
resourceVersion: "3763835"
selfLink: /api/v1/namespaces/default/secrets/regcred
uid: 46028dd4-5ce7-11e9-bc12-0050568417a2
type: kubernetes.io/dockerconfigjson
$
</code></pre>
<p>其中<code>.dockerconfigjson</code>字段为docker的登陆信息,我们可以通过base64解码来查看它:</p>
<pre><code class="language-bash">kubectl get secret regcred --output="jsonpath={.data.\.dockerconfigjson}" | base64 --decode
</code></pre>
<pre><code class="language-bash">$ kubectl get secret regcred --output="jsonpath={.data.\.dockerconfigjson}" | base64 --decode
{"auths":{"192.168.124.43:8002":{"Username":"admin","Password":"admin123","Email":""}}}$

</code></pre>
<p>可以看到以上正信息与我们创建密码时输入的信息吻合</p>
<blockquote>
<p>注 以上是我们通过命令行生成的,如果是通过config.json生成的则解码后的信息与以上略有不同.通过config.json生成的密钥解码后有 个<code>auth</code>字段,它仍然是base64编码形式存在的,我们需要再次对其解码才能查看到信息</p>
</blockquote>
<p>比如</p>
<pre><code class="language-bash">$ echo "YWRtaW46YWRtaW4xMjM="|base64 --decode
admin:admin123
$
</code></pre>
<h2 id="创建pod使用密钥拉取私仓镜像">创建pod使用密钥拉取私仓镜像</h2>
<p>编排文件示例如下:</p>
<pre><code class="language-yml">apiVersion: v1
kind: Pod
metadata:
name: ubuntu-demo
spec:
restartPolicy: Never
containers:
- name: u-demo
    image: 192.168.124.43:8002/ubuntu
    imagePullPolicy: IfNotPresent
    command: ["printenv"]
    args: ["HOSTNAME"]
imagePullSecrets:
- name: regcred
</code></pre>
<p>以上示例我们使用了预先上传到私服里的一个ubuntu镜像来创建一个pod,由于以上镜像创建完成后马上就结束了.因此我们让打印一个环境变量信息,然后再使用log命令来查看,以证明操作是成功的.</p>
<p>镜像创建完成以后,我们查看pod的状态</p>
<pre><code class="language-bash">$ kubectl get po
NAME                        READY   STATUS      RESTARTS   AGE
busybox                     1/1   Running   552      23d
consul-0                      1/1   Running   2          28h
consul-1                      1/1   Running   3          28h
consul-2                      1/1   Running   2          28h
helloworld-7fdc8d9855-ncfdz   1/1   Running   3          30d
hostaliases-pod               1/1   Running   0          3h42m
ubuntu-demo                   0/1   Completed   0          50m
</code></pre>
<p>可以看到pod已处于完成状态,我们使用<code>kubectl logs</code>来查看它是否打印了信息</p>
<pre><code class="language-bash">$ kubectl logs ubuntu-demo
ubuntu-demo
</code></pre>
<p>可以看到,输出了host的名称.</p><br><br>
来源:https://www.cnblogs.com/tylerzhou/p/11112086.html
頁: [1]
查看完整版本: kubernetes实战篇之创建密钥自动拉取私服镜像