如何在 kubernetes 上使用 redis 部署 node.js?

mpb*_*mpb 7 redis node.js 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"]
Run Code Online (Sandbox Code Playgroud)

我用 docker build -t my-app .

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

apiVersion: v1
kind: Pod
metadata:
  name: my-app
spec:
  containers:
  - name: my-app
    image: my-app:latest
    imagePullPolicy: Never
    ports:
    - containerPort: 8080
Run Code Online (Sandbox Code Playgroud)

到现在为止还挺好。但是从现在开始,我不知道如何继续使用redis:

  1. redis 应该是另一个 Pod 还是服务(就 Kubernetes 而言)?

  2. 如何从我的应用程序内部引用 redis?根据redis是否会被定义为Pod/Service,如何获取连接URL和端口?我了解了 Kubernetes 创建的环境变量,但我不确定它们是否适用于 Pod 或服务。

  3. 如何在单一配置下聚合两者(我的应用程序和 redis)?如何确保首先启动 redis,然后是我的应用程序(需要运行 redis 实例),以及如何将我的 HTTP 端点公开给“外部世界”?我阅读了有关部署的信息,但我不确定如何将这些部分连接在一起。

理想情况下,我希望在 YAML 文件中包含所有配置,以便在一天结束时可以使用单个命令启动整个基础架构。

mpb*_*mpb 5

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

对于我的部署,我在一个 Pod 中使用了两个容器(webapp + redis),因为 webapp 在没有活动 redis 实例的情况下运行是没有意义的,而且它在应用程序启动时连接到 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: {}
Run Code Online (Sandbox Code Playgroud)

这是服务定义:

apiVersion: v1
kind: Service
metadata:
  name: my-app-service
spec:
  ports:
  - port: 8080
    protocol: TCP
  type: NodePort
  selector:
    app: my-app-deployment
Run Code Online (Sandbox Code Playgroud)

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


Shu*_*rma 4

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

首先,在文件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
Run Code Online (Sandbox Code Playgroud)

接下来将您的应用程序的清单放入另一个文件中,例如my-app.yaml. 这里我放置了volume字段,以便您可以使用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
Run Code Online (Sandbox Code Playgroud)

现在我们可以使用以下 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
Run Code Online (Sandbox Code Playgroud)

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

所以,现在一切都在你的手中。