Skip to content

Commit 566a399

Browse files
authored
feat: 增加选择器对转义的支持 (#9)
1 parent afdc17d commit 566a399

File tree

2 files changed

+57
-20
lines changed

2 files changed

+57
-20
lines changed

ognl.go

+37-20
Original file line numberDiff line numberDiff line change
@@ -318,16 +318,9 @@ func GetE(value interface{}, path string) (Result, error) {
318318
}
319319

320320
default:
321-
start := index
322-
loop:
323-
for ; index < len(path); index++ {
324-
switch path[index] {
325-
case '.', '#':
326-
index--
327-
break loop
328-
}
329-
}
330-
sv := path[start:min(index+1, len(path))]
321+
key, newIndex := parseNextKey(path, index)
322+
index = newIndex
323+
sv := key
331324
v, err := strconv.Atoi(sv)
332325
digit := err == nil && v >= 0
333326
switch result.deployment {
@@ -449,16 +442,9 @@ func Get(value interface{}, path string) Result {
449442
}
450443

451444
default:
452-
start := index
453-
loop:
454-
for ; index < len(path); index++ {
455-
switch path[index] {
456-
case '.', '#':
457-
index--
458-
break loop
459-
}
460-
}
461-
sv := path[start:min(index+1, len(path))]
445+
key, newIndex := parseNextKey(path, index)
446+
index = newIndex
447+
sv := key
462448
v, err := strconv.Atoi(sv)
463449
digit := err == nil && v >= 0
464450
switch result.deployment {
@@ -508,6 +494,37 @@ func Get(value interface{}, path string) Result {
508494
return result
509495
}
510496

497+
// 从选择器中解析出下一个要处理的key
498+
// params:
499+
// selector: 路径选择器,比如"Foo.Bar.Name",比如"Foo\\.Bar\\.Name"
500+
// index: 选择器上次消费到的位置
501+
//
502+
// returns:
503+
// string 下一个要解析的key
504+
// index selector被消费到的位置
505+
func parseNextKey(selector string, index int) (string, int) {
506+
key := make([]byte, 0)
507+
loop:
508+
for ; index < len(selector); index++ {
509+
switch selector[index] {
510+
case '\\':
511+
// 先跳过转义字符
512+
index++
513+
// 转义字符,无论下一个字符是什么都跳过,如果有的话
514+
if index < len(selector) {
515+
key = append(key, selector[index])
516+
}
517+
case '.', '#':
518+
index--
519+
break loop
520+
default:
521+
// 普通字符,当做key的一部分消费掉
522+
key = append(key, selector[index])
523+
}
524+
}
525+
return string(key), index
526+
}
527+
511528
func GetMany(value interface{}, path ...string) []Result {
512529
results := make([]Result, 0, len(path))
513530
for _, s := range path {

ognl_test.go

+20
Original file line numberDiff line numberDiff line change
@@ -408,3 +408,23 @@ func TestBaseType(t *testing.T) {
408408
assert.Equal(t, int(rv.Kind()), int(tp))
409409
}
410410
}
411+
412+
type foo struct {
413+
Bar *bar
414+
}
415+
416+
type bar struct {
417+
Name string
418+
}
419+
420+
// 测试选择器转义字符
421+
func TestEscape(t *testing.T) {
422+
m := make(map[string]*foo)
423+
t1 := &foo{Bar: &bar{Name: "bar name 001"}}
424+
m["Foo.Bar.Name"] = t1
425+
m["Foo"] = &foo{Bar: &bar{Name: "bar name 002"}}
426+
427+
assert.Equal(t, t1, Get(m, "Foo\\.Bar\\.Name").Value())
428+
assert.Equal(t, "bar name 002", Get(m, "Foo.Bar.Name").Value())
429+
430+
}

0 commit comments

Comments
 (0)