使用 go-control-plane 自定义服务网格控制面

写在前面

阅读本文需要最起码了解envoy相关的概念

本文只是一个类似于demo的测试,只为了学习istio,更好的理解istio中的控制面和数据面(pilot -> proxy)是如何交互的,下图的蓝色虚线
在这里插入图片描述
先说go-control-plane是什么,它是一个用于实现xDS APIgolang库,xDS APIEnvoy用于动态配置的协议。我们实现了go-control-plane就可以做到

  1. 动态配置管理:允许控制面动态更新数据面代理的配置
  2. 支持多种 xDS 资源
  3. 缓存和版本管理:提供快照缓存机制,管理配置的版本和更新

使用go-control-plane我们可以创建一个自定义控制面来管理·Envoy`,从而实现动态的服务发现、负载均衡和路由等等…

xds则是一系列服务发现的总称

  • lds 监听器服务发现
  • rds 路由服务发现
  • cds 集群服务发现
  • eds 端点服务发现
  • ads 聚合服务发现

等等等等,还有一些其他的服务发现,本文不涉及就没有说到,如果不理解这些概念,建议先去官网了解一下 https://www.envoyproxy.io

实现 go-control-plane

功能描述

在这里插入图片描述
上文是envoy代理的架构,程序中的逻辑我使用倒叙的方式描述

  • 创建endpoint地址是www.envoyproxy.io端口是443
  • 创建cluster叫做xds_demo_cluster它的端点就是上面创建的
  • 创建路由在filter_chins下的http_connection_manager中名称叫做xds_demo_route,没有任何的路由规则,路由的cluster名称(请求转发的目的地)叫做xds_demo_cluster
  • 最后创建listener名称是listener_xds_demo监听的地址是0.0.0.0:12000

整体就是当我们访问localhost:12000的时候会将我们的请求转发到www.envoyproxy.io
如果运行过默认的envoy的用户可能就会发现我程序中下发的配置就是默认运行envoy时的配置,只不过默认运行envoy静态配置文件的方式就是所有的配置都写在envoy.yaml中,而本文是动态的方式。

envoy有多种运行方式本文不做赘述

功能实现

package mainimport ("context""log""net""time"cluster "github.com/envoyproxy/go-control-plane/envoy/config/cluster/v3"corev3 "github.com/envoyproxy/go-control-plane/envoy/config/core/v3"endpoint "github.com/envoyproxy/go-control-plane/envoy/config/endpoint/v3"listener "github.com/envoyproxy/go-control-plane/envoy/config/listener/v3"route "github.com/envoyproxy/go-control-plane/envoy/config/route/v3"routerv3 "github.com/envoyproxy/go-control-plane/envoy/extensions/filters/http/router/v3"http_connection_manager "github.com/envoyproxy/go-control-plane/envoy/extensions/filters/network/http_connection_manager/v3"tlsv3 "github.com/envoyproxy/go-control-plane/envoy/extensions/transport_sockets/tls/v3"clusterservice "github.com/envoyproxy/go-control-plane/envoy/service/cluster/v3"discoverygrpc "github.com/envoyproxy/go-control-plane/envoy/service/discovery/v3"endpointservice "github.com/envoyproxy/go-control-plane/envoy/service/endpoint/v3"listenerservice "github.com/envoyproxy/go-control-plane/envoy/service/listener/v3"routeservice "github.com/envoyproxy/go-control-plane/envoy/service/route/v3""github.com/envoyproxy/go-control-plane/pkg/cache/types""github.com/envoyproxy/go-control-plane/pkg/cache/v3""github.com/envoyproxy/go-control-plane/pkg/resource/v3""github.com/envoyproxy/go-control-plane/pkg/server/v3""google.golang.org/grpc""google.golang.org/protobuf/types/known/anypb""google.golang.org/protobuf/types/known/durationpb"
)func main() {ctx := context.Background()lis, err := net.Listen("tcp", ":18000")if err != nil {log.Fatalf("Failed to listen: %v", err)}sc := cache.NewSnapshotCache(true, cache.IDHash{}, nil)srv := server.NewServer(ctx, sc, nil)// new grpc servergs := grpc.NewServer()clusterservice.RegisterClusterDiscoveryServiceServer(gs, srv)endpointservice.RegisterEndpointDiscoveryServiceServer(gs, srv)listenerservice.RegisterListenerDiscoveryServiceServer(gs, srv)routeservice.RegisterRouteDiscoveryServiceServer(gs, srv)discoverygrpc.RegisterAggregatedDiscoveryServiceServer(gs, srv)err = setSnapshot(ctx, "xds-node-id", sc)if err != nil {log.Fatalf("set snapshot error: %v", err)} else {log.Println("set snapshot success")}log.Println("Starting control plane server...")if err := gs.Serve(lis); err != nil {log.Fatalf("Failed to serve: %v", err)}
}func setSnapshot(ctx context.Context, nodeId string, sc cache.SnapshotCache) error {clusterName := "xds_demo_cluster"manager := buildHttpManager(clusterName)fcs := buildFilterChain(manager)serviceListener := buildListener("0.0.0.0", 12000, fcs)serviceEndpoint := buildEndpoint()serviceCluster := buildCluster(clusterName, serviceEndpoint)rs := map[resource.Type][]types.Resource{resource.ClusterType:  {serviceCluster},resource.EndpointType: {serviceEndpoint},resource.ListenerType: {serviceListener},resource.RouteType:    {manager},}snapshot, err := cache.NewSnapshot("1", rs)if err != nil {log.Fatalf("new snapshot error: %v", err)}return sc.SetSnapshot(ctx, nodeId, snapshot)
}func buildFilterChain(manager *http_connection_manager.HttpConnectionManager) []*listener.FilterChain {managerPB, err := anypb.New(manager)if err != nil {log.Fatalf("Failed to marshal HttpConnectionManager: %v", err)}fcs := make([]*listener.FilterChain, 0, 0)fcs = append(fcs, &listener.FilterChain{Filters: []*listener.Filter{{Name: "envoy.filters.network.http_connection_manager",ConfigType: &listener.Filter_TypedConfig{TypedConfig: managerPB,},},},})return fcs
}func buildHttpManager(clusterName string) *http_connection_manager.HttpConnectionManager {xdsRoute := &route.RouteConfiguration{Name: "xds_demo_route",VirtualHosts: []*route.VirtualHost{{Name:    "xds_demo_service",Domains: []string{"*"},Routes: []*route.Route{{Match: &route.RouteMatch{PathSpecifier: &route.RouteMatch_Prefix{Prefix: "/",},},Action: &route.Route_Route{Route: &route.RouteAction{HostRewriteSpecifier: &route.RouteAction_HostRewriteLiteral{HostRewriteLiteral: "www.envoyproxy.io",},// 集群要去下文一致ClusterSpecifier: &route.RouteAction_Cluster{Cluster: clusterName,},},},},},},},}routerConfig, _ := anypb.New(&routerv3.Router{})// http 链接管理器manager := &http_connection_manager.HttpConnectionManager{StatPrefix: "ingress_http",RouteSpecifier: &http_connection_manager.HttpConnectionManager_RouteConfig{RouteConfig: xdsRoute,},HttpFilters: []*http_connection_manager.HttpFilter{{Name: "envoy.filters.http.router",ConfigType: &http_connection_manager.HttpFilter_TypedConfig{TypedConfig: routerConfig,},},},SchemeHeaderTransformation: &corev3.SchemeHeaderTransformation{Transformation: &corev3.SchemeHeaderTransformation_SchemeToOverwrite{SchemeToOverwrite: "https",},},}return manager
}func buildEndpoint() *endpoint.LbEndpoint {epTarget := &endpoint.LbEndpoint{HostIdentifier: &endpoint.LbEndpoint_Endpoint{Endpoint: &endpoint.Endpoint{Address: &corev3.Address{Address: &corev3.Address_SocketAddress{SocketAddress: &corev3.SocketAddress{Address: "www.envoyproxy.io",PortSpecifier: &corev3.SocketAddress_PortValue{PortValue: 443,},},},},},},}return epTarget
}func buildCluster(clusterName string, ep *endpoint.LbEndpoint) *cluster.Cluster {serviceCluster := &cluster.Cluster{Name:           clusterName,ConnectTimeout: durationpb.New(time.Second * 3),ClusterDiscoveryType: &cluster.Cluster_Type{Type: cluster.Cluster_STRICT_DNS,},DnsLookupFamily: cluster.Cluster_V4_ONLY,LbPolicy:        cluster.Cluster_ROUND_ROBIN,LoadAssignment: &endpoint.ClusterLoadAssignment{ClusterName: clusterName,Endpoints: []*endpoint.LocalityLbEndpoints{{LbEndpoints: []*endpoint.LbEndpoint{ep},},},},TransportSocket: &corev3.TransportSocket{Name:       "envoy.transport_sockets.tls",ConfigType: nil,},}us := &tlsv3.UpstreamTlsContext{Sni: "www.envoyproxy.io",}tlsConfig, _ := anypb.New(us)serviceCluster.TransportSocket.ConfigType = &corev3.TransportSocket_TypedConfig{TypedConfig: tlsConfig,}return serviceCluster
}func buildListener(ip string, port uint32, fcs []*listener.FilterChain) *listener.Listener {return &listener.Listener{Name: "listener_xds_demo",Address: &corev3.Address{Address: &corev3.Address_SocketAddress{SocketAddress: &corev3.SocketAddress{Address: ip,PortSpecifier: &corev3.SocketAddress_PortValue{PortValue: port,},},},},// 过滤器链FilterChains: fcs,}
}
  • cache.NewSnapshotCache返回一个SnapshotCachego-control-plane中的一个核心组件,用于管理和存储Envoy所需的 xDS资源的快照,并且向Envoy实例推送更新
  • server.NewServer创建xDS服务实例
  • sc.SetSnapshot用于将生成的配置快照设置到指定的节点上,是动态配置和更新Envoy代理的行为,入参有一个id 是下文envoy引导配置中的id

截止到现在我们就可以启动这个服务,我们要记住当前服务监听的地址,因为在envoy.yaml中会需要使用到的

为 envoy 设置引导配置

引导配置(bootstrap configuration),引导配置文件通常指定控制面地址(如xDS服务器地址)、监听器、集群、管理接口等基本信息。

node:id: xds-node-idcluster: xds-node-clusterdynamic_resources:ads_config:api_type: GRPCgrpc_services:- envoy_grpc:cluster_name: xds_clustercds_config:ads: {}lds_config:ads: {}static_resources:clusters:- name: xds_clusterconnect_timeout: 1stype: STRICT_DNSlb_policy: ROUND_ROBINhttp2_protocol_options: {}load_assignment:cluster_name: xds_clusterendpoints:- lb_endpoints:- endpoint:address:socket_address:address: 127.0.0.1port_value: 18000admin:access_log_path: /dev/nulladdress:socket_address:address: 0.0.0.0port_value: 9901

解释

  1. static_resources
    1. 定义静态集群xds_cluster,用于与xDS服务器通信。这里的xDS服务器运行在127.0.0.1:18000就是我们上面服务监听的地址
  2. dynamic_resources
    1. ads_config: 配置聚合发现服务(ADS)来动态获取配置,使用gRPC服务与xds_cluster进行通信
    2. cds_configlds_config分别表示使用ADS来获取配置
  3. admin
    1. 定义管理接口,监听0.0.0.0:9901,用于查看Envoy的状态和配置
  4. node
    1. id 节点的唯一标识符,用于在xDS服务器中区分不同的Envoy实例
    2. cluster 节点所属的集群名称。

然后启动envoy,从输出的日志中我们可以看到通过控制面下发的配置,数据面已经加载成功了。

[2024-06-27 07:47:53.524][1][info][main] [source/server/server.cc:977] starting main dispatch loop
[2024-06-27 07:47:53.526][1][info][upstream] [source/common/upstream/cds_api_helper.cc:32] cds: add 1 cluster(s), remove 1 cluster(s)
[2024-06-27 07:47:53.527][1][info][upstream] [source/common/upstream/cds_api_helper.cc:71] cds: added/updated 1 cluster(s), skipped 0 unmodified cluster(s)
[2024-06-27 07:47:53.544][1][info][upstream] [source/common/upstream/cluster_manager_impl.cc:240] cm init: all clusters initialized
[2024-06-27 07:47:53.544][1][info][main] [source/server/server.cc:957] all clusters initialized. initializing init manager
[2024-06-27 07:47:53.546][1][info][upstream] [source/common/listener_manager/lds_api.cc:106] lds: add/update listener 'listener_xds_demo'
[2024-06-27 07:47:53.546][1][info][config] [source/common/listener_manager/listener_manager_impl.cc:930] all dependencies initialized. starting workers

下一步我们可以访问一下监听的地址,可以看到成功转发到了envoy的官方网站。

~ $ curl http://localhost:12000 | grep title% Total    % Received % Xferd  Average Speed   Time    Time     Time  CurrentDload  Upload   Total   Spent    Left  Speed19 15571   19  3098    0     0   5286      0  0:00:02 --:--:--  0:00:02  5286    <title>Envoy proxy - home</title>
100 15571  100 15571    0     0  21685      0 --:--:-- --:--:-- --:--:-- 21686

我们还可以通过 ~ $ curl http://localhost:9901/config_dump 来查看envoy的实时配置

写在最后

动态配置的方式是在内存中加载配置,不会更新到静态的文件中。

更高级、复杂的用法可以参考istio;具体来说pilot watch集群中的服务、端点、配置等资源的变化。当检测到这些资源的变化时,pilot会生成新的配置,并通过xDS API将更新推送到相应的Envoy实例,从而实现动态配置和管理服务网格中的流量控制和路由规则。这样可以确保 Envoy始终具有最新的服务发现信息和路由配置。

源码目录 https://github.com/istio/istio/tree/master/pilot

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/38210.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

nginx 1024 worker_connections are not enough while connecting to upstream

现象 请求api响应慢&#xff0c;甚至出现504 gateway timeout&#xff0c;重启后端服务不能恢复&#xff0c;但重启nginx可以恢复。 解决方案 worker_connections使用了默认值 1024&#xff0c;当流量增长时&#xff0c;导致连接不够 在nginx.conf中修改连接数就可以了&…

小白学python(第四天)顺序与分支篇

这几天因为个人原因&#xff0c;python篇会更新比较慢&#xff0c;还望大家谅解&#xff0c;那么废话不多说&#xff0c;我们现在就进入正题 顺序篇 这个没啥好说的&#xff0c;就是自上而下&#xff0c;依次执行 分支篇 条件&#xff08;if&#xff09;语句语法格式&#…

【C++】————string基础用法及部分函数底层实现

作者主页&#xff1a; 作者主页 本篇博客专栏&#xff1a;C 创作时间 &#xff1a;2024年6月30日 前言&#xff1a; 本文主要介绍STL容器之一 ---- string&#xff0c;在学习C的过程中&#xff0c;我们要将C视为一个语言联邦&#xff08;摘录于Effective C 条款一&#x…

实验6 形态学图像处理

1. 实验目的 ①掌握数字图像处理中&#xff0c;形态学方法的基本思想&#xff1b; ②掌握膨胀、腐蚀、开运算、闭运算等形态学基本运算方法&#xff1b; ③能够利用形态学基本运算方法&#xff0c;编程实现图像去噪&#xff0c;边界提取等功能。 2. 实验内容 ①调用Matlab /…

PMBOK® 第六版 结束项目或阶段

目录 读后感—PMBOK第六版 目录 不论是阶段的收尾还是项目整体的收尾&#xff0c;都应是令人振奋的事。然而&#xff0c;在实际生活中&#xff0c;收尾工作却相当艰难。会遭遇负责人调离、换任&#xff0c;导致不再需要已购产品&#xff1b;项目收尾时对照招标文件或合同&…

51-61 CVPR 2024 最佳论文 | Rich Human Feedback for Text-to-Image Generation

23年12月&#xff0c;加州大学圣地亚哥、谷歌研究院、南加州大学、剑桥大学联合发布Rich Human Feedback for Text-to-Image Generation论文。 作者受大模型中RLHF技术的启发&#xff0c;用人类反馈来改进Stable Diffusion等文生图模型&#xff0c;提出了先进的RichHF-18K数据…

足球虚拟越位线技术FIFA OT(一)

此系列文章用于记录和回顾开发越位线系统的过程&#xff0c;平时工作较忙&#xff0c;有空时更新。 越位线技术 越位技术已被用于图形化分析足球中潜在的越位情况。 自 2018 年将视频助理裁判 &#xff08;VAR&#xff09; 引入比赛规则以来&#xff0c;人们越来越关注准确确…

完美世界|单机版合集(共22个版本)

前言 我是研究单机的老罗&#xff0c;今天给大家带来的是完美世界的单机版合集&#xff0c;一共22个版本。本人亲自测试了一个版本&#xff0c;运行视频如下&#xff1a; 完美世界|单机版合集 先看所有的版本的文件&#xff0c;文件比较大&#xff0c;准备好空间&#xff0c;差…

Transformer详解encoder

目录 1. Input Embedding 2. Positional Encoding 3. Multi-Head Attention 4. Add & Norm 5. Feedforward Add & Norm 6.代码展示 &#xff08;1&#xff09;layer_norm &#xff08;2&#xff09;encoder_layer1 最近刚好梳理了下transformer&#xff0c;今…

【VScode】常用配置

1.indenticator 增加白色竖条&#xff0c;显示方法范围 2.Git Graph 给git变换分支增添颜色区分 3.Vue 系列 vue 系列&#xff1a;给纯白色代码添加 颜色区分 3.eslint eslint警告&#xff0c;比如{ } 只写了半个会标红提示错误 等错误信息提示 需要配置js等页面 非下…

1.linux操作系统CPU负载

目录 概述CPU平均负载查看平均负载结束 概述 CPU 使用率 和CPU 平均使用率。 CPU平均负载 单位时间内系统处于 [可运行状态] 和 [不可中断状态] 的平均进程数&#xff0c;就是平均活跃进程数&#xff0c;和CPU使用率并没有直接关系 可运行状态 正在使用CPU或者正等待CPU的进…

【Elasticsearch】linux使用supervisor常驻Elasticsearch,centos6.10安装 supervisor

背景&#xff1a; linux服务器&#xff0c;CentOS 6操作系统&#xff0c;默认版本python2.6.6&#xff0c;避免安装过多的依赖不升级python 在网上查的资料python2.6.6兼容supervisor版本 3.1.3 安装supervisor 手动在python官网下载supervisor&#xff0c;并上传到服务器 下…

Linux_动、静态库

目录 一、静态库 1、静态库的概念 2、制作静态库的指令 3、制作静态库 4、链接静态库 二、动态库 1、动态库的概念 2、制作动态库的指令 3、制作动态库 4、链接动态库 5、动态库的加载 三、静态库与动态库的区别 结语 前言&#xff1a; 在Linux下大部分程序进…

【JavaEE】多线程代码案例(1)

&#x1f38f;&#x1f38f;&#x1f38f;个人主页&#x1f38f;&#x1f38f;&#x1f38f; &#x1f38f;&#x1f38f;&#x1f38f;JavaEE专栏&#x1f38f;&#x1f38f;&#x1f38f; &#x1f38f;&#x1f38f;&#x1f38f;上一篇文章&#xff1a;多线程&#xff08;2…

Android Studio环境搭建(4.03)和报错解决记录

1.本地SDK包导入 安装好IDE以及下好SDK包后&#xff0c;先不要管IDE的引导配置&#xff0c;直接新建一个新工程&#xff0c;进到开发界面。 SDK路径配置&#xff1a;File---->>Other Settings---->>Default Project Structure 拷贝你SDK解压的路径来这&#xff0c;…

ros笔记01--初次体验ros2

ros笔记01--初次体验ros2 介绍安装ros2测试验证ros2说明 介绍 机器人操作系统(ROS)是一组用于构建机器人应用程序的软件库和工具。从驱动程序和最先进的算法到强大的开发者工具&#xff0c;ROS拥有我们下一个机器人项目所需的开源工具。 当前ros已经应用到各类机器人项目开发中…

操作符详解(下) (C语言)

操作符详解下 操作符的属性1.优先级2.结合级 表达式求值1.整型提升2.如何进行整形提升呢&#xff1f;3.算术转换4.问题表达式解析 操作符的属性 C语言的操作符有2个重要的属性&#xff1a;优先级、结合性&#xff0c;这两个属性决定了表达式求值的计算顺序。 1.优先级 优先级…

问题:第一次世界大战的起止时间是 #其他#学习方法#微信

问题&#xff1a;第一次世界大战的起止时间是 A.1913 ~1918 年 B.1913 ~1918 年 C.1914 ~1918 年 D.1914 ~1919 年 参考答案如图所示

RabbitMQ 进程内流控(Flow Control) 源码解析

1. 概述 1.1 为什么要流控&#xff1f; 流控主要是为了防止生产者生产消息速度过快&#xff0c;超过 Broker 可以处理的速度。这时需要暂时限制生产者的生产速度&#xff0c;让 Broker 的处理能够跟上生产速度。 Erlang进程之间不共享内存&#xff0c;每个进程都有自己的进程邮…

42.HOOK引擎核心代码

上一个内容&#xff1a;41.HOOK引擎设计原理 以 40.设计HOOK引擎的好处 它的代码为基础进行修改 主要做的是读写寄存器 效果图 添加一个类 htdHook.h文件中的实现 #pragma once class htdHook { public:htdHook(); };htdHook.cpp文件中的实现&#xff1a; #include "…