作为标题,我想知道如何使用golang的toml文件。
在此之前,我展示了我的toml示例。是不是?
[datatitle]
enable = true
userids = [
"12345", "67890"
]
[datatitle.12345]
prop1 = 30
prop2 = 10
[datatitle.67890]
prop1 = 30
prop2 = 10
然后,我想将这些数据设置为struct的类型。
因此,我想访问子元素,如下所示。
datatitle["12345"].prop1
datatitle["67890"].prop2
提前致谢!
答案 0 :(得分:9)
首先获得BurntSushi的toml解析器:
go get github.com/BurntSushi/toml
BurntSushi解析toml并将其映射到结构,这就是你想要的。
然后执行以下示例并从中学习:
package main
import (
"github.com/BurntSushi/toml"
"log"
)
var tomlData = `title = "config"
[feature1]
enable = true
userids = [
"12345", "67890"
]
[feature2]
enable = false`
type feature1 struct {
Enable bool
Userids []string
}
type feature2 struct {
Enable bool
}
type tomlConfig struct {
Title string
F1 feature1 `toml:"feature1"`
F2 feature2 `toml:"feature2"`
}
func main() {
var conf tomlConfig
if _, err := toml.Decode(tomlData, &conf); err != nil {
log.Fatal(err)
}
log.Printf("title: %s", conf.Title)
log.Printf("Feature 1: %#v", conf.F1)
log.Printf("Feature 2: %#v", conf.F2)
}
请注意tomlData
以及它如何映射到tomlConfig
结构。
答案 1 :(得分:2)
使用推荐的pkg BurntSushi / toml解决了这个问题!! 我做了如下,它是代码的一部分。
[toml example]
[title]
enable = true
[title.clientinfo.12345]
distance = 30
some_id = 6
[Golang示例]
type TitleClientInfo struct {
Distance int `toml:"distance"`
SomeId int `toml:"some_id"`
}
type Config struct {
Enable bool `toml:"enable"`
ClientInfo map[string]TitleClientInfo `toml:"clientinfo"`
}
var config Config
_, err := toml.Decode(string(d), &config)
然后,它可以像我预期的那样使用。
config.ClientInfo[12345].Distance
谢谢!
答案 2 :(得分:1)
2019年的一个小更新-BurntSushi/toml有了更新的替代方法,它具有更丰富的API,可以处理 .toml 文件:
pelletier/go-toml(和documentation)
例如具有config.toml
文件(或在内存中):
[postgres]
user = "pelletier"
password = "mypassword"
除了使用pelletier/go-toml将整个事物进行常规编组和解组为预定义的结构(您可以在接受的答案中看到)之外,您还可以查询这样的单个值:
config, err := toml.LoadFile("config.toml")
if err != nil {
fmt.Println("Error ", err.Error())
} else {
// retrieve data directly
directUser := config.Get("postgres.user").(string)
directPassword := config.Get("postgres.password").(string)
fmt.Println("User is", directUser, " and password is", directPassword)
// or using an intermediate object
configTree := config.Get("postgres").(*toml.Tree)
user := configTree.Get("user").(string)
password := configTree.Get("password").(string)
fmt.Println("User is", user, " and password is", password)
// show where elements are in the file
fmt.Printf("User position: %v\n", configTree.GetPosition("user"))
fmt.Printf("Password position: %v\n", configTree.GetPosition("password"))
// use a query to gather elements without walking the tree
q, _ := query.Compile("$..[user,password]")
results := q.Execute(config)
for ii, item := range results.Values() {
fmt.Println("Query result %d: %v", ii, item)
}
}
更新
还有spf13/viper可以与.toml配置文件一起使用(在其他受支持的格式中),但是在许多情况下可能有点过头了。
答案 3 :(得分:1)
使用解决方案 Viper ,您可以使用JSON,TOML,YAML,HCL,INI和其他属性格式的配置文件。
创建文件
./config.toml
首次导入:
import (config "github.com/spf13/viper")
初始化:
config.SetConfigName("config")
config.AddConfigPath(".")
err := config.ReadInConfig()
if err != nil {
log.Println("ERROR", err.Error())
}
并获取值:
config.GetString("datatitle.12345.prop1")
config.Get("datatitle.12345.prop1").(int32)
答案 4 :(得分:0)
我正在使用 spf13/viper
状态 | 项目 | 开始 | 叉子 |
---|---|---|---|
活着 | spf13/viper | ||
未维护 | BurntSushi/toml |
所以我选择viper
我试着用一张表格把代码和配置文件的内容放在一起,但很明显,编辑和最终的结果不符,所以我把图片放上来,希望它可以更容易地你来比较
package main
import (
"github.com/spf13/viper"
"log"
"os"
)
func main() {
check := func(err error) {
if err != nil {
panic(err)
}
}
myConfigPath := "test_config.toml"
fh, err := os.OpenFile(myConfigPath, os.O_RDWR, 0666)
check(err)
viper.SetConfigType("toml") // do not ignore
err = viper.ReadConfig(fh)
check(err)
// Read
log.Printf("%#v", viper.GetString("title")) // "my config"
log.Printf("%#v", viper.GetString("DataTitle.12345.prop1")) // "30"
log.Printf("%#v", viper.GetString("dataTitle.12345.prop1")) // "30" // case-insensitive
log.Printf("%#v", viper.GetInt("DataTitle.12345.prop1")) // 30
log.Printf("%#v", viper.GetIntSlice("feature1.userids")) // []int{456, 789}
// Write
viper.Set("database", "newuser")
viper.Set("owner.name", "Carson")
viper.Set("feature1.userids", []int{111, 222}) // overwrite
err = viper.WriteConfigAs(myConfigPath)
check(err)
}
title = "my config"
[datatitle]
[datatitle.12345]
prop1 = 30
[feature1]
userids = [456,789]
database = "newuser" # New
title = "my config"
[datatitle]
[datatitle.12345]
prop1 = 30
[feature1]
userids = [111,222] # Update
[owner] # New
name = "Carson"