[英]How to return a partial struct based on query parameters in Go?
我正在嘗試根據查詢參數在 Rest 資源上實現屬性選擇。 API 客戶端將提供一個名為fields
的查詢參數。 服務器將只返回查詢字符串中提到的資源的屬性。 服務器應根據查詢參數返回資源的不同部分表示。 以下是一些示例請求。
GET /api/person/42/?fields=id,createdAt
GET /api/person/42/?fields=address,account
GET /api/person/42/?fields=id,priority,address.city
我嘗試 go map[string]any
路線,但它沒有 go 好。 我正在使用 MongoDB。當我將 mongo 文檔解碼為map[string]any
字段名稱和類型都不匹配。 因此,我正在嘗試動態創建一個新結構。
這是我的嘗試:
func main() {
query, _ := url.ParseQuery("fields=id,priority,address.city")
fields := strings.Split(query.Get("fields"), ",") // TODO: extractFields
person := getPerson() // Returns a Person Struct
personish := PartialStruct(person, fields)
marshalled, _ := json.Marshal(personish) // TODO: err
fmt.Println(string(marshalled))
}
func PartialStruct(original any, fields []string) any {
// Is there any alternative to reflect ?
originalType := reflect.TypeOf(original)
partialFields := make([]reflect.StructField, 0)
for _, field := range reflect.VisibleFields(originalType) {
queryName := field.Tag.Get("json") // TODO: extractQueryName
if slices.Contains(fields, queryName) {
partialFields = append(partialFields, field)
}
}
partialType := reflect.StructOf(partialFields)
// Is there any alternative to Marshal/Unmarshal?
partial := reflect.New(partialType).Interface()
marshalled, _ := json.Marshal(original) // TODO: err
_ = json.Unmarshal(marshalled, partial) // TODO: err
return partial
}
這是一個可運行的例子https://go.dev/play/p/Egomxe5NjEc
資源被建模為嵌套結構。 嵌套字段將用“.”表示。 查詢字符串中的點。
如何改進PartialStruct
以處理嵌套字段,例如address.city
?
如果有更好的方法,我願意改變方向。
查看第三方庫: graphql
我寫的一個例子可能對你有幫助:
package main
import (
"encoding/json"
"fmt"
"log"
"github.com/graphql-go/graphql"
)
func main() {
// Schema
fields := graphql.Fields{
"id": &graphql.Field{
Type: graphql.ID,
Resolve: func(p graphql.ResolveParams) (interface{}, error) {
return 111, nil
},
},
"priority": &graphql.Field{
Type: graphql.String,
Resolve: func(p graphql.ResolveParams) (interface{}, error) {
return "admin", nil
},
},
"address": &graphql.Field{
Type: graphql.NewObject(graphql.ObjectConfig{
Name: "address",
Fields: graphql.Fields{
"city": &graphql.Field{
Type: graphql.String,
},
"country": &graphql.Field{
Type: graphql.String,
},
},
}),
Resolve: func(p graphql.ResolveParams) (interface{}, error) {
return map[string]string{
"city":"New York",
"country": "us",
}, nil
},
},
}
rootQuery := graphql.ObjectConfig{Name: "RootQuery", Fields: fields}
schemaConfig := graphql.SchemaConfig{Query: graphql.NewObject(rootQuery)}
schema, err := graphql.NewSchema(schemaConfig)
if err != nil {
log.Fatalf("failed to create new schema, error: %v", err)
}
// Query
query := `
{
id,
address {
city,country
},
priority
}
`
params := graphql.Params{Schema: schema, RequestString: query}
r := graphql.Do(params)
if len(r.Errors) > 0 {
log.Fatalf("failed to execute graphql operation, errors: %+v", r.Errors)
}
rJSON, _ := json.Marshal(r)
fmt.Printf("%s \n", rJSON)
}
這是一個可運行的例子https://go.dev/play/p/pHH2iBzCLT-
聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.