<?xml version="1.0" encoding="utf-8" standalone="yes" ?>
<rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom">
  <channel>
    <title>krew | 伪架构师</title>
    <link>/tags/krew/</link>
      <atom:link href="/tags/krew/index.xml" rel="self" type="application/rss+xml" />
    <description>krew</description>
    <generator>Source Themes Academic (https://sourcethemes.com/academic/)</generator><language>zh</language><lastBuildDate>Fri, 26 Nov 2021 16:36:44 +0800</lastBuildDate>
    <image>
      <url>/img/logo-wide.png</url>
      <title>krew</title>
      <link>/tags/krew/</link>
    </image>
    
    <item>
      <title>Release 自动更新 Krew Index</title>
      <link>/post/auto-refresh-krew-index/</link>
      <pubDate>Fri, 26 Nov 2021 16:36:44 +0800</pubDate>
      <guid>/post/auto-refresh-krew-index/</guid>
      <description>&lt;p&gt;前面介绍过&lt;a href=&#34;https://blog.fleeto.us/post/how-to-krew/&#34; target=&#34;_blank&#34;&gt;创建 Krew 插件的方法&lt;/a&gt;，其中生成插件描述文件的部分，可以说是又直接又麻烦，而且每次发布的时候，都要重新生成并刷新 Krew Index，于是我写了个&lt;a href=&#34;https://gist.github.com/fleeto/11eb8792b3b96e7f7b85846e95c5f4e3&#34; target=&#34;_blank&#34;&gt;脚本&lt;/a&gt;，可以根据既有信息和已经发布的二进制文件生成 YAML 文件。这个脚本的功能，简单说就是写一个 YAML 模板，然后下载二进制文件，计算哈希，生成最终的插件描述文件，把描述文件覆盖旧版本，然后发布 PR 即可。有了脚本之后其实还是挺麻烦的，不过在浏览 Krew 文档时发现有一节 &lt;a href=&#34;https://krew.sigs.k8s.io/docs/developer-guide/release/automating-updates/&#34; target=&#34;_blank&#34;&gt;Automating plugin updates&lt;/a&gt; ，其中提到了可以使用 &lt;a href=&#34;https://github.com/rajatjindal/krew-release-bot&#34; target=&#34;_blank&#34;&gt;Krew Release Bot&lt;/a&gt; 来自动地完成这个过程，官方强烈推荐使用这个 Bot 进行更新，理由是该 Bot 生成清单文件后自动提交的 PR，如果其中包含的变更内容只是简单的版本升级，能够被 &lt;a href=&#34;https://github.com/kubernetes-sigs/krew-index&#34; target=&#34;_blank&#34;&gt;Krew Index&lt;/a&gt; 的 Bot 自动处理，免除人工环节之后，整个更新过程可以在几分钟之内完成，避免了新版本二进制无法及时更新的尴尬。下面就介绍一下这个 Bot 的用法。&lt;/p&gt;

&lt;p&gt;这个 Bot 是一个 &lt;a href=&#34;https://github.com/features/actions&#34; target=&#34;_blank&#34;&gt;Github Action&lt;/a&gt; 的插件，其工作原理和之前的脚本类似，在前文&lt;a href=&#34;https://blog.fleeto.us/post/auto-build-with-github-action/&#34; target=&#34;_blank&#34;&gt;用 Github Action 自动发布二进制包&lt;/a&gt;中，在 &lt;a href=&#34;https://github.com/karmada-io/karmada&#34; target=&#34;_blank&#34;&gt;Karmada 项目&lt;/a&gt; 里使用 Github Action 进行了二进制文件的构建和发布，接下来继续使用这个基础，根据新发布的二进制来更新 Krew Index。&lt;/p&gt;

&lt;p&gt;要在项目里使用 Krew Release Bot，首先要求项目插件已经成功发布到 Krew Index 中；接下来，需要生成一个文件 &lt;code&gt;.krew.yaml&lt;/code&gt;，作为 Bot 的模板，这个模板和 Krew Index 的 YAML 是基本一致的，不过其中的二进制 URL 和 Hash 部分被替换为模板的占位符，例如 &lt;code&gt;{{addURIAndSha &amp;quot;https://github.com/karmada-io/karmada/releases/download/{{ .TagName }}/kubectl-karmada-darwin-arm64.tgz&amp;quot; .TagName }}&lt;/code&gt;，甚至连这一步简单操作，Bot 作者也帮你解决了——他做了一个 &lt;a href=&#34;https://rajatjindal.com/tools/krew-release-bot-helper/&#34; target=&#34;_blank&#34;&gt;Krew Release Bot Helper&lt;/a&gt;，在输入框输入已发布的 Karmada 插件名称，就可以自动生成 YAML 模板，放到项目里就可以了。&lt;/p&gt;

&lt;p&gt;接下来，在 Release Action 里面加入如下语句：&lt;/p&gt;

&lt;pre&gt;&lt;code class=&#34;language-yaml&#34;&gt;- name: Update new version in krew-index
  uses: rajatjindal/krew-release-bot@v0.0.40
&lt;/code&gt;&lt;/pre&gt;

&lt;p&gt;提交代码之后，创建一个新的 Release，会发现 Action 失败了，错误信息大概如下：&lt;/p&gt;

&lt;p&gt;&lt;img src=&#34;images/error.png&#34; alt=&#34;error&#34; /&gt;&lt;/p&gt;

&lt;p&gt;原因在于前面使用的 &lt;code&gt;Matrix&lt;/code&gt;，我们的 PR 应该在生成并上传所有的二进制文件之后才能发起，所以改成这样：&lt;/p&gt;

&lt;pre&gt;&lt;code class=&#34;language-yaml&#34;&gt;  update-krew-index:
    needs: release-assests
    name: Update krew-index
    runs-on: ubuntu-18.04
    steps:
    - uses: actions/checkout@master
    - name: Update new version in krew-index
      uses: rajatjindal/krew-release-bot@v0.0.40
&lt;/code&gt;&lt;/pre&gt;

&lt;p&gt;这里用了一个新的 &lt;code&gt;job&lt;/code&gt;，使用 &lt;a href=&#34;https://docs.github.com/en/actions/learn-github-actions/workflow-syntax-for-github-actions#jobsjob_idneeds&#34; target=&#34;_blank&#34;&gt;&lt;code&gt;needs&lt;/code&gt;&lt;/a&gt; 关键字明确指出，需要等待 &lt;code&gt;release-assests&lt;/code&gt; 任务完成。&lt;/p&gt;

&lt;p&gt;再次提交，并创建 Tag，创建 Release。Action 运行示意图如下：&lt;/p&gt;

&lt;p&gt;&lt;img src=&#34;images/follow.png&#34; alt=&#34;follow&#34; /&gt;&lt;/p&gt;

&lt;p&gt;这里生成的 URL 所指向的二进制文件是无法访问的，因此还是无法成功，但如果是官方自行发布的话，就可以了，而且如上文所说，这样生成的 PR 会在几分钟之内得到 Approve。&lt;/p&gt;

&lt;p&gt;读到这里会产生一个问题，会不会有人冒充发表呢？应该是不会的：Krew index 项目的 PR 机器人会对新 PR 的差异进行判定，如果不是典型的版本更新，会转入人工通道，因此新发的 PR 必须是和上一个版本具备这样的差异关系，而且必须是基于官方的二进制发布，才能够得到自动审批，因此完全可以放心使用。&lt;/p&gt;
</description>
    </item>
    
    <item>
      <title>如何编写一个支持 Krew 的 kubectl 插件</title>
      <link>/post/how-to-krew/</link>
      <pubDate>Thu, 02 Apr 2020 10:13:24 +0800</pubDate>
      <guid>/post/how-to-krew/</guid>
      <description>

&lt;h2 id=&#34;krew-简介&#34;&gt;krew 简介&lt;/h2&gt;

&lt;p&gt;&lt;a href=&#34;https://github.com/kubernetes-sigs/krew&#34; target=&#34;_blank&#34;&gt;Krew&lt;/a&gt; 是一个用来管理 Kubectl 插件的工具，名字大概来自于 OS X 下著名的软件包管理器 &lt;code&gt;Homebrew&lt;/code&gt;，使用 Krew 能够方便的查找、安装和使用 Kubectl 插件，例如：&lt;/p&gt;

&lt;pre&gt;&lt;code class=&#34;language-shell&#34;&gt;$ kubectl krew search
NAME                            DESCRIPTION                                         INSTALLED
access-matrix                   Show an RBAC access matrix for server resources     no
advise-psp                      Suggests PodSecurityPolicies for cluster.           no
...

$ kubectl krew install tree
Updated the local copy of plugin index.
Installing plugin: tree
...

$  kubectl tree deployment coredns -nkube-system
NAMESPACE    NAME                                READY  REASON  AGE
kube-system  Deployment/coredns                  -              140d
kube-system  └─ReplicaSet/coredns-76d9d9bcc7   -              140d
kube-system    ├─Pod/coredns-76d9d9bcc7-m6d4c  True           4d10h
kube-system    └─Pod/coredns-76d9d9bcc7-zvf9c  True           4d10h
&lt;/code&gt;&lt;/pre&gt;

&lt;p&gt;很方便的几个步骤，就可以查询、安装和使用新插件了。&lt;/p&gt;

&lt;p&gt;Krew 除了落在客户端的可执行文件之外，和其它软件包管理系统一样，也同样需要有一个索引系统，并根据索引进行软件查询和下载，下载之后的软件保存在本地，供 kubectl 调用。&lt;/p&gt;

&lt;h3 id=&#34;索引&#34;&gt;索引&lt;/h3&gt;

&lt;p&gt;Krew 的索引保存在一个名为 &lt;a href=&#34;https://github.com/kubernetes-sigs/krew-index&#34; target=&#34;_blank&#34;&gt;krew-index&lt;/a&gt; 的代码库中。其中的 &lt;code&gt;plugins&lt;/code&gt; 目录保存了一组 yaml 文件，就是插件的目录。&lt;/p&gt;

&lt;h3 id=&#34;yaml-清单&#34;&gt;YAML 清单&lt;/h3&gt;

&lt;p&gt;随意打开一个清单文件，可以看到这样的内容：&lt;/p&gt;

&lt;pre&gt;&lt;code class=&#34;language-yaml&#34;&gt;apiVersion: krew.googlecontainertools.github.com/v1alpha2
kind: Plugin
metadata:
  name: access-matrix
spec:
  version: v0.4.4
  platforms:
  - bin: access-matrix
    uri: https://github.com/corneliusweig/rakkess/releases/download/v0.4.4/access-matrix-amd64-linux.tar.gz
    sha256: 53b1ee5865d11360cea3e59b91cdc6707ee30845567e63657782ee11815f1de4
    files:
      - from: ./LICENSE
        to: .
      - from: ./access-matrix-amd64-linux
        to: access-matrix
    selector:
      matchLabels:
        os: linux
        arch: amd64
  shortDescription: Show an RBAC access matrix for server resources
  homepage: https://github.com/corneliusweig/rakkess
  caveats: |
      Usage:
        kubectl access-matrix
  description: ..
&lt;/code&gt;&lt;/pre&gt;

&lt;p&gt;其中 &lt;code&gt;apiVersion&lt;/code&gt; 和 &lt;code&gt;kind&lt;/code&gt; 是固定内容。&lt;code&gt;platforms&lt;/code&gt; 是一个数组，指定不同平台下的不同用法。下一级的 &lt;code&gt;bin&lt;/code&gt; 表明了执行命令；&lt;code&gt;uri&lt;/code&gt; 和 &lt;code&gt;sha256&lt;/code&gt; 分别指的是下载位置以及压缩包的校验码；接下来的 &lt;code&gt;files&lt;/code&gt; 是一个拷贝命令——从解压后的文件夹中拷贝文件；最后的 &lt;code&gt;selector&lt;/code&gt; 则是针对不同平台的选择标准。&lt;/p&gt;

&lt;p&gt;所以要编写一个能够通过 Krew 进行管理的 kubectl 插件，需要以下几个步骤：&lt;/p&gt;

&lt;ol&gt;
&lt;li&gt;编写插件代码&lt;/li&gt;
&lt;li&gt;制作清单和调试&lt;/li&gt;
&lt;li&gt;上传到 krew-index&lt;/li&gt;
&lt;/ol&gt;

&lt;p&gt;下面用一个实际的例子来说明一下这个过程。&lt;/p&gt;

&lt;h2 id=&#34;编写插件代码&#34;&gt;编写插件代码&lt;/h2&gt;

&lt;p&gt;插件代码本身的编写非常简单和随意，可以用你喜欢的任何语言，例如 golang、python 或者 shell。只有一个推荐的命名规则：&lt;code&gt;kubectl-rm&lt;/code&gt;，在 kubectl 中调用时就可以使用 &lt;code&gt;kubectl rm&lt;/code&gt; 了。例如我要编写一个对输出 JSON 进行过滤的插件，代码如下：&lt;/p&gt;

&lt;pre&gt;&lt;code class=&#34;language-bash&#34;&gt;#!/bin/sh

METADATA=${JSON_METADATA-&amp;quot;.metadata.resourceVersion, .metadata.selfLink, .metadata.managedFields, .metadata.generation, .metadata.uid, .metadata.creationTimestamp&amp;quot;}
STATUS=${JSON_STATUS-&amp;quot;.status&amp;quot;}
ANNOTATION=${JSON_ANNOTATION-&amp;quot;.metadata.annotations.\&amp;quot;kubectl.kubernetes.io/last-applied-configuration\&amp;quot;, .metadata.annotations.\&amp;quot;deployment.kubernetes.io/revision\&amp;quot;&amp;quot;}
SPEC=${JSON_SPEC-&amp;quot;.spec.template.metadata.creationTimestamp, .spec.revisionHistoryLimit, .spec.templateGeneration&amp;quot;}

if ! [ -x &amp;quot;$(command -v jq)&amp;quot; ]; then
  echo &#39;Error: jq is not installed.&#39; &amp;gt;&amp;amp;2
  exit 1
fi

if [ $# -lt 2 ]
  then
    echo &amp;quot;Usage: $0 [workload-type] [object-name] [other parameters for kubectl]&amp;quot;
    echo &amp;quot;Workload types: &#39;deployment&#39;, &#39;daemonset&#39;, &#39;configmap&#39;, &#39;statefulset&#39;, &#39;secret&#39;&amp;quot;
    echo &amp;quot;Example: $0 deploy coredns -n kube-system&amp;quot;
    exit 1
fi

TYPE=$1
NAME=$2
OTHER=$*

kubectl get ${OTHER} -ojson | jq -S &amp;quot;del(${METADATA}, ${STATUS}, ${ANNOTATION}, ${SPEC})&amp;quot;
&lt;/code&gt;&lt;/pre&gt;

&lt;p&gt;想法很简单，获取运行中的对象描述，使用 JQ 对数据进行清理和排序，输出一个相对标准的结果，便于不同环境间的比较和部署的导出。&lt;/p&gt;

&lt;blockquote&gt;
&lt;p&gt;虽然最后是通过 &lt;code&gt;kubectl std-json&lt;/code&gt; 的方式调用，这里的 &lt;code&gt;$0&lt;/code&gt; 指的仍然是脚本自身。&lt;/p&gt;
&lt;/blockquote&gt;

&lt;h2 id=&#34;制作清单和测试&#34;&gt;制作清单和测试&lt;/h2&gt;

&lt;p&gt;照猫画虎，按照上面的 YAML 代码，编写自己的清单。&lt;/p&gt;

&lt;p&gt;清单要求，需要打一个压缩包便于下载，我们把可执行文件和 LICENSE 文件放置到单独的目录 &lt;code&gt;kubectl-std-json-v0.1.0&lt;/code&gt; 中，压缩生成一个 &lt;code&gt;.tar.gz&lt;/code&gt; 文件，部分清单如下&lt;/p&gt;

&lt;pre&gt;&lt;code class=&#34;language-yaml&#34;&gt;    uri: https://github.com/fleeto/kubectl-std-json/releases/download/v0.1.0/kubectl-std-json-v0.1.0.tar.gz
    sha256: e1ad2398eaed5442042da134fb046fa8276042dd4122da4d872a8e91aeb2a339
    bin: kubectl-std-json
    files:
    - from: kubectl-std-json-*/kubectl-std-json
      to: .
    - from: kubectl-std-json-*/LICENSE
      to: .
&lt;/code&gt;&lt;/pre&gt;

&lt;p&gt;平台选择方面，我们只支持 OSX 和 Linux，因此只要一个平台元素即可。&lt;/p&gt;

&lt;p&gt;压缩包的校验码可以使用 &lt;code&gt;shasum -a 256&lt;/code&gt; 命令生成。&lt;/p&gt;

&lt;p&gt;上传压缩包之后，可以使用 &lt;code&gt;kubectl krew install --manifest&lt;/code&gt; 命令来测试安装。如果一切顺利，在本地就可以使用了。&lt;/p&gt;

&lt;h2 id=&#34;krew-index&#34;&gt;krew-index&lt;/h2&gt;

&lt;p&gt;接下来的操作很常规：fork krew-index，把你的清单写入 &lt;code&gt;plugins&lt;/code&gt; 目录，提交 PR 即可。&lt;/p&gt;

&lt;h2 id=&#34;相关链接&#34;&gt;相关链接&lt;/h2&gt;

&lt;ul&gt;
&lt;li&gt;Krew：&lt;code&gt;https://github.com/kubernetes-sigs/krew&lt;/code&gt;&lt;/li&gt;
&lt;li&gt;Krew index：&lt;code&gt;https://github.com/kubernetes-sigs/krew-index&lt;/code&gt;&lt;/li&gt;
&lt;/ul&gt;
</description>
    </item>
    
  </channel>
</rss>
