<?xml version="1.0" encoding="utf-8" standalone="yes" ?>
<rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom">
  <channel>
    <title>kubectl | 伪架构师</title>
    <link>/tags/kubectl/</link>
      <atom:link href="/tags/kubectl/index.xml" rel="self" type="application/rss+xml" />
    <description>kubectl</description>
    <generator>Source Themes Academic (https://sourcethemes.com/academic/)</generator><language>zh</language><lastBuildDate>Thu, 02 Apr 2020 10:13:24 +0800</lastBuildDate>
    <image>
      <url>/img/logo-wide.png</url>
      <title>kubectl</title>
      <link>/tags/kubectl/</link>
    </image>
    
    <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>
    
    <item>
      <title>在 kubectl 中使用 Service Account Token</title>
      <link>/post/account-token-for-kubectl/</link>
      <pubDate>Thu, 11 May 2017 23:27:37 +0800</pubDate>
      <guid>/post/account-token-for-kubectl/</guid>
      <description>

&lt;p&gt;在运行基于 K8S 的 CI/CD 过程中，经常有需求在容器中对 Kubernetes 的资源进行操作，其中隐藏的安全问题，目前推荐的最佳实践也就是使用 Service Account 了。而调试账号能力的最好方法，必须是 kubectl 了。下面就讲讲如何利用 kubectl 引用 Servie Account 凭据进行 K8S 操作的方法。&lt;/p&gt;

&lt;p&gt;这里用 default Service Account 为例。&lt;/p&gt;

&lt;h2 id=&#34;假设&#34;&gt;假设&lt;/h2&gt;

&lt;p&gt;目前已经能对目标集群进行操作，文中需要的权限主要就是读取命名空间中的 Secret 和 Service Account。&lt;/p&gt;

&lt;h2 id=&#34;准备配置文件&#34;&gt;准备配置文件&lt;/h2&gt;

&lt;p&gt;新建一个 Yaml 文件，命名请随意，例如 kubectl.yaml。内容：&lt;/p&gt;

&lt;pre&gt;&lt;code class=&#34;language-yaml&#34;&gt;
apiVersion: v1
kind: Config
clusters:
- cluster:
    certificate-authority-data: {ca data}
    server: https://{server}
  name: awesome-cluster
users:
- user:
    token: {token}
  name: account
- context:
    cluster: awesome-cluster
    user: account
  name: sa
current-context: sa
&lt;/code&gt;&lt;/pre&gt;

&lt;p&gt;其中的 {ca data} 可以从现有连接凭据中获取。&lt;/p&gt;

&lt;ul&gt;
&lt;li&gt;&lt;p&gt;&lt;code&gt;{server}&lt;/code&gt;：服务器地址&lt;/p&gt;&lt;/li&gt;

&lt;li&gt;&lt;p&gt;&lt;code&gt;{token}&lt;/code&gt;：将在后面设置&lt;/p&gt;&lt;/li&gt;
&lt;/ul&gt;

&lt;h2 id=&#34;获取数据&#34;&gt;获取数据&lt;/h2&gt;

&lt;p&gt;首先查看 Service Account 的 Token 在哪里：&lt;/p&gt;

&lt;p&gt;&lt;code&gt;kubectl get serviceaccount default -o yaml&lt;/code&gt;&lt;/p&gt;

&lt;p&gt;返回内容如下：&lt;/p&gt;

&lt;pre&gt;&lt;code class=&#34;language-yaml&#34;&gt;apiVersion: v1
kind: ServiceAccount
metadata:
  creationTimestamp: 2017-05-07T10:41:50Z
  name: default
  namespace: default
  resourceVersion: &amp;quot;26&amp;quot;
  selfLink: /api/v1/namespaces/default/serviceaccounts/default
  uid: c715217d-3311-11e7-a4ae-42010a8c0095
secrets:
- name: default-token-7h4bd
&lt;/code&gt;&lt;/pre&gt;

&lt;p&gt;这里我们看到他包含了 secret: &amp;ldquo;default-token-7h4bd&amp;rdquo;，获取其中的内容：&lt;/p&gt;

&lt;p&gt;&lt;code&gt;kubectl get secret default-token-7h4bd -o yaml&lt;/code&gt;&lt;/p&gt;

&lt;pre&gt;&lt;code class=&#34;language-yaml&#34;&gt;apiVersion: v1
data:
  ca.crt: [ca data]
  namespace: ZGVmYXVsdA==
  token: [token data]
  kind: Secret
metadata:
  annotations:
    kubernetes.io/service-account.name: default
    kubernetes.io/service-account.uid: c715217d-3311-11e7-a4ae-42010a8c0095
  creationTimestamp: 2017-05-07T10:41:50Z
  name: default-token-7h4bd
  namespace: default
  resourceVersion: &amp;quot;24&amp;quot;
  selfLink: /api/v1/namespaces/default/secrets/default-token-7h4bd
  uid: c71cc72d-3311-11e7-a4ae-42010a8c0095
type: kubernetes.io/service-account-token
&lt;/code&gt;&lt;/pre&gt;

&lt;p&gt;上面 Token Data 内容就是我们需要的认证 Token 了&lt;/p&gt;

&lt;pre&gt;&lt;code class=&#34;language-bash&#34;&gt;export my_token=&amp;quot;[tokendata]&amp;quot;
kubectl --kubeconfig=kubectl.yaml \
config set-credentials account \
--token=`echo ${tokendata} | base64 -D`
&lt;/code&gt;&lt;/pre&gt;

&lt;p&gt;这样就把 Service Account 的 Token 取出来，并保存在 kubectl.yaml 中。利用这一配置文件就可以凭 Service Account 的身份来执行 kubectl 指令了。&lt;/p&gt;
</description>
    </item>
    
  </channel>
</rss>
