向 yaml 文件中添加新数据项的代码示例
来源:stackoverflow
时间:2024-02-17 18:42:24 366浏览 收藏
大家好,今天本人给大家带来文章《向 yaml 文件中添加新数据项的代码示例》,文中内容主要涉及到,如果你对Golang方面的知识点感兴趣,那就请各位朋友继续看下去吧~希望能真正帮到你们,谢谢!
我有 yaml 文件,我需要使用 go 代码在运行时向其中添加数据 我的意思是,路径如下所示
这是一个 yaml 文件,在 snk_dev
的 sif
下有一个条目
spec: mec: tolerations: - effect: noschedule key: workgroup operator: equal value: goxy resources: requests: cpu: 100m memory: 1gi customconfig: sif: prom_exporter: type: prometheus_exporter snk_dev: type: sk_hc_logs inputs: - tesslt ep: ${nt} dken: ${sn} encoding: codec: "json" index: us compression: gzip buffer: type: memory
在以下 yaml 路径下我需要添加一个新条目
spec->mec->customconfig->sif
一个新条目 snd_prd
snk_prod: type: sk_hc_logs inputs: - tesslt ep: ${NT} dken: ${SN} encoding: codec: "json" index: us compression: gzip buffer: type: memory
我们正在使用 kustomize,我想知道是否有办法通过代码来完成它,我的意思是提前选择我需要添加的文件并在运行时添加它
或者也许使用 更好 https://github.com/go-yaml/yaml
正确答案
这里的关键是生成等效的 go 结构来对 yaml 进行建模并使用 gopkg.in/yaml.v3 包中的 marhshal/unmarshal 函数。
您可以使用 yaml-to-go 等工具来自动生成 yaml 所需的结构,然后在其之上执行更多其他自定义操作。下面的答案采用了此类工具的定义。
您的 yaml 结构可以稍微改进一下,因为 snk_dev
和 snk_prod
字段看起来很相似。您应该为这两者定义一个通用类型,并定义一个 yaml 对象列表,这些对象又会转换为该特定类型的结构切片。但由于原始 yaml 将它们两个保留为不同的实体,因此您的结构也需要不同。
根据您 comment 的答案,字段 snk_dev
和 snk_prod
是动态派生的,因此将您的 customconfig
定义为 map[string]interface{}
以允许动态键名称是有意义的。
package main import ( "fmt" "log" "gopkg.in/yaml.v3" ) type yamldata struct { spec spec `yaml:"spec"` } type tolerations struct { effect string `yaml:"effect"` key string `yaml:"key"` operator string `yaml:"operator"` value string `yaml:"value"` } type requests struct { cpu string `yaml:"cpu"` memory string `yaml:"memory"` } type resources struct { requests requests `yaml:"requests"` } type promexporter struct { type string `yaml:"type"` } type encoding struct { codec string `yaml:"codec"` } type buffer struct { type string `yaml:"type"` } type sifconfig struct { type string `yaml:"type"` inputs []string `yaml:"inputs"` ep string `yaml:"ep"` dken string `yaml:"dken"` encoding encoding `yaml:"encoding"` index string `yaml:"index"` compression string `yaml:"compression"` buffer buffer `yaml:"buffer"` } type customconfig struct { sif map[string]interface{} `yaml:"sif"` } type mec struct { tolerations []tolerations `yaml:"tolerations"` resources resources `yaml:"resources"` customconfig customconfig `yaml:"customconfig"` } type spec struct { mec mec `yaml:"mec"` } var data = `spec: mec: tolerations: - effect: noschedule key: workgroup operator: equal value: goxy resources: requests: cpu: 100m memory: 1gi customconfig: sif: prom_exporter: type: prometheus_exporter snk_dev: type: sk_hc_logs inputs: - tesslt ep: ${nt} dken: ${sn} encoding: codec: "json" index: us compression: gzip buffer: type: memory ` func main() { t := yamldata{} err := yaml.unmarshal([]byte(data), &t) if err != nil { log.fatalf("error: %v", err) } config := &t.spec.mec.customconfig config.sif["snk_prod"] = sifconfig{ type: "sk_hc_logs", inputs: []string{"tesslt"}, ep: "${nt}", dken: "${sn}", encoding: encoding{codec: "json"}, index: "us", compression: "gzip", buffer: buffer{type: "memory"}, } yamlbytes, err := yaml.marshal(t) if err != nil { log.fatalf("error: %v", err) } fmt.println(string(yamlbytes)) }
yamlbytes
可以进一步用作单独的文件,该文件被排除在上面。
yaml 编解码器支持解码为 map[string]any
并将此类映射编码为 yaml。
这个想法是解码文档和额外的树,然后将额外的映射放在所需的路径下,然后编码回来。
type yamlobject map[string]any func main() { // parse the initial document doc := make(yamlobject) yaml.unmarshal([]byte(document), &doc) // parse the additional nodes addon := make(yamlobject) yaml.unmarshal([]byte(extra), &addon) // find the node by the path node := findchild(doc, "spec", "mec", "customconfig", "sif") if node == nil { panic("must not happen") } // add the keys from the additional document // under the specified path for key, val := range addon { (*node)[key] = val } // output the modified document outdoc, _ := yaml.marshal(doc) println(string(outdoc)) } func findchild(obj yamlobject, path ...string) *yamlobject { if len(path) == 0 { return &obj } key := path[0] child, ok := obj[key] if !ok { return nil } obj, ok = child.(yamlobject) if !ok { return nil } return findchild(obj, path[1:]...) }
完整示例 https://go.dev/play/p/pTdXR53p0mq
输出:
spec: mec: customConfig: sif: prom_exporter: type: prometheus_exporter snk_dev: buffer: type: memory compression: gzip dken: ${SN} encoding: codec: json ep: ${NT} index: us inputs: - tesslt type: sk_hc_logs snk_prod: buffer: type: memory compression: gzip dken: ${SN} encoding: codec: json ep: ${NT} index: us inputs: - tesslt type: sk_hc_logs resources: requests: cpu: 100m memory: 1Gi tolerations: - effect: NoSchedule key: WorkGroup operator: Equal value: goxy
yaml 编解码器按字母顺序输出键
今天关于《向 yaml 文件中添加新数据项的代码示例》的内容就介绍到这里了,是不是学起来一目了然!想要了解更多关于的内容请关注golang学习网公众号!
-
502 收藏
-
502 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
139 收藏
-
204 收藏
-
325 收藏
-
477 收藏
-
486 收藏
-
439 收藏
-
357 收藏
-
352 收藏
-
101 收藏
-
440 收藏
-
212 收藏
-
143 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 508次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 497次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 484次学习