服务目录

服务目录(Service Catalog)是 服务目录是一种扩展 API,它能让 Kubernetes 集群中运行的应用易于使用外部托管的的软件服务,例如云供应商提供的数据仓库服务。

服务目录可以检索、供应、和绑定由 服务代理人(Service Brokers) 提供的外部托管服务(Managed Services), 而无需知道那些服务具体是怎样创建和托管的。

服务代理(Service Broker)是由Open Service Broker API 规范定义的一组托管服务的端点,这些服务由第三方提供并维护,其中的第三方可以是 AWS、GCP 或 Azure 等云服务提供商。 托管服务的一些示例是 Microsoft Azure Cloud Queue、Amazon Simple Queue Service 和 Google Cloud Pub/Sub,但它们可以是应用程序能够使用的任何软件交付物。

使用服务目录,集群操作员 可以浏览某服务代理所提供的托管服务列表,供应托管服务实例并与之绑定, 以使其可以被 Kubernetes 集群中的应用程序使用。

示例用例

应用开发人员, 希望使用消息队列,作为其在 Kubernetes 集群中运行的应用程序的一部分。 但是,他们不想承受构造这种服务的开销,也不想自行管理。 幸运的是,有一家云服务提供商通过其服务代理以托管服务的形式提供消息队列服务。

集群操作员可以设置服务目录并使用它与云服务提供商的服务代理通信,进而部署消息队列服务的实例 并使其对 Kubernetes 中的应用程序可用。 应用开发者于是可以不关心消息队列的实现细节,也不用对其进行管理。 他们的应用程序可以简单的将其作为服务使用。

架构

服务目录使用Open Service Broker API 与服务代理进行通信,并作为 Kubernetes API 服务器的中介,以便协商启动部署和获取 应用程序使用托管服务时必须的凭据。

服务目录实现为一个扩展 API 服务器和一个控制器,使用 Etcd 提供存储。 它还使用了 Kubernetes 1.7 之后版本中提供的 聚合层 来呈现其 API。

服务目录架构

API 资源

服务目录安装 servicecatalog.k8s.io API 并提供以下 Kubernetes 资源:

  • ClusterServiceBroker:服务目录的集群内表现形式,封装了其服务连接细节。集群运维人员创建和管理这些资源,并希望使用该代理服务在集群中提供新类型的托管服务。
  • ClusterServiceClass:由特定服务代理提供的托管服务。当新的 ClusterServiceBroker 资源被添加到集群时,服务目录控制器将连接到服务代理以获取可用的托管服务列表。然后为每个托管服务创建对应的新 ClusterServiceClass 资源。
  • ClusterServicePlan:托管服务的特定产品。例如托管服务可能有不同的计划可用,如免费版本和付费版本,或者可能有不同的配置选项,例如使用 SSD 存储或拥有更多资源。与 ClusterServiceClass 类似,当一个新的 ClusterServiceBroker 被添加到集群时,服务目录会为每个托管服务的每个可用服务计划创建对应的新 ClusterServicePlan 资源。
  • ServiceInstanceClusterServiceClass 提供的示例。由集群运维人员创建,以使托管服务的特定实例可供一个或多个集群内应用程序使用。当创建一个新的 ServiceInstance 资源时,服务目录控制器将连接到相应的服务代理并指示它调配服务实例。
  • ServiceBindingServiceInstance 的访问凭据。由希望其应用程序使用服务 ServiceInstance 的集群运维人员创建。创建之后,服务目录控制器将创建一个 Kubernetes Secret,其中包含服务实例的连接细节和凭据,可以挂载到 Pod 中。

认证

服务目录支持这些认证方法:

使用方式

集群运维人员可以使用服务目录 API 资源来供应托管服务并使其在 Kubernetes 集群内可用。涉及的步骤有:

  1. 列出服务代理提供的托管服务和服务计划。
  2. 配置托管服务的新实例。
  3. 绑定到托管服务,它将返回连接凭证。
  4. 将连接凭证映射到应用程序中。

列出托管服务和服务计划

首先,集群运维人员在 servicecatalog.k8s.io 组内创建一个 ClusterServiceBroker 资源。此资源包含访问服务代理终结点所需的 URL 和连接详细信息。

这是一个 ClusterServiceBroker 资源的例子:

  1. apiVersion: servicecatalog.k8s.io/v1beta1
  2. kind: ClusterServiceBroker
  3. metadata:
  4. name: cloud-broker
  5. spec:
  6. # 指向服务代理的末端。(这里的 URL 是无法使用的)
  7. url: https://servicebroker.somecloudprovider.com/v1alpha1/projects/service-catalog/brokers/default
  8. #####
  9. # 这里可以添加额外的用来与服务代理通信的属性值,
  10. # 例如持有者令牌信息或者 TLS 的 CA 包
  11. #####

下面的时序图展示了从服务代理列出可用托管服务和计划所涉及的各个步骤:

列举服务

  1. 一旦 ClusterServiceBroker 资源被添加到了服务目录之后,将会触发一个到外部服务代理的 调用,以列举所有可用服务;
  2. 服务代理返回可用的托管服务和服务计划列表,这些列表将本地缓存在 ClusterServiceClassClusterServicePlan 资源中。
  3. 集群运维人员接下来可以使用以下命令获取可用托管服务的列表:
  1. kubectl get clusterserviceclasses \
  2. -o=custom-columns=SERVICE\ NAME:.metadata.name,EXTERNAL\ NAME:.spec.externalName

它应该输出一个和以下格式类似的服务名称列表:

  1. SERVICE NAME EXTERNAL NAME
  2. 4f6e6cf6-ffdd-425f-a2c7-3c9258ad2468 cloud-provider-service
  3. ... ...

他们还可以使用以下命令查看可用的服务计划:

  1. kubectl get clusterserviceplans \
  2. -o=custom-columns=PLAN\ NAME:.metadata.name,EXTERNAL\ NAME:.spec.externalName
  1. 它应该输出一个和以下格式类似的服务计划列表:
  1. PLAN NAME EXTERNAL NAME
  2. 86064792-7ea2-467b-af93-ac9694d96d52 service-plan-name
  3. ... ...

供应一个新实例

集群运维人员 可以通过创建一个 ServiceInstance 资源来启动一个新实例的配置。

下面是一个 ServiceInstance 资源的例子:

  1. apiVersion: servicecatalog.k8s.io/v1beta1
  2. kind: ServiceInstance
  3. metadata:
  4. name: cloud-queue-instance
  5. namespace: cloud-apps
  6. spec:
  7. # 引用之前返回的服务之一
  8. clusterServiceClassExternalName: cloud-provider-service
  9. clusterServicePlanExternalName: service-plan-name
  10. #####
  11. # 这里可添加额外的参数,供服务代理使用
  12. #####

以下时序图展示了配置托管服务新实例所涉及的步骤:

供应服务

  1. 创建 ServiceInstance 资源时,服务目录将启动一个到外部服务代理的调用, 请求供应一个实例。
  2. 服务代理创建一个托管服务的新实例并返回 HTTP 响应。
  3. 接下来,集群运维人员可以检查实例的状态是否就绪。

绑定到托管服务

在设置新实例之后,集群运维人员必须绑定到托管服务才能获取应用程序使用服务所需的连接凭据和服务账户的详细信息。该操作通过创建一个 ServiceBinding 资源完成。

以下是 ServiceBinding 资源的示例:

  1. apiVersion: servicecatalog.k8s.io/v1beta1
  2. kind: ServiceBinding
  3. metadata:
  4. name: cloud-queue-binding
  5. namespace: cloud-apps
  6. spec:
  7. instanceRef:
  8. name: cloud-queue-instance
  9. #####
  10. # 这里可以添加供服务代理使用的额外信息,例如 Secret 名称或者服务账号参数,
  11. #####

以下顺序图展示了绑定到托管服务实例的步骤:

绑定到托管服务

  1. 在创建 ServiceBinding 之后,服务目录调用外部服务代理,请求绑定服务实例所需的信息。
  2. 服务代理为相应服务账户启用应用权限/角色。
  3. 服务代理返回连接和访问托管服务示例所需的信息。这是由提供商和服务特定的,故返回的信息可能因服务提供商和其托管服务而有所不同。

映射连接凭据

完成绑定之后的最后一步就是将连接凭据和服务特定的信息映射到应用程序中。这些信息存储在 secret 中,集群中的应用程序可以访问并使用它们直接与托管服务进行连接。

映射连接凭据

Pod 配置文件

执行此映射的一种方法是使用声明式 Pod 配置。

以下示例描述了如何将服务账户凭据映射到应用程序中。名为 sa-key 的密钥保存在一个名为 provider-cloud-key 的卷中,应用程序会将该卷挂载在 /var/secrets/provider/key.json 路径下。环境变量 PROVIDER_APPLICATION_CREDENTIALS 将映射为挂载文件的路径。

  1. ...
  2. spec:
  3. volumes:
  4. - name: provider-cloud-key
  5. secret:
  6. secretName: sa-key
  7. containers:
  8. ...
  9. volumeMounts:
  10. - name: provider-cloud-key
  11. mountPath: /var/secrets/provider
  12. env:
  13. - name: PROVIDER_APPLICATION_CREDENTIALS
  14. value: "/var/secrets/provider/key.json"

以下示例描述了如何将 Secret 值映射为应用程序的环境变量。 在这个示例中,消息队列的主题名从 Secret provider-queue-credentials 中名为 topic 的主键映射到环境变量 TOPIC 中。

  1. ...
  2. env:
  3. - name: "TOPIC"
  4. valueFrom:
  5. secretKeyRef:
  6. name: provider-queue-credentials
  7. key: topic

接下来