trie: new package (#3729)

This implements a simple trie tree for minio server/tools.
This package borrows idea from
https://godoc.org/golang.org/x/text/internal/triegen.
This commit is contained in:
Bala FA 2017-02-11 01:21:41 +05:30 committed by Harshavardhana
parent a799351926
commit 8912b6bf3b
4 changed files with 55 additions and 46 deletions

View File

@ -16,13 +16,16 @@
package cmd package cmd
import "github.com/minio/cli" import (
"github.com/minio/cli"
"github.com/minio/minio/pkg/trie"
)
// Collection of minio commands currently supported are. // Collection of minio commands currently supported are.
var commands = []cli.Command{} var commands = []cli.Command{}
// Collection of minio commands currently supported in a trie tree. // Collection of minio commands currently supported in a trie tree.
var commandsTree = newTrie() var commandsTree = trie.NewTrie()
// registerCommand registers a cli command. // registerCommand registers a cli command.
func registerCommand(command cli.Command) { func registerCommand(command cli.Command) {

View File

@ -94,7 +94,7 @@ func findClosestCommands(command string) []string {
sort.Strings(closestCommands) sort.Strings(closestCommands)
// Suggest other close commands - allow missed, wrongly added and // Suggest other close commands - allow missed, wrongly added and
// even transposed characters // even transposed characters
for _, value := range commandsTree.walk(commandsTree.root) { for _, value := range commandsTree.Walk(commandsTree.Root()) {
if sort.SearchStrings(closestCommands, value.(string)) < len(closestCommands) { if sort.SearchStrings(closestCommands, value.(string)) < len(closestCommands) {
continue continue
} }

View File

@ -1,5 +1,5 @@
/* /*
* Minio Cloud Storage, (C) 2015 Minio, Inc. * Minio Cloud Storage, (C) 2014, 2015, 2016, 2017 Minio, Inc.
* *
* Licensed under the Apache License, Version 2.0 (the "License"); * Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License. * you may not use this file except in compliance with the License.
@ -14,80 +14,78 @@
* limitations under the License. * limitations under the License.
*/ */
package cmd // Package trie implements a simple trie tree for minio server/tools borrows
// idea from - https://godoc.org/golang.org/x/text/internal/triegen.
package trie
// This package borrows idea from - https://godoc.org/golang.org/x/text/internal/triegen // Node trie tree node container carries value and children.
type Node struct {
exists bool
value interface{}
child map[rune]*Node // runes as child.
}
// Trie trie container // newNode create a new trie node.
func newNode() *Node {
return &Node{
exists: false,
value: nil,
child: make(map[rune]*Node),
}
}
// Trie is a trie container.
type Trie struct { type Trie struct {
root *trieNode root *Node
size int size int
} }
// newTrie get new trie // Root returns root node.
func newTrie() *Trie { func (t *Trie) Root() *Node {
return &Trie{ return t.root
root: newTrieNode(),
size: 0,
}
} }
// trieNode trie tree node container carries value and children // Insert insert a key.
type trieNode struct {
exists bool
value interface{}
child map[rune]*trieNode // runes as child
}
func newTrieNode() *trieNode {
return &trieNode{
exists: false,
value: nil,
child: make(map[rune]*trieNode),
}
}
// Insert insert a key
func (t *Trie) Insert(key string) { func (t *Trie) Insert(key string) {
curNode := t.root curNode := t.root
for _, v := range key { for _, v := range key {
if curNode.child[v] == nil { if curNode.child[v] == nil {
curNode.child[v] = newTrieNode() curNode.child[v] = newNode()
} }
curNode = curNode.child[v] curNode = curNode.child[v]
} }
if !curNode.exists { if !curNode.exists {
// increment when new rune child is added // increment when new rune child is added.
t.size++ t.size++
curNode.exists = true curNode.exists = true
} }
// value is stored for retrieval in future // value is stored for retrieval in future.
curNode.value = key curNode.value = key
} }
// PrefixMatch - prefix match // PrefixMatch - prefix match.
func (t *Trie) PrefixMatch(key string) []interface{} { func (t *Trie) PrefixMatch(key string) []interface{} {
node, _ := t.findNode(key) node, _ := t.findNode(key)
if node != nil { if node != nil {
return t.walk(node) return t.Walk(node)
} }
return []interface{}{} return []interface{}{}
} }
// walk the tree // Walk the tree.
func (t *Trie) walk(node *trieNode) (ret []interface{}) { func (t *Trie) Walk(node *Node) (ret []interface{}) {
if node.exists { if node.exists {
ret = append(ret, node.value) ret = append(ret, node.value)
} }
for _, v := range node.child { for _, v := range node.child {
ret = append(ret, t.walk(v)...) ret = append(ret, t.Walk(v)...)
} }
return return
} }
// find nodes corresponding to key // find nodes corresponding to key.
func (t *Trie) findNode(key string) (node *trieNode, index int) { func (t *Trie) findNode(key string) (node *Node, index int) {
curNode := t.root curNode := t.root
f := false f := false
for k, v := range key { for k, v := range key {
@ -110,3 +108,11 @@ func (t *Trie) findNode(key string) (node *trieNode, index int) {
return curNode, index return curNode, index
} }
// NewTrie create a new trie.
func NewTrie() *Trie {
return &Trie{
root: newNode(),
size: 0,
}
}

View File

@ -14,7 +14,7 @@
* limitations under the License. * limitations under the License.
*/ */
package cmd package trie
import ( import (
"testing" "testing"
@ -23,7 +23,7 @@ import (
// Simply make sure creating a new tree works. // Simply make sure creating a new tree works.
func TestNewTrie(t *testing.T) { func TestNewTrie(t *testing.T) {
var trie *Trie var trie *Trie
trie = newTrie() trie = NewTrie()
if trie.size != 0 { if trie.size != 0 {
t.Errorf("expected size 0, got: %d", trie.size) t.Errorf("expected size 0, got: %d", trie.size)
@ -33,7 +33,7 @@ func TestNewTrie(t *testing.T) {
// Ensure that we can insert new keys into the tree, then check the size. // Ensure that we can insert new keys into the tree, then check the size.
func TestInsert(t *testing.T) { func TestInsert(t *testing.T) {
var trie *Trie var trie *Trie
trie = newTrie() trie = NewTrie()
// We need to have an empty tree to begin with. // We need to have an empty tree to begin with.
if trie.size != 0 { if trie.size != 0 {
@ -52,10 +52,10 @@ func TestInsert(t *testing.T) {
// Ensure that PrefixMatch gives us the correct two keys in the tree. // Ensure that PrefixMatch gives us the correct two keys in the tree.
func TestPrefixMatch(t *testing.T) { func TestPrefixMatch(t *testing.T) {
var trie *Trie var trie *Trie
trie = newTrie() trie = NewTrie()
// Feed it some fodder: only 'minio' and 'miny-os' should trip the matcher. // Feed it some fodder: only 'minio' and 'miny-os' should trip the matcher.
trie.Insert(globalMinioDefaultOwnerID) trie.Insert("minio")
trie.Insert("amazon") trie.Insert("amazon")
trie.Insert("cheerio") trie.Insert("cheerio")
trie.Insert("miny-o's") trie.Insert("miny-o's")
@ -65,7 +65,7 @@ func TestPrefixMatch(t *testing.T) {
t.Errorf("expected two matches, got: %d", len(matches)) t.Errorf("expected two matches, got: %d", len(matches))
} }
if matches[0] != globalMinioDefaultOwnerID && matches[1] != globalMinioDefaultOwnerID { if matches[0] != "minio" && matches[1] != "minio" {
t.Errorf("expected one match to be 'minio', got: '%s' and '%s'", matches[0], matches[1]) t.Errorf("expected one match to be 'minio', got: '%s' and '%s'", matches[0], matches[1])
} }
} }