Skip to content

Commit

Permalink
Merge pull request #36 from LaurenceLiZhixin/feature/jParser
Browse files Browse the repository at this point in the history
ftr: add json struct parser to support cli tool
  • Loading branch information
AlexStocks authored Dec 9, 2020
2 parents a125b16 + 5772146 commit 6acc198
Show file tree
Hide file tree
Showing 5 changed files with 285 additions and 0 deletions.
237 changes: 237 additions & 0 deletions encoding/json/parser.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,237 @@
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/

package json

import (
"io/ioutil"
"log"
"reflect"
"strconv"
"strings"
"time"
)

import (
"github.com/dubbogo/jsonparser"
perrors "github.com/pkg/errors"
)

// HessianRegisterPair define the pair to register to hessian
type HessianRegisterPair struct {
JavaClassName string
Obj interface{}
}

// jsonStructParser can use reflect to create arbitrary interface{} of go, from user defined json file.
type jsonStructParser struct {
structFields []reflect.StructField
hessianRegisterPair []HessianRegisterPair
valueMap map[string]string
subObjValueMap map[string]reflect.Value
}

// newJSONStructParser create a new json struct parser
func newJSONStructParser() *jsonStructParser {
return &jsonStructParser{
structFields: make([]reflect.StructField, 0, 16),
valueMap: make(map[string]string, 8),
hessianRegisterPair: make([]HessianRegisterPair, 0, 16),
subObjValueMap: make(map[string]reflect.Value, 8),
}
}

// File2Interface first read json byte from @filePath, and parse it to interface
func File2Interface(filePath string) ([]HessianRegisterPair, interface{}, error) {
defer func() {
defaultJSONStructParser = newJSONStructParser()
}()
return defaultJSONStructParser.jsonFilePath2Struct(filePath)
}

func init() {
defaultJSONStructParser = newJSONStructParser()
}

var defaultJSONStructParser *jsonStructParser

// RemoveTargetNameField remove target file in @v
func RemoveTargetNameField(v interface{}, targetName string) interface{} {
defer func() {
defaultJSONStructParser = newJSONStructParser()
}()
return defaultJSONStructParser.removeTargetNameField(v, targetName)
}

func (jsp *jsonStructParser) cb(key []byte, value []byte, dataType jsonparser.ValueType, offset int) error {
switch dataType {
case jsonparser.Object:
// parse sub interface, use a new parser to deal with it the same way
newParser := newJSONStructParser()
subObj := newParser.json2Struct(value)
javaClassName, err := getJavaClassName(subObj)
if err != nil {
return err
}
jsp.hessianRegisterPair = append(jsp.hessianRegisterPair, HessianRegisterPair{
JavaClassName: javaClassName,
Obj: subObj,
})
jsp.structFields = append(jsp.structFields, reflect.StructField{
Name: string(key),
Type: reflect.TypeOf(subObj),
})
jsp.subObjValueMap[string(key)] = reflect.ValueOf(subObj)

case jsonparser.Array: // TODO slice parse
case jsonparser.String: // normal struct parse
// "type@value"
arr := strings.Split(string(value), "@")
var userDefinedType reflect.Type
switch arr[0] {
case "int":
userDefinedType = reflect.TypeOf(0)
case "string":
userDefinedType = reflect.TypeOf("")
case "uint64":
userDefinedType = reflect.TypeOf(uint64(0))
case "time.Time":
userDefinedType = reflect.TypeOf(time.Time{})
case "float32":
userDefinedType = reflect.TypeOf(float32(0))
case "float64":
userDefinedType = reflect.TypeOf(float64(0))
case "bool":
userDefinedType = reflect.TypeOf(false)
default:
log.Printf("error: dataType %s in json is not supported\n", string(value))
return perrors.Errorf("dataType %s in json is not supported", string(value))
}
if len(arr) > 1 {
jsp.valueMap[string(key)] = arr[1]
}
jsp.structFields = append(jsp.structFields, reflect.StructField{
Name: string(key),
Type: userDefinedType,
})
default:
log.Printf("error: dataType %s in json is not supported\n", string(value))
return perrors.Errorf("dataType %s in json is not supported", string(value))
}
return nil
}

// json2Struct parse data from json file to user defined interface
func (jsp *jsonStructParser) json2Struct(jsonData []byte) interface{} {
// first: call ObjectEach to parse jsonData to reflect.StructField item
if err := jsonparser.ObjectEach(jsonData, jsp.cb); err != nil {
log.Println("jsonparser.ObjectEach error = ", err)
}

// second: parse structField to reflectType
typ := reflect.StructOf(jsp.structFields)
v := reflect.New(typ).Elem()
newty := reflect.TypeOf(v.Addr().Interface()).Elem()

// finally: traverse each json field, and set user defined value
for i := 0; i < typ.NumField(); i++ {
valStr, ok1 := jsp.valueMap[newty.Field(i).Name]
subObj, ok2 := jsp.subObjValueMap[newty.Field(i).Name]
if !ok1 && !ok2 {
continue
}

if newty.Field(i).Type.Kind() == reflect.Ptr {
v.Field(i).Set(subObj)
continue
}
switch newty.Field(i).Type {
case reflect.TypeOf(0), reflect.TypeOf(uint64(0)):
if parsedInt, err := strconv.Atoi(valStr); err == nil {
v.Field(i).SetInt(int64(parsedInt))
break
}
v.Field(i).SetInt(0)
case reflect.TypeOf(""):
v.Field(i).SetString(valStr)
case reflect.TypeOf(time.Time{}):
//todo time support v.Field(i).
case reflect.TypeOf(float64(0)), reflect.TypeOf(float32(0)):
if parsedFloat, err := strconv.ParseFloat(valStr, 64); err == nil {
v.Field(i).SetFloat(parsedFloat)
break
}
v.Field(i).SetFloat(0)
case reflect.TypeOf(false):
if valStr == "true" || valStr == "1" {
v.Field(i).SetBool(true)
}
default:
log.Printf("error: val %s in value is not supported\n", valStr)
return perrors.Errorf("val %s in value is not supported", valStr)
}
}
s := v.Addr().Interface()
return s
}

// jsonFilePath2Struct read file from @filePath and parse data to interface
func (jsp *jsonStructParser) jsonFilePath2Struct(filePath string) ([]HessianRegisterPair, interface{}, error) {
jsonData, err := ioutil.ReadFile(filePath)
if err != nil {
return []HessianRegisterPair{}, nil, err
}
return jsp.hessianRegisterPair, jsp.json2Struct(jsonData), nil
}

// removeTargetNameField remove origin interface @v's target field by @targetName
func (jsp *jsonStructParser) removeTargetNameField(v interface{}, targetName string) interface{} {
typ := reflect.TypeOf(v).Elem()
val := reflect.ValueOf(v).Elem()
nums := val.NumField()
structFields := make([]reflect.StructField, 0)
fieldMap := make(map[string]reflect.Value)
for i := 0; i < nums; i++ {
if typ.Field(i).Name != targetName {
structFields = append(structFields, reflect.StructField{
Name: typ.Field(i).Name,
Type: typ.Field(i).Type,
})
fieldMap[typ.Field(i).Name] = val.Field(i)
}
}
newtyp := reflect.StructOf(structFields)
newi := reflect.New(newtyp).Elem()
newty := reflect.TypeOf(newi.Addr().Interface()).Elem()
for i := 0; i < nums-1; i++ {
newi.Field(i).Set(fieldMap[newty.Field(i).Name])
}
return newi.Addr().Interface()
}

// getJavaClassName can read field JavaClassName of interface{}, used in cli-tool to do hessian registry
func getJavaClassName(pkg interface{}) (string, error) {
val := reflect.ValueOf(pkg).Elem()
typ := reflect.TypeOf(pkg).Elem()
nums := val.NumField()
for i := 0; i < nums; i++ {
if typ.Field(i).Name == "JavaClassName" {
return val.Field(i).String(), nil
}
}
return "", perrors.Errorf("JavaClassName field not found error")
}
33 changes: 33 additions & 0 deletions encoding/json/parser_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,33 @@
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/

package json

import (
"testing"
)

import (
"github.com/stretchr/testify/assert"
)

func Test_newJsonStructParser(t *testing.T) {
path := "./user.json"
hessianPair, _, err := File2Interface(path)
assert.NotEmpty(t, hessianPair)
assert.Nil(t, err)
}
12 changes: 12 additions & 0 deletions encoding/json/user.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
{
"ID": "string",
"Name": "string",
"Age": "int",
"JavaClassName": "string@com.ikurento.user.User",
"SubInfo": {
"SubID": "string",
"SubMale": "bool",
"SubAge": "int",
"JavaClassName":"string@com.ikurento.user.SubInfo"
}
}
1 change: 1 addition & 0 deletions go.mod
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@ module github.com/dubbogo/gost
require (
github.com/StackExchange/wmi v0.0.0-20190523213315-cbe66965904d // indirect
github.com/davecgh/go-spew v1.1.1
github.com/dubbogo/jsonparser v1.0.1
github.com/go-ole/go-ole v1.2.4 // indirect
github.com/k0kubun/colorstring v0.0.0-20150214042306-9440f1994b88 // indirect
github.com/k0kubun/pp v3.0.1+incompatible
Expand Down
2 changes: 2 additions & 0 deletions go.sum
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,8 @@ github.com/davecgh/go-spew v1.1.0 h1:ZDRjVQ15GmhC3fiQ8ni8+OwkZQO4DARzQgrnXU1Liz8
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
github.com/dubbogo/jsonparser v1.0.1 h1:sAIr8gk+gkahkIm6CnUxh9wTCkbgwLEQ8dTXTnAXyzo=
github.com/dubbogo/jsonparser v1.0.1/go.mod h1:tYAtpctvSP/tWw4MeelsowSPgXQRVHHWbqL6ynps8jU=
github.com/go-ole/go-ole v1.2.4 h1:nNBDSCOigTSiarFpYE9J/KtEA1IOW4CNeqT9TQDqCxI=
github.com/go-ole/go-ole v1.2.4/go.mod h1:XCwSNxSkXRo4vlyPy93sltvi/qJq0jqQhjqQNIwKuxM=
github.com/k0kubun/colorstring v0.0.0-20150214042306-9440f1994b88 h1:uC1QfSlInpQF+M0ao65imhwqKnz3Q2z/d8PWZRMQvDM=
Expand Down

0 comments on commit 6acc198

Please sign in to comment.