如何在kubernetes上部署带有Redis的node.js?

时间:2018-10-28 13:07:14

标签: node.js redis kubernetes minikube

我有一个非常简单的node.js应用程序(HTTP服务),可与Redis“对话”。我想创建一个部署并使用minikube运行它。

据我了解,我需要一个基于docker镜像的kubernetes Pod。这是我的Dockerfile:

FROM node:8.9.1
WORKDIR /usr/src/app
COPY package*.json ./
RUN npm install
COPY . .
EXPOSE 8080
CMD ["npm", "start"]

我使用docker build -t my-app .

构建docker映像

接下来,我为应用的Pod创建了Pod定义:

apiVersion: v1
kind: Pod
metadata:
  name: my-app
spec:
  containers:
  - name: my-app
    image: my-app:latest
    imagePullPolicy: Never
    ports:
    - containerPort: 8080

到目前为止,太好了。但是从现在开始,我还不清楚如何进行Redis:

  1. 应该重新使用另一个Pod还是服务(就Kubernetes而言)?

  2. 如何从应用程序内部引用Redis?根据是否将Redis定义为Pod /服务,如何获取连接URL和端口?我读到有关Kubernetes创建的环境变量的信息,但是我不确定这些变量是否适用于Pod或Services。

  3. 如何在单一配置下同时汇总两个(我的应用和Redis)?如何确保Redis首先启动,然后确保我的应用程序启动(需要运行Redis实例),以及如何将HTTP端点暴露给“外部世界”?我读过有关部署的文章,但不确定如何将这些部分连接在一起。

理想情况下,我想将所有配置都包含在YAML文件中,以便最终可以通过单个命令启动整个基础架构。

5 个答案:

答案 0 :(得分:3)

我想我想出了一个解决方案(使用部署和服务)。

对于我的部署,我在一个Pod中使用了两个容器(webapp + redis),因为在没有活动redis实例的情况下运行webapp没有意义,此外,它在应用程序启动时连接到redis。我在这种推理上可能是错误的,所以如果您有其他看法,请随时纠正我。

这是我的部署:

apiVersion: apps/v1
kind: Deployment
metadata:
  name: my-app-deployment
spec:
  selector:
    matchLabels:
      app: my-app-deployment
  template:
    metadata:
      labels:
        app: my-app-deployment
    spec:
      containers:
      - name: redis
        image: redis:latest
        ports:
        - containerPort: 6379
        volumeMounts:
        - mountPath: /srv/www
          name: redis-storage
      - name: my-app
        image: my-app:latest
        imagePullPolicy: Never
        ports:
        - containerPort: 8080
      volumes:
      - name: redis-storage
        emptyDir: {}

这是服务定义:

apiVersion: v1
kind: Service
metadata:
  name: my-app-service
spec:
  ports:
  - port: 8080
    protocol: TCP
  type: NodePort
  selector:
    app: my-app-deployment

我使用以下方法创建部署: kubectl create -f deployment.yaml 然后,我使用kubectl create -f service.yaml创建服务 我用minikube ip读取IP,并从kubectl describe service my-app-service的输出中提取端口。

答案 1 :(得分:1)

  1. 是的,您需要分别使用deployementservice进行Redis

  2. 使用kubernetes服务发现,应内置在KubeDNS,CoreDNS中

  3. 使用就绪和活跃度探针

是的,您可以编写一个大的yaml文件来描述所有部署和服务。然后:

kubectl apply -f yourfile.yml

或者您可以将yaml放在单独的文件中,然后执行:

 kubectl apply -f dir/

答案 2 :(得分:1)

我建议您进一步阅读k8s文档,但总的来说,请回答您在上面提出的问题:

  1. 是的,另一个Pod(具有相关配置)和其他服务取决于您的用例,请查看以下示例:https://kubernetes.io/docs/tutorials/configuration/configure-redis-using-configmap/
  2. 使用服务,请在此处了解更多信息:https://kubernetes.io/docs/concepts/services-networking/connect-applications-service/
  3. 有几种管理依赖项的方法-搜索部署依赖项,但是通常,您可以将它们附加到具有就绪端点的同一文件中,并使用服务进行公开-在项目符号2的链接中了解更多信息

答案 3 :(得分:1)

我将在一个单独的Pod中运行Redis(即:因此,如果您的Web应用崩溃,它不会关闭Redis服务器)。

这是您的Redis部署和服务:

deployment.yaml:

apiVersion: apps/v1
kind: Deployment
metadata:
  name: redis
spec:
  selector:
    matchLabels:
      app: redis
  replicas: 1
  template:
    metadata:
      labels:
        app: redis
    spec:
      volumes:
        - name: host-sys
          hostPath:
            path: /sys
      initContainers:
        - name: disable-thp
          image: redis:4.0-alpine
          volumeMounts:
            - name: host-sys
              mountPath: /host-sys
          command: ["sh", "-c", "echo never > /host-sys/kernel/mm/transparent_hugepage/enabled"]
      containers:
      - name: redis
        image: redis:4.0-alpine
        imagePullPolicy: IfNotPresent
        resources:
          requests:
            cpu: 350m
            memory: 1024Mi
        ports:
        - containerPort: 6379

service.yaml:

apiVersion: v1
kind: Service
metadata:
  name: redis
  labels:
    app: redis
spec:
  ports:
  - port: 6379
    name: redis
  selector:
    app: redis

由于我们已经公开了kubernetes Service,因此您可以通过主机名或它的“服务名”(redis)来访问redis实例。

您可以在https://github.com/mateothegreat/k8-byexamples-redis检出我的kubernetes redis存储库。如果您想使用更简单的方法,可以只运行make install

祝您好运,如果您仍然遇到麻烦,请伸出手!

答案 4 :(得分:1)

我同意之前的所有答案。我只是想通过执行单个命令使事情变得更简单。

首先,在文件redis.yaml中为Redis创建必要的清单,并进行服务以将其公开。

apiVersion: v1
kind: Service
metadata:
  name: redis
  labels:
    app: node-redis
spec:
  ports:
  - name: redis
    port: 6379
    targetPort: 6379
  type: NodePort
  selector:
    app: node-redis
---
apiVersion: apps/v1
kind: Deployment
metadata:
  name: redis
spec:
  selector:
    matchLabels:
      app: node-redis
  replicas: 1
  template:
    metadata:
      labels:
        app: node-redis
    spec:
      containers:
      - name: redis
        image: redis:latest
        imagePullPolicy: IfNotPresent
        ports:
        - containerPort: 6379
        # data volume where redis writes data
        volumeMounts:
        - name: data
          mountPath: /data
          readOnly: false
      volumes:
      - name: data
        persistentVolumeClaim:
          claimName: redis-data
---
# data volume
apiVersion: v1
kind: PersistentVolumeClaim
metadata:
  name: redis-data
  labels:
    app: node-redis
spec:
  accessModes:
  - ReadWriteOnce
  resources:
    requests:
      storage: 100Mi

接下来,将您应用的清单放入另一个文件my-app.yaml中。在这里,我放置了音量字段,以便您可以使用redis存储的数据。

apiVersion: v1
kind: Pod
metadata:
  name: my-app
  labels:
    app: node-redis
spec:
  containers:
  - name: my-app
    image: my-app:latest
    ports:
    - containerPort: 8080
    # data volume from where my-app read data those are written by redis
    volumeMounts:
    - name: data
      mountPath: /data
      readOnly: false
  volumes:
  - name: data
    persistentVolumeClaim:
      claimName: redis-data

现在我们可以使用以下bash文件my-app.sh

#!/bin/bash

kubectl create -f redis.yaml

pod_name=$(kubectl get po -l app=node-redis | grep app-with-redis | awk '{print $1}')

# check whether redis server is ready or not
while true; do
  pong=$(kubectl exec -it $pod_name -c redis redis-cli ping)
  if [[ "$pong" == *"PONG"* ]]; then
    echo ok;
    break
  fi
done

kubectl create -f my-app.yaml

只需运行chmod +x my-app.sh; ./my-app.sh进行部署。要获取网址,请运行minikube service redis --url。您可以类似地获取应用程序的URL。唯一的事情是您需要一个nodePort类型的服务,您的应用才能从集群外部访问它。

所以,现在一切都在您手中。