OpenWhisk๋ฅผ ์‚ฌ์šฉํ•œ ์„œ๋ฒ„๋ฆฌ์Šค ์ปดํ“จํŒ…, 4๋ถ€

OpenWhisk๋ฅผ ์‚ฌ์šฉํ•œ ์„œ๋ฒ„๋ฆฌ์Šค ์ปดํ“จํŒ…, 4๋ถ€

์ด ๊ธฐ์‚ฌ๋Š” ์ €์ž๊ฐ€ OpenWhisk์— ๋Œ€ํ•ด ๋ฒˆ์—ญํ•œ ์ผ๋ จ์˜ ๋…ธํŠธ๋ฅผ ๋งˆ๋ฌด๋ฆฌํ•ฉ๋‹ˆ๋‹ค. ํ”„๋ฆฌํ‹ฐ ๋ฐ์‚ฌ์ด. ์˜ค๋Š˜์€ ํ˜„์žฌ ๋ฒ„์ „์˜ ์• ํ”Œ๋ฆฌ์ผ€์ด์…˜๊ณผ ํ•จ๊ป˜ ์ž‘๋™ํ•˜๋„๋ก ์ˆ˜์ •๋œ ๋ช…๋ น์„ ์‚ฌ์šฉํ•˜์—ฌ Kubernetes๋ฅผ ํ†ตํ•ด OpenWhisk๋ฅผ ๋ฐฐํฌํ•˜๋Š” ํ”„๋กœ์„ธ์Šค๋ฅผ ์‚ดํŽด๋ณด๊ฒ ์Šต๋‹ˆ๋‹ค. ๋˜ํ•œ Nodejs ๋Ÿฐํƒ€์ž„์„ ์‚ฌ์šฉํ•˜๋Š” Kubernetes์—์„œ Knative ๋ฐ TektonCD๋ฅผ ์‚ฌ์šฉํ•˜์—ฌ OpenWhisk ๊ธฐ๋Šฅ์„ ์‹คํ–‰ํ•˜๋Š” ํ”„๋กœ์„ธ์Šค๋„ ๋‹ค๋ฃน๋‹ˆ๋‹ค.

Kubernetes์— OpenWhisk ๋ฐฐํฌ

๋ฉฐ์น ์— ๊ฑธ์ณ ๊ฐ„๋‹จํ•˜๊ณ  ๋น ๋ฅธ ํ…Œ์ŠคํŠธ ํ™˜๊ฒฝ์„ ๋งŒ๋“ค๊ธฐ ์œ„ํ•ด OpenWhisk๋ฅผ Kubernetes์— ๋ฐฐํฌํ•˜๋Š” ์‹คํ—˜์„ ํ–ˆ์Šต๋‹ˆ๋‹ค. ๊ทธ๋ฆฌ๊ณ  ์ €๋Š” Kubernetes๋ฅผ ์ฒ˜์Œ ์ ‘ํ•˜๊ธฐ ๋•Œ๋ฌธ์— ์„ฑ๊ณต์ ์ธ ๋ฐฐํฌ์— ํ•˜๋ฃจ ๋ฐ˜์ด ์†Œ์š”๋˜์—ˆ๋‹ค๊ณ  ์ƒ๊ฐํ•ฉ๋‹ˆ๋‹ค. ์•ˆ์— ์ด ๋ฆฌํฌ์ง€ํ† ๋ฆฌ์—๋Š” Kubernetes์— OpenWhisk๋ฅผ ๋ฐฐํฌํ•˜๊ธฐ ์œ„ํ•œ ๋งค์šฐ ๋ช…ํ™•ํ•œ ์ง€์นจ์ด ์žˆ์Šต๋‹ˆ๋‹ค. ๋‹ค์Œ์€ Mac์šฉ ๋ฐฐํฌ ์ง€์นจ์ž…๋‹ˆ๋‹ค(๋‚˜๋Š” ๋˜ํ•œ Linux๋ฅผ ์„ ํ˜ธํ•˜๊ธฐ ๋•Œ๋ฌธ์— ๋ชจ๋“  ๊ฒƒ์„ Linux์—์„œ ํ•  ๊ฒƒ์ž…๋‹ˆ๋‹ค. โ€” ๋Œ€๋žต. ์—ญ์ž).

  1. ํŒจํ‚ค์ง€ ๊ด€๋ฆฌ์ž ์„ค์น˜ asdf, ๊ทธ ํ›„์—๋Š” ์ž๋™์œผ๋กœ ์ˆ˜์ •๋ฉ๋‹ˆ๋‹ค. ~/.bash_profile ๋˜๋Š” ์ด์™€ ๋™๋“ฑํ•œ ๋‚ด์šฉ์€ ๋‹ค์Œ๊ณผ ๊ฐ™์Šต๋‹ˆ๋‹ค.

$ brew install asdf
$ [ -s "/usr/local/opt/asdf/asdf.sh" ] && . /usr/local/opt/asdf/asdf.sh
$ source ~/.bash_profile

[Linux์—์„œ๋Š” Brew๊ฐ€ ๊ฐ€๋Šฅํ•˜์ง€๋งŒ ์ด ๋‹จ๊ณ„๊ฐ€ ํ•„์š”ํ•˜์ง€ ์•Š์Šต๋‹ˆ๋‹ค. โ€” ๋Œ€๋žต. ์—ญ์ž]

  1. ํ”Œ๋Ÿฌ๊ทธ์ธ ์ถ”๊ฐ€ minikube ะธ kubelet:

$ asdf plugin-add kubectl
$ asdf plugin-add minikube

[๋‹ค์‹œ ํ•œ๋ฒˆ ๋งํ•˜์ง€๋งŒ, Linux์—์„œ๋Š” ์ด ๋‹จ๊ณ„๋ฅผ ๊ฑด๋„ˆ๋›ฐ์„ธ์š”. โ€” ๋Œ€๋žต. ์—ญ์ž]

  1. minikube์™€ kubelet์„ ์„ค์น˜ํ•ฉ๋‹ˆ๋‹ค:

$ asdf install kubectl 1.9.0
$ asdf global kubectl 1.9.0
$ asdf install minikube 0.25.2
$ asdf global minikube 0.25.2

[ํŠน์ • ๋ฒ„์ „์ด ์„ค์น˜๋˜์–ด ์žˆ์ง€๋งŒ Linux์— ์‚ฌ์šฉ ๊ฐ€๋Šฅํ•œ ์ตœ์‹  ๋ฒ„์ „์˜ ๋ชจ๋“  ํ•ญ๋ชฉ์„ ํ™•์ธํ–ˆ์Šต๋‹ˆ๋‹ค. ์ตœ์‹  ๋ฒ„์ „์„ ์•ˆ์ „ํ•˜๊ฒŒ ์„ค์น˜ํ•  ์ˆ˜ ์žˆ๋‹ค๊ณ  ์ƒ๊ฐํ•ฉ๋‹ˆ๋‹ค. โ€” ๋Œ€๋žต. ์—ญ์ž]

Linux์—์„œ ์ด ๋‹จ๊ณ„๋Š” ๋‹ค์Œ๊ณผ ๊ฐ™์ด ์ˆ˜ํ–‰๋ฉ๋‹ˆ๋‹ค(๋ชจ๋“  ๊ฒƒ์€ ๋ฒˆ์—ญ์ž์˜ ๋ฉ”๋ชจ์ธ ๋‚ด PATH์— ๋‚˜์—ด๋˜์–ด ์žˆ๋Š” ~/bin์— ์ €์žฅ๋ฉ๋‹ˆ๋‹ค).

$ curl -L0 minikube https://storage.googleapis.com/minikube/releases/latest/minikube-linux-amd64 && chmod +x minikube && mv minikube ~/bin/
$ curl -L0 https://storage.googleapis.com/kubernetes-release/release/$(curl -s https://storage.googleapis.com/kubernetes-release/release/stable.txt)/bin/linux/amd64/kubectl && chmod +x kubectl && mv kubectl ~/bin/

  1. Minikube ๊ฐ€์ƒ ๋จธ์‹ ์„ ์ƒ์„ฑํ•ฉ๋‹ˆ๋‹ค(VirtualBox๊ฐ€ ์‚ฌ์ „ ์„ค์น˜๋˜์–ด ์žˆ์–ด์•ผ ํ•จ):

$ minikube start --cpus 2 --memory 4096 --kubernetes-version=v1.9.0 --extra-config=apiserver.Authorization.Mode=RBAC

[ํŒ€๊ณผ ํ•จ๊ป˜๋ผ๋ฉด ๋ชจ๋“  ๊ฒƒ์ด ๋‚˜์—๊ฒŒ ์ž˜ ๋งž๋Š”๋‹ค minikube start , ๋งค๊ฐœ๋ณ€์ˆ˜ ์—†์ด ๊ธฐ๋ณธ๊ฐ’์„ ์‚ฌ์šฉํ•ฉ๋‹ˆ๋‹ค. โ€” ๋Œ€๋žต. ์—ญ์ž]

$ minikube start
  minikube v1.5.2 on Debian 8.11
  Automatically selected the 'virtualbox' driver
  Downloading VM boot image ...
    > minikube-v1.5.1.iso.sha256: 65 B / 65 B [--------------] 100.00% ? p/s 0s
    > minikube-v1.5.1.iso: 143.76 MiB / 143.76 MiB [-] 100.00% 5.63 MiB p/s 26s
  Creating virtualbox VM (CPUs=2, Memory=4096MB, Disk=20000MB) ...
  Preparing Kubernetes v1.16.2 on Docker '18.09.9' ...
  Downloading kubelet v1.16.2
  Downloading kubeadm v1.16.2
  Pulling images ...
  Launching Kubernetes ...  Waiting for: apiserver
  Done! kubectl is now configured to use "minikube"

  1. Docker์˜ ๋„คํŠธ์›Œํฌ๋ฅผ ๋ฌด์ฐจ๋ณ„ ๋ชจ๋“œ๋กœ ์ „ํ™˜:

$ minikube ssh -- sudo ip link set docker0 promisc on

  1. ๋„ค์ž„์ŠคํŽ˜์ด์Šค๋ฅผ ๋งŒ๋“ค๊ณ  ์ž‘์—…์ž ๋…ธ๋“œ๋ฅผ ํ‘œ์‹œํ•ฉ๋‹ˆ๋‹ค.

$ kubectl create namespace openwhisk
$ kubectl label nodes --all openwhisk-role=invoker

  1. ์ €์žฅ์†Œ์˜ ์ฝ˜ํ…์ธ ๋ฅผ ๊ฐ€์ ธ์˜ค๊ณ  mycluster.yaml ํŒŒ์ผ์—์„œ ์ˆ˜์‹  ์œ ํ˜•์„ ์žฌ์ •์˜ํ•ฉ๋‹ˆ๋‹ค.

$ git clone https://github.com/apache/incubator-openwhisk-deploy-kube.git
$ cd incubator-openwhisk-deploy-kube/
$ cat << "EOF" > mycluster.yaml
whisk:
    ingress:
        type: NodePort
            api_host_name: 192.168.99.100
            api_host_port: 31001
nginx:
    httpsNodePort: 31001
EOF

  1. Helm์„ ์„ค์น˜ํ•˜๊ณ  ์ด๋ฅผ ์‚ฌ์šฉํ•˜์—ฌ ๋ฐฐํฌํ•ฉ๋‹ˆ๋‹ค.

$ brew install kubernetes-helm
$ helm init # init Helm Tiller, ะฝะต ะฝัƒะถะฝะพ ะฝะฐ Helm v3+
$ kubectl get pods -n kube-system # verify that tiller-deploy is in the running state, ะฝะต ะฝัƒะถะฝะพ ะฝะฐ helm v3+
$ kubectl create clusterrolebinding tiller-cluster-admin --clusterrole=cluster-admin --serviceaccount=kube-system:default
$ helm install ./openwhisk/helm/ --namespace=openwhisk -f mycluster.yaml

[์ตœ์‹  ๋ฒ„์ „(v3.0.1 ์‚ฌ์šฉ ๊ฐ€๋Šฅ)์ด ์„ค์น˜๋œ Linux์—์„œ๋Š” ์•ฝ๊ฐ„ ๋‹ค๋ฆ…๋‹ˆ๋‹ค. โ€” ๋Œ€๋žต. ์—ญ์ž]

$ curl -L0 https://get.helm.sh/helm-v3.0.1-linux-amd64.tar.gz | tar -xzvf - linux-amd64/helm --strip-components=1; sudo mv helm /usr/local/bin
$ kubectl create clusterrolebinding tiller-cluster-admin --clusterrole=cluster-admin --serviceaccount=kube-system:default
$ helm install ./openwhisk/helm/ --namespace=openwhisk --generate-name -f mycluster.yaml

  1. ๋ชจ๋“  ๊ฒƒ์ด ์ƒ์Šนํ–ˆ๋Š”์ง€ ํ™•์ธํ•ฉ๋‹ˆ๋‹ค(STATUS = ์‹คํ–‰ ์ค‘ ๋˜๋Š” ์™„๋ฃŒ๋จ).

$ kubectl get pods -n openwhisk
NAME                                                              READY   STATUS      RESTARTS   AGE
openwhisk-1576070780-alarmprovider-6868dc694-plvpf                1/1     Running     1          1d5h
openwhisk-1576070780-apigateway-8d56f4979-825hf                   1/1     Running     1          1d5h
openwhisk-1576070780-cloudantprovider-544bb46596-9scph            1/1     Running     1          1d5h
openwhisk-1576070780-controller-0                                 1/1     Running     2          1d5h
openwhisk-1576070780-couchdb-7fd7f6c7cc-42tw6                     1/1     Running     1          1d5h
openwhisk-1576070780-gen-certs-z9nsb                              0/1     Completed   0          1d5h
openwhisk-1576070780-init-couchdb-r2vmt                           0/1     Completed   0          1d5h
openwhisk-1576070780-install-packages-27dtr                       0/1     Completed   0          1d4h
openwhisk-1576070780-invoker-0                                    1/1     Running     1          1d5h
openwhisk-1576070780-kafka-0                                      1/1     Running     1          1d5h
openwhisk-1576070780-kafkaprovider-f8b4cf4fc-7z4gt                1/1     Running     1          1d5h
openwhisk-1576070780-nginx-6dbdbf69bc-5x76n                       1/1     Running     1          1d5h
openwhisk-1576070780-redis-cfd8756f4-hkrt6                        1/1     Running     1          1d5h
openwhisk-1576070780-wskadmin                                     1/1     Running     1          1d5h
openwhisk-1576070780-zookeeper-0                                  1/1     Running     1          1d5h
wskopenwhisk-1576070780-invoker-00-1-prewarm-nodejs10             1/1     Running     0          61s
wskopenwhisk-1576070780-invoker-00-2-prewarm-nodejs10             1/1     Running     0          61s
wskopenwhisk-1576070780-invoker-00-3-whisksystem-invokerhealtht   1/1     Running     0          59s

  1. wsk๊ฐ€ ์ž‘๋™ํ•˜๋„๋ก ๊ตฌ์„ฑ:

$ wsk property set --apihost 192.168.99.100:31001
$ wsk property set --auth 23bc46b1-71f6-4ed5-8c54-816aa4f8c502:123zO3xZCLrMN6v2BKK1dXYFpXlPkccOFqm12CdAsMgRU4VrNZ9lyGVCGuMDGIwP

์šฐ๋ฆฌ๋Š” ๋‹ค์Œ์„ ํ™•์ธํ•ฉ๋‹ˆ๋‹ค.

$ wsk -i list
Entities in namespace: default
packages
actions
triggers
rules

๋ฌธ์ œ์™€ ํ•ด๊ฒฐ์ฑ…

getsockopt: ์—ฐ๊ฒฐ์ด ๊ฑฐ๋ถ€๋˜์—ˆ์Šต๋‹ˆ๋‹ค.

$ wsk -i list
error: Unable to obtain the list of entities for namespace 'default': Get http://192.168.99.100:31001/api/v1/namespaces/_/actions?limit=0&skip=0: dial tcp 192.168.99.100:31001: getsockopt: connection refused

์ปจํ…Œ์ด๋„ˆ๊ฐ€ ๋„ค์ž„์ŠคํŽ˜์ด์Šค์— ์žˆ๋Š”์ง€ ํ™•์ธ openwhisk ์ƒํƒœ Running, ์™œ๋ƒํ•˜๋ฉด ๋•Œ๋กœ๋Š” ์˜ค๋ฅ˜๋กœ ์ธํ•ด ์ถฉ๋Œ์ด ๋ฐœ์ƒํ•ฉ๋‹ˆ๋‹ค. CreateContainerConfigError.

ํ˜ธ์ถœ์ž๊ฐ€ ์•„์ง ์ดˆ๊ธฐํ™” ์ค‘์ž…๋‹ˆ๋‹ค โ€” Init:1/2

๋‹ค์–‘ํ•œ ๋Ÿฐํƒ€์ž„ ํ™˜๊ฒฝ์„ ๋‹ค์šด๋กœ๋“œํ•˜๋Š” ํ”„๋กœ์„ธ์Šค์—๋Š” ์‹œ๊ฐ„์ด ์˜ค๋ž˜ ๊ฑธ๋ฆด ์ˆ˜ ์žˆ์Šต๋‹ˆ๋‹ค. ์†๋„๋ฅผ ๋†’์ด๋ ค๋ฉด ํŒŒ์ผ์— ๋‹จ์ถ•๋œ ์ตœ์†Œ ๋ชฉ๋ก์„ ์ง€์ •ํ•  ์ˆ˜ ์žˆ์Šต๋‹ˆ๋‹ค. mycluster.yaml:

whisk:
  runtimes: "runtimes-minimal-travis.json"

์ด๋ฆ„์ด ์žˆ๋Š” ์ปจํ…Œ์ด๋„ˆ -์„ค์น˜-ํŒจํ‚ค์ง€- ์˜ค๋ฅ˜๋กœ ์ธํ•ด ์ถฉ๋Œ

ํ™œ์„ฑ ํ…Œ์ŠคํŠธ์˜ ์‹œ๊ฐ„ ์ œํ•œ์„ ๋Š˜๋ฆฌ๋ฉด ๋ฉ๋‹ˆ๋‹ค.

Knative๋ฅผ ํ†ตํ•ด OpenWhisk ์„ค์น˜

Priti Desai๋Š” Knative Build ๋ฐ BuildTemplates๋ฅผ ์‚ฌ์šฉํ•˜์—ฌ IBM ํด๋ผ์šฐ๋“œ์˜ ํด๋Ÿฌ์Šคํ„ฐ์™€ ์ผ๋ฐ˜ Minikube์— ์„ค์น˜๋ฅผ ์ˆ˜ํ–‰ํ–ˆ์Šต๋‹ˆ๋‹ค. ๋ฐฉ๋ฒ•์— ๋”ฐ๋ผ minukube ์œ„์—๋„ ์„ค์น˜ํ•˜๊ฒ ์Šต๋‹ˆ๋‹ค. ์„ค๋ช…๋๋‹ค ์ด์ „ ๋ธ”๋กœ๊ทธ์—์„œ - ์ตœ์‹  ์†Œํ”„ํŠธ์›จ์–ด ๋ฒ„์ „์„ ์‚ฌ์šฉํ•ฉ๋‹ˆ๋‹ค. Knative Build ๋ฐ BuildTemplates๋Š” ๊ณต์‹์ ์œผ๋กœ ๋” ์ด์ƒ ์‚ฌ์šฉ๋˜์ง€ ์•Š์œผ๋ฏ€๋กœ Tekton Pipelines ํ˜•์‹์˜ ๊ถŒ์žฅ ๋Œ€์ฒด ๋ฒ„์ „์„ ์‚ฌ์šฉํ•˜๊ฒ ์Šต๋‹ˆ๋‹ค. ๊ธฐ์‚ฌ์˜ ๋‚˜๋จธ์ง€ ๋ถ€๋ถ„์€ Tekton Pipelines์˜ ๋ฌธ์„œ๋ฅผ ์ฝ์€ ํ›„ ์ž‘์„ฑ๋˜์—ˆ์ง€๋งŒ Priti์˜ ์•„์ด๋””์–ด๋ฅผ ๊ธฐ๋ฐ˜์œผ๋กœ ํ•ฉ๋‹ˆ๋‹ค. ์ž‘์—…ํ•˜๋ ค๋ฉด ์ผ๋ถ€ Docker ๋ ˆ์ง€์ŠคํŠธ๋ฆฌ์— ์•ก์„ธ์Šคํ•ด์•ผ ํ•ฉ๋‹ˆ๋‹ค. ์ €๋Š” ์›์ €์ž์™€ ๋งˆ์ฐฌ๊ฐ€์ง€๋กœ DockerHub๋ฅผ ์‚ฌ์šฉํ•ฉ๋‹ˆ๋‹ค.

$ curl -L0 https://github.com/solo-io/gloo/releases/download/v1.2.10/glooctl-linux-amd64; chmod +x glooctl-linux-amd64; mv glooctl-linux-amd64 ~/bin
$ glooctl install knative
$ kubectl get pods -n knative-serving
NAME                              READY   STATUS    RESTARTS   AGE
activator-77fc555665-rvrst        1/1     Running   0          2m23s
autoscaler-5c98b7c9b6-x8hh4       1/1     Running   0          2m21s
autoscaler-hpa-5cfd4f6845-w87kq   1/1     Running   0          2m22s
controller-7fd74c8f67-tprm8       1/1     Running   0          2m19s
webhook-74847bb77c-txr2g          1/1     Running   0          2m17s
$ kubectl get pods -n gloo-system
NAME                                      READY   STATUS    RESTARTS   AGE
discovery-859d7fbc9c-8xhvh                1/1     Running   0          51s
gloo-545886d9c6-85mwt                     1/1     Running   0          51s
ingress-67d4996d75-lkkmw                  1/1     Running   0          50s
knative-external-proxy-767dfd656c-wwv2z   1/1     Running   0          50s
knative-internal-proxy-6fdddcc6b5-7vqd8   1/1     Running   0          51s

OpenWhisk๋ฅผ ์‚ฌ์šฉํ•œ ์„œ๋ฒ„๋ฆฌ์Šค ์ปดํ“จํŒ…, 4๋ถ€
Knative ์œ„์— OpenWhisk ๋นŒ๋“œ ๋ฐ ์‹คํ–‰

  1. ๋‚ด์šฉ ๊ฐ€์ ธ์˜ค๊ธฐ ์ด ์ €์žฅ์†Œ:

$ git clone https://github.com/tektoncd/catalog/
$ cd catalog/openwhisk

  1. ๋ ˆ์ง€์ŠคํŠธ๋ฆฌ์— ์•ก์„ธ์Šคํ•˜๊ธฐ ์œ„ํ•œ ๋ฐ์ดํ„ฐ๋ฅผ ํ™˜๊ฒฝ ๋ณ€์ˆ˜๋กœ ์„ค์ •ํ•˜๊ณ  ์ด๋ฅผ Kubernetes ๋น„๋ฐ€๋กœ ์ €์žฅํ•ฉ๋‹ˆ๋‹ค.

$ export DOCKER_USERNAME=<your docker hub username>
$ export DOCKER_PASSWORD=<your docker hub password>
$ sed -e 's/${DOCKER_USERNAME}/'"$DOCKER_USERNAME"'/' -e 's/${DOCKER_PASSWORD}/'"$DOCKER_PASSWORD"'/' docker-secret.yaml.tmpl > docker-secret.yaml
$ kubectl apply -f docker-secret.yaml

์šฐ๋ฆฌ๋Š” ๋‹ค์Œ์„ ํ™•์ธํ•ฉ๋‹ˆ๋‹ค.

$ kubectl get secret
NAME                    TYPE                                  DATA      AGE
dockerhub-user-pass     kubernetes.io/basic-auth              2         21s

  1. ํ™˜๊ฒฝ ๊ตฌ์ถ•์„ ์œ„ํ•œ ๊ณ„์ •์„ ๋งŒ๋“ญ๋‹ˆ๋‹ค.

$ kubectl apply -f service-account.yaml

์šฐ๋ฆฌ๋Š” ๋‹ค์Œ์„ ํ™•์ธํ•ฉ๋‹ˆ๋‹ค.

$ kubectl get serviceaccount/openwhisk-runtime-builder
NAME                        SECRETS   AGE
openwhisk-runtime-builder   2         31m

  1. OpenWhisk์šฉ ์ด๋ฏธ์ง€๋ฅผ ๋นŒ๋“œํ•˜๋Š” ์ž‘์—… ๋งŒ๋“ค๊ธฐ

$ kubectl apply -f openwhisk.yaml
task.tekton.dev/openwhisk created

  1. ์ด๋ฏธ์ง€๋ฅผ ๋นŒ๋“œํ•˜๋Š” ์ž‘์—…์„ ์‹คํ–‰ํ•ฉ๋‹ˆ๋‹ค(์˜ˆ: NodeJS ์‚ฌ์šฉ).

๋‹ค์Œ ๋‚ด์šฉ์œผ๋กœ taskrun.yaml ํŒŒ์ผ์„ ์ƒ์„ฑํ•ฉ๋‹ˆ๋‹ค.

# Git Pipeline Resource for OpenWhisk NodeJS Runtime
apiVersion: tekton.dev/v1alpha1
kind: PipelineResource
metadata:
    name: openwhisk-nodejs-runtime-git
spec:
    type: git
    params:
        - name: revision
          value: master
        - name: url
          value: https://github.com/apache/openwhisk-runtime-nodejs.git
---

# Image Pipeline Resource for OpenWhisk NodeJS Sample Application
apiVersion: tekton.dev/v1alpha1
kind: PipelineResource
metadata:
    name: openwhisk-nodejs-helloworld-image
spec:
    type: image
    params:
        - name: url
          value: docker.io/${DOCKER_USERNAME}/openwhisk-nodejs-helloworld
---

# Task Run to build NodeJS image with the action source
apiVersion: tekton.dev/v1alpha1
kind: TaskRun
metadata:
    name: openwhisk-nodejs-helloworld
spec:
    serviceAccountName: openwhisk-runtime-builder
    taskRef:
        name: openwhisk
    inputs:
        resources:
            - name: runtime-git
              resourceRef:
                name: openwhisk-nodejs-runtime-git
        params:
            - name: DOCKERFILE
              value: "./runtime-git/core/nodejs10Action/knative/Dockerfile"
            - name: OW_ACTION_NAME
              value: "nodejs-helloworld"
            - name: OW_ACTION_CODE
              value: "function main() {return {payload: 'Hello World!'};}"
            - name: OW_PROJECT_URL
              value: ""
    outputs:
        resources:
            - name: runtime-image
              resourceRef:
                name: openwhisk-nodejs-helloworld-image
---

์ด ํŒŒ์ผ์— ํ˜„์žฌ ๋ฐ์ดํ„ฐ๋ฅผ ์ ์šฉํ•ฉ๋‹ˆ๋‹ค.

$ sed 's/${DOCKER_USERNAME}/'"$DOCKER_USERNAME"'/' -i taskrun.yaml

์šฐ๋ฆฌ๋Š” ๋‹ค์Œ์„ ์ ์šฉํ•ฉ๋‹ˆ๋‹ค:

$ kubectl apply -f taskrun.yaml
pipelineresource.tekton.dev/openwhisk-nodejs-runtime-git created
pipelineresource.tekton.dev/openwhisk-nodejs-helloworld-image created
taskrun.tekton.dev/openwhisk-nodejs-helloworld created

์ž‘์—… ํ™•์ธ์€ ํฌ๋“œ ์ด๋ฆ„ ๊ฐ€์ ธ์˜ค๊ธฐ ๋ฐ ์ƒํƒœ ๋ณด๊ธฐ๋กœ ๊ตฌ์„ฑ๋ฉ๋‹ˆ๋‹ค. ๋˜ํ•œ ๊ฐ ๋‹จ๊ณ„์˜ ์‹คํ–‰ ๋กœ๊ทธ๋ฅผ ๋ณผ ์ˆ˜๋„ ์žˆ์Šต๋‹ˆ๋‹ค. ์˜ˆ๋ฅผ ๋“ค๋ฉด ๋‹ค์Œ๊ณผ ๊ฐ™์Šต๋‹ˆ๋‹ค.

$ kubectl get taskrun
NAME                          SUCCEEDED   REASON      STARTTIME   COMPLETIONTIME
openwhisk-nodejs-helloworld   True        Succeeded   5m15s       44s
$ kubectl get pod openwhisk-nodejs-helloworld-pod-4640d3
NAME                                     READY   STATUS      RESTARTS   AGE
openwhisk-nodejs-helloworld-pod-4640d3   0/6     Completed   0          5m20s
$ kubectl logs openwhisk-nodejs-helloworld-pod-4640d3 -c step-git-source-openwhisk-nodejs-runtime-git-r8vhr
{"level":"info","ts":1576532931.5880227,"logger":"fallback-logger","caller":"logging/config.go:69","msg":"Fetch GitHub commit ID from kodata failed: open /var/run/ko/refs/heads/master: no such file or directory"}
{"level":"info","ts":1576532936.538926,"logger":"fallback-logger","caller":"git/git.go:81","msg":"Successfully cloned https://github.com/apache/openwhisk-runtime-nodejs.git @ master in path /workspace/runtime-git"}
{"level":"warn","ts":1576532936.5395331,"logger":"fallback-logger","caller":"git/git.go:128","msg":"Unexpected error: creating symlink: symlink /tekton/home/.ssh /root/.ssh: file exists"}
{"level":"info","ts":1576532936.8202565,"logger":"fallback-logger","caller":"git/git.go:109","msg":"Successfully initialized and updated submodules in path /workspace/runtime-git"}

์‹คํ–‰ ํ›„์—๋Š” Knative ์„œ๋น„์Šค์™€ ํ•จ๊ป˜ ์ž‘๋™ํ•˜๋„๋ก ์„ค๊ณ„๋œ kn ์œ ํ‹ธ๋ฆฌํ‹ฐ๋ฅผ ์‚ฌ์šฉํ•˜์—ฌ ๋ฐฐํฌํ•  ์ˆ˜ ์žˆ๋Š” ์ด๋ฏธ์ง€๊ฐ€ ๋ ˆ์ง€์ŠคํŠธ๋ฆฌ์— ์ƒ์„ฑ๋ฉ๋‹ˆ๋‹ค. ์˜ˆ๋ฅผ ๋“ค๋ฉด ๋‹ค์Œ๊ณผ ๊ฐ™์Šต๋‹ˆ๋‹ค.

kn service create nodejs-helloworld --image docker.io/${DOCKER_USERNAME}/openwhisk-nodejs-helloworld
Service 'nodejs-helloworld' successfully created in namespace 'default'.
Waiting for service 'nodejs-helloworld' to become ready ... OK

Service URL:
http://nodejs-helloworld.default.example.com

Gloo๋ฅผ ์‚ฌ์šฉํ•˜๋Š” ๊ฒฝ์šฐ ๋‹ค์Œ๊ณผ ๊ฐ™์ด ๊ธฐ๋Šฅ์„ ํ™•์ธํ•  ์ˆ˜ ์žˆ์Šต๋‹ˆ๋‹ค.

$ curl -H "Host: nodejs-helloworld.default.example.com" -X POST $(glooctl proxy url --name knative-external-proxy)
{"OK":true}
$ curl -H "Host: nodejs-helloworld.default.example.com" -X POST $(glooctl proxy url --name knative-external-proxy)
{"payload":"Hello World!"}

์‹œ๋ฆฌ์ฆˆ์˜ ๋‹ค๋ฅธ ๊ธฐ์‚ฌ

OpenWhisk๋ฅผ ์‚ฌ์šฉํ•œ ์„œ๋ฒ„๋ฆฌ์Šค ์ปดํ“จํŒ…, 1๋ถ€
OpenWhisk๋ฅผ ์‚ฌ์šฉํ•œ ์„œ๋ฒ„๋ฆฌ์Šค ์ปดํ“จํŒ…, 2๋ถ€
OpenWhisk๋ฅผ ์‚ฌ์šฉํ•œ ์„œ๋ฒ„๋ฆฌ์Šค ์ปดํ“จํŒ…, 3๋ถ€
OpenWhisk๋ฅผ ์‚ฌ์šฉํ•œ ์„œ๋ฒ„๋ฆฌ์Šค ์ปดํ“จํŒ…, 4๋ถ€

์ถœ์ฒ˜ : habr.com

์ฝ”๋ฉ˜ํŠธ๋ฅผ ์ถ”๊ฐ€