【kubernetes】使用luakube访问kubernetes api

文章目录

      • 1 kubernetes client
      • 2 luakube初体验
      • 3 luakube代码分析
      • 4 luakube包的调用
      • 5 lua相关
        • 5.1 self
        • 5.2 metatable
          • 5.2.1 使用metatable对table新增操作符
          • 5.2.2 使用metatable对table新增方法
          • 5.2.3 再探luakube
      • 6 参考文档

1 kubernetes client

客户端列出了各种语言对应的访问k8s的客户端库,有官方维护的,还有一些是社区和个人维护的。

这里面没有列出lua的库,在GitHub上搜索可以看到有luakube,虽然好像没啥star,而且4个多月没更新了,可以用它进行测试下,如果基础的连接k8s的部分完成了,后面的各种资源操作可以自行开发和添加。当然,了解kubernetes api的都知道,这些api库基本就是对http api的调用封装,因此,luakube也是对http api的封装,它的优点在于,使用体验跟k8s的golang库的api类似,也就是拥有类k8s的调用方式。

2 luakube初体验

要使用luakube,第一步当然是环境安装,然后执行其中的示例代码。

luakube是个lua中的包,使用luarocks进行管理,因此,先安装lua和luarocks,然后再安装luakube。

luakube的依赖:

  • lyaml:yaml文件的解析
  • luajson:json的解析
  • base64:解析token
  • luasec:https
  • luasocket:https依赖luasocket
  • fun:提供了一些iter、map等操作,可以改写
  • lpeg:luajson依赖lpeg

luarocks中每个包都有一个rockspec文件,该文件描述了该包的依赖以及一些安装信息。因此,安装luakube有两种方式:要么按照rockspec的提示一个依赖一个依赖的安装,要么直接使用rockspec文件安装:

安装方法1:

yum install -y libyaml-devel
luarocks install lyaml
yum install -y lua-lpeg
luarocks install luajson
luarocks install luasocket
yum install -y openssl-devel
luarocks install luasec
luarocks install base64
luarocks install fun
luarocks install luakube

安装方法2:

luarocks install https://raw.githubusercontent.com/f4z3r/luakube/master/rockspecs/luakube-0.1.0-0.rockspec

安装方法3:

git clone https://github.com/f4z3r/luakube.git
cd luakube && luarocks make

第1种方法和第2种方法都是将luakube作为模块安装,相当于直接下载源码然后拷贝到安装目录,第3种方式是直接用本地目录中的rockspec文件进行安装,将本地目录的源代码拷贝到安装目录。

因此,如果是需要对luakube进行修改,可以用第3种方式,在本地目录修改后执行luarocks make,然后再进行测试。

安装完luakube,可以执行简单的demo代码进行测试,在luakube仓库的examples目录中有获取pod的logs的测试代码get_logs.lua:

local config = require "kube.config"
local api = require "kube.api"-- Use local kube config to connect to cluster
local conf = config.from_kube_config()
local global_client = api.Client:new(conf)-- Get the Core V1 client
local client = global_client:corev1()-- Get the last three lines of logs from the coredns container as a string
local container_logs = client:pods("kube-system"):logs("coredns-558bd4d5db-xr5tj", {tailLines = 3, container = "coredns"})-- Get the logs over the last 10 seconds for all containers in the pod
local last_logs = client:pods("kube-system"):logs("coredns-558bd4d5db-xr5tj", {sinceSeconds = 10})

测试时需要修改上面的名称空间和pod名,就可以得到对应pod的日志。

3 luakube代码分析

luakube源代码分成3个部分:

  • config.lua:对kubeconfig文件进行操作,外部代码实际调用时通常是调用from_kube_config(如果参数指定了kubeconfig文件,则用该文件作为凭证,如果没有指定,则用默认的kubeconfig文件路径)和in_cluster_config(在k8s集群中运行时获取sa的token作为凭证)
  • api.lua:对https调用的封装
  • api/:对每个version对应的资源进行操作

下面从分析examples/get_logs.lua的角度看下luakube的具体实现。

local conf = config.from_kube_config()

首先获取到kubeconfig配置,如果没有提供文件名,则使用默认配置文件($HOME/.kube/config)。

local global_client = api.Client:new(conf)

根据上面得到的kubeconfig配置创建client,这里创建的就是api.lua中的api.Client。

local client = global_client:corev1()

调用api.lua中的api.Client:corev1()获取某个版本的api。

local container_logs = client:pods("kube-system"):logs("coredns-558bd4d5db-xr5tj", {tailLines = 3, container = "coredns"})

上面的代码是获取kube-system命名空间的coredns-558bd4d5db-xr5tj这个pod的coredns容器的后面3行日志。

调用client:pods就是调用core_v1.lua中的core_v1.Client.pods = utils.generate_object_client("pods", pod_base, true, true, true, extras),而utils.generate_object_client则是返回一个client,该client有各种操作资源的方法,同时还会将extras中指定的一些方法也放到该client中,对于core_v1.Client.pods,extras中有两个方法:logs和ephemeralcontainers,分别用于获取日志和临时容器。

当执行logs()时,就会调用extras中的logs方法,它会调用自身的raw_call()方法,而raw_call该方法就定义于api.lua中。

相当于调用的起点和终点都位于api.lua中。

在整体的实现中,用到了lua中的metadata编程,让返回的client拥有各种方法。

4 luakube包的调用

了解了上述实现的函数调用流程,在具体使用luakube包时,就可以按照k8s的设计调用相关的通用的函数。

例如,对于pod,可以使用client:pods(“kube-system”):list()获取kube-system命名空间的所有pod的列表,其他的如nodes、services、configmaps、secrets、serviceaccounts等都有类似的调用。

当前资源支持的操作有:

  • get(name, query):获取某个资源,可以给出资源名称
  • status(name):资源状态
  • create(obj, query):创建资源,提供创建资源的yaml
  • update(obj, query):更新资源
  • update_status(obj, query):更新资源状态
  • patch(name, patch, query, style):变更资源
  • path_status(name, patch, query, style):变更资源状态
  • delete(name, query):删除资源
  • delete_collection(body, query)
  • list(query):列出资源

其中:

  • name:资源名称
  • query:获取资源的条件,这个需要传table,然后会拼接到https请求的后面
  • obj:资源的yaml
  • patch:变更的字段
  • style
  • body

5 lua相关

5.1 self

self类似于c++/java中的this和python中的self,c++/java中的this由编译器自动处理,python中的self需要开发人员添加,而lua中的self还提供了语言级别的区分:

local t = {a = 1, b = 2}
function t:Add()return (self.a + self.b)
end
function t.Sub(self)return (self.a - self.b)
endprint(t.Add(t))
print(t:Sub())

如果用冒号定义和调用方法,lua解释器会自动添加self,如果用点号,则需要开发人员添加self。

5.2 metatable

metatable翻译为元表,可以理解为table的额外属性,当访问table的一些操作时,如果table不存在,可以由元表进行完成。

lua中的metatable的概念跟python中的保留方法很像:

python中对于双下划线开头和结尾的方法有特殊含义,例如,当创建一个对象时,就是调用类的__init__()方法,当用iter某个序列时,就是调用类的__iter__()方法,也就是说,当对对象调用某个方法时,会调用某个约定好的方法,如果开发人员没有定义则调用默认的方法,当然,默认方法必须要支持此类行为,例如,如果用iter遍历一个不能遍历的对象,如果没有定义__iter__()则会调用失败。

在lua中,除了常用的基本数据类型(nil、布尔、数字、字符串)之外,table是lua提供的唯一的的复合数据类型(其他数据类型都是针对特定场景),table可以用于实现数组、集合、字典、类等类型,而metatable就是在table上面附加的一种属性,也就是说,metatable只针对table类型。

跟python类似,在lua中,双下划线开头的方法有特殊含义,例如,当用+操作符对两个table相加时,就是调用第一个table的metatable中的__add()方法,这种相当于实现了对操作符的自定义,跟python中的__add__和C++中的operator +类似;当调用table一个不存在的方法时,就会调用table的metatable中的对应的方法,这种相当于实现了对类方法的动态增加。

要想使用metatable,有两个重要的方法:

  • setmetatable(table, metable):设置table的metatable,虽然这里是table,但是根据lua的文档可以知道,任何值都有metatable,但是只有table的metable可以在lua中通过setmtatable修改,其他类型只能通过C语言修改。如果metatable设置为nil,则删除table的metatable;如果table的metatable中有__metatable元素,则抛出异常,函数返回新的table
  • getmetatable(object):返回对象的metatable。如果object没有metatable,则返回nil;如果object的metatable有__metatable元素,则返回对应的值;如果object的metatable没有__metatable元素,则返回对象的metatable。

下面演示操作符的实现和方法的增加:

5.2.1 使用metatable对table新增操作符
mytable = setmetatable({ 1, 2, 3 }, {__add = function(mytable, newtable)return table.move(newtable, 1, #newtable, #mytable+1, mytable)end
})secondtable = {4,5,6}mytable = mytable + secondtablefor k,v in ipairs(mytable) doprint(k,v)
end

使用setmetatable对table增加__add方法,__add实现时使用了table的move方法将第二个table中的元素放到第一个table中。

5.2.2 使用metatable对table新增方法
local mytable = {1, 2, 3}setmetatable(mytable, {__index = function(mytable, key)return function(self, newtable)table.move(newtable, 1, #newtable, #self+1, self)return selfendend
})local secondtable = {4,5,6}mytable:plus(secondtable)for i=1,#mytable doprint(mytable[i])
end

这里是给table增加了plus方法,通过增加带__index的metatable实现,__index对应的是个函数,而且它返回的也是个函数,返回的这个函数完成的就是两个table拼接的操作。因此,当调用mytable:plus时,由于mytable没有plus对应的值,就会查找metatable中的__index元素,由于__index存在,则会调用,并传入mytable和调用的键,因此,这里key==plus,而__index返回的时个函数,也就是说,mytable:plus返回的是个函数,然后用里面的这层函数调用,newtable传入的就是secondtable。

5.2.3 再探luakube

luakube通过k8s的rest api访问,因此,只要知道调用接口的url和参数即可。

k8s的rest api的url格式如下:

http://[api/apis]/api-group/api-version/namespaces/namespace/resource-kind/resource-name

例如,获取命名空间default的所有pod:http://127.0.0.1:8001/api/v1/namespaces/default/pods

而k8s的rest api,其实就是完成资源操作的CRUD,只有在需要提供yaml文件时,才会把yaml文件放到body里面,其他的参数都放到url里面,因此,重点就是要对url进行拼接和调用。

luakube为了使得调用封装的调用更加灵活,将url中不同的部分放到不同的地方。

开始的api和apis在api.lua中,现在只有api/v1是api,其他都是apis。

api-group/api-version在utils.lua的utils.generate_base(api)中,其中参数api就是api-group/api-version,在下面client.call中会将self.api_拼接到路径前面进行调用。

剩下的部分则在utils.lua的utils.generate_object_client(api,concat,namespaced)中,其中参数api就是资源类型resource-kind,concat是yaml中的apiVersion和kind,对于需要yaml文件作为参数的需要,namespaced则是说明该资源是否可以指定namespace,当然,最终是否要加上namespace,还要基于是否提供namespace参数。而最后的resource-name则是通过utils.generate_object_client()中的调用方法提供。

为了将上述的调用串起来,代码中大量使用下面的操作:

self.__index = self
setmetatable(o, self)

上面的代码将self作为o的metatable中的__index元素,于是,当访问o中的元素没有时,就会访问self中的元素,这样能够实现类似继承的机制,同时也可以给o添加方法,让o去调用self的方法,相当于子类调用父类的方法。

因此,将上面的url串起来的方式就是,在utils.generate_object_client()里面创建client,然后将core_v1.Client当作parent参数作为client的metatable,而core_v1.Client是通过utils.generate_base()返回的函数创建的client,这里又通过类似的机制调用api.lua中的方法。

总结下,通过metatable实现类似继承的机制,然后通过继承关系拼接url,最终调用api.lua中的https接口实现k8s的api的访问。

6 参考文档

  • Lua中的self
  • Lua 元表(Metatable)
  • Lua 5.4 Reference Manual
  • k8s restful API 结构分析
  • 对lua继承中self.__index = self的释疑
  • Kubernetes API

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

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

C/C++:[Error] ld returned 1 exit status 解决方案

好久没用了,今天写了会儿代码,各种BUg,emmmmmm 出现了很多次以下这个问题:[Error] ld returned 1 exit status 可能问题&解决方式: 常见的语法/单词拼写错误:常见的Main,printf,scanf等拼写错误 函数名或者声明有…

QT商业播放器

QT商业播放器 总体架构图 架构优点:解耦,采用生产者消费者设计模式,各个线程各司其职,通过消息队列高效协作 这个项目是一个基于ijkplayer和ffplayer.c的QT商业播放器, 项目有5部分构成: 前端QT用户界面 后端是集成了…

制作电子期刊没模板?请疯狂看我

你们是不是也在为制作电子期刊而烦恼?没有合适的模板,内容再精彩也难以展现。今天给大家分享一个超级实用的秘籍!✨ 首先,我们要明白,电子期刊制作的关键在于模板的选择。一个好的模板可以让你的内容瞬间焕发光彩。但是…

分类预测 | MATLAB实现SSA-FS-SVM麻雀算法同步优化特征选择结合支持向量机分类预测

分类预测 | MATLAB实现SSA-FS-SVM麻雀算法同步优化特征选择结合支持向量机分类预测 目录 分类预测 | MATLAB实现SSA-FS-SVM麻雀算法同步优化特征选择结合支持向量机分类预测效果一览基本介绍程序设计参考资料 效果一览 基本介绍 MATLAB实现SSA-FS-SVM麻雀算法同步优化特征选择结…

OOTD | 美式复古穿搭耳机,复古轻便的头戴式耳机推荐

复古耳机更能带来年代感的复古数码产品,头戴式耳机就好似是时光滤镜的时髦配饰,不说功能实用性,在造型上添加就很酷。 随着时代的发展,时尚有了新的定义。对如今的消费者来说,时尚不仅是美学与个性的展现,…

C10K问题:高并发模型设计

一、循环服务器模型 #include <stdio.h> #include <stdlib.h> #include <string.h> #include <errno.h> #include <unistd.h> #include <signal.h> #include <sys/types.h> #include <sys/socket.h> //*******// #include &l…

头戴式耳机怎么戴好看?头戴式耳机正确代法

走在大街上总能看到那么一些人&#xff0c;他们眼神时而朦胧涣散&#xff0c;时而精神奕奕&#xff0c;全身上下始终散发着#请勿打扰#的气息&#xff0c;因为他们都戴着头&#xff01;戴&#xff01;式&#xff01;耳&#xff01;机&#xff01;但是头戴式耳机把头压得扁扁的&a…

《C和指针》笔记31:多维数组的数组名、指向多维数组的指针、作为函数参数的多维数组

文章目录 1. 指向多维数组的数组名2. 指向多维数组的指针3. 作为函数参数的多维数组 1. 指向多维数组的数组名 我们知道一维数组名的值是一个指针常量&#xff0c;它的类型是“指向元素类型的指针”&#xff0c;它指向数组的第1个元素。那么多维数组的数组名代表什么呢&#x…

[管理与领导-113]:IT人看清职场中的隐性规则 - 10 - 看清人的行动、行为、手段、方法背后的动机与背景条件

目录 前言&#xff1a; 一、冰山模型 1.1 冰山模型&#xff0c;系统思考的工具 1.2 冰山模型&#xff1a;发现人行为背后的动机 二、动机、行为模型 "说一套"&#xff1a; "做一套"&#xff1a; "演一套"&#xff1a; "学一套&quo…

【已解决】 Expected linebreaks to be ‘LF‘ but found ‘CRLF‘.

问题描述 团队都是用mac&#xff0c;只有我自己是windows&#xff0c;启动项目一直报错 Expected linebreaks to be ‘LF‘ but found ‘CRLF‘. 但我不能因为自己的问题去改团队配置&#xff0c;也尝试过该vscode配置默认是LF还是报错 思路 看文章vscode如何替换所有文件的…

深入剖析红黑树:优雅地平衡二叉搜索树

目录 一.红黑树的概念二.插入操作三.与AVL树的比较 一.红黑树的概念 在之前的学习中&#xff0c;我们了解了二叉搜索平衡树&#xff0c;AVL树通过控制每个结点中的平衡因子的绝对值不超过1&#xff0c;实现了一个高性能的树。而相较于AVL的高度平衡&#xff0c;红黑树觉得AVL为…

传输层协议—UDP协议

传输层协议—UDP协议 文章目录 传输层协议—UDP协议传输层再谈端口号端口号范围划分pidofnetstat UDP协议端格式UDP报文UDP特点UDP缓冲区基于UDP的应用层协议 传输层 在学习HTTP/HTTPS等应用层协议时&#xff0c;为了方便理解&#xff0c;可以简单认为HTTP将请求和响应直接发送…

JMeter性能分析实战一:日常登录接口

负载测试 日常需求&#xff1a;负载测试&#xff01; 对于桥的负载测试&#xff1a;我给你20t的一排车辆&#xff0c;看你能不能撑得住20t&#xff01; 对于系统的负载测试&#xff1a; 逐步增加负载&#xff0c;便于问题的发现和定位&#xff0c;不要操之过急。逐步增加负载…

Stable Diffusion云服务器部署完整版教程

Stable Diffusion云服务器部署完整版教程 2023年07月04日 22:30 3607浏览 18喜欢 22评论 <span class"bili-avatar-icon bili-avatar-right-icon "></span> </div>薯片_AI 粉丝&#xff1a; 1513 文章&#xff1a; 1 设置分组取消关注 已关注 …

【MySql】3- 实践篇(一)

文章目录 1. 普通索引和唯一索引的选择1.1 查询过程1.2 更新过程1.2.1 change buffer1.2.2 change buffer 的使用场景 1.3 索引选择和实践1.4 change buffer 和 redo log2. MySQL为何有时会选错索引?2.1 优化器的逻辑2.1.1 扫描行数是怎么判断的?2.1.2 重新统计索引信息 2.2 …

C语言中柔性数组的讲解与柔性数组的优势

前言:也许你从来没有听说过柔性数组&#xff08;flexible array&#xff09;这个概念&#xff0c;但是它确实是存在的。C99 中&#xff0c;结构中的最后一个元素允许是未知大小的数组&#xff0c;这就叫做"柔性数组"成员。 目录标题 柔性数组什么是柔性数组呢&#…

【C语言】八大排序算法

文章目录 一、冒泡排序1、定义2、思想及图解3、代码 二、快速排序1、hoare版本2、挖坑法3、前后指针法4、非递归快排5、快速排序优化1&#xff09;三数取中选key值2&#xff09;小区间优化 三、直接插入排序1、定义2、代码 四、希尔排序1、定义2、图解3、代码 五、选择排序1、排…

sheng的学习笔记-【中文】【吴恩达课后测验】Course 2 - 改善深层神经网络 - 第二周测验

课程2_第2周_测验题 目录&#xff1a;目录 第一题 1.当输入从第8个mini-batch的第7个的例子的时候&#xff0c;你会用哪种符号表示第3层的激活&#xff1f; A. 【  】 a [ 3 ] { 8 } ( 7 ) a^{[3]\{8\}(7)} a[3]{8}(7) B. 【  】 a [ 8 ] { 7 } ( 3 ) a^{[8]\{7\}(3)} a…

代码随想录 Day11 二叉树 LeetCode T144,145,94 前中后序遍历 (递归解法)

题解及更详细解答来自于:代码随想录 (programmercarl.com) 前言: 递归三要素 确定递归函数的参数和返回值&#xff1a; 确定哪些参数是递归的过程中需要处理的&#xff0c;那么就在递归函数里加上这个参数&#xff0c; 并且还要明确每次递归的返回值是什么进而确定递归函数的返…

【Redis】基础数据结构-skiplist跳跃表

有序集合Sorted Set zadd zadd用于向集合中添加元素并且可以设置分值&#xff0c;比如添加三门编程语言&#xff0c;分值分别为1、2、3&#xff1a; 127.0.0.1:6379> zadd language 1 java (integer) 1 127.0.0.1:6379> zadd language 2 c (integer) 1 127.0.0.1:6379…