2018-04-26 11:03:44 +00:00
|
|
|
package parser
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"strings"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Build builds an URI scheme from a pattern string
|
|
|
|
func Build(s string) (*Scheme, error){
|
|
|
|
|
|
|
|
/* (1) Manage '/' at the start */
|
|
|
|
if len(s) < 1 || s[0] != '/' {
|
|
|
|
return nil, fmt.Errorf("URI must begin with '/'")
|
|
|
|
}
|
|
|
|
|
|
|
|
/* (2) Split by '/' */
|
|
|
|
parts := strings.Split(s, "/")
|
|
|
|
|
|
|
|
/* (3) Max exceeded */
|
|
|
|
if len(parts)-2 > maxMatch {
|
|
|
|
for i, p := range parts {
|
|
|
|
fmt.Printf("%d: '%s'\n", i, p);
|
|
|
|
}
|
|
|
|
return nil, fmt.Errorf("URI must not exceed %d slash-separated components, got %d", maxMatch, len(parts))
|
|
|
|
}
|
|
|
|
|
|
|
|
/* (4) Build for each part */
|
|
|
|
sch, err := buildScheme(parts)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2018-04-26 13:01:18 +00:00
|
|
|
/* (5) Optimise structure */
|
2018-04-26 11:03:44 +00:00
|
|
|
opti, err := sch.optimise()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &opti, nil
|
|
|
|
|
2018-04-26 13:01:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Match returns if the given URI is matched by the scheme
|
|
|
|
func (s Scheme) Match(str string) bool {
|
|
|
|
|
|
|
|
/* (1) Nothing -> match all */
|
|
|
|
if len(s) == 0 { return true }
|
|
|
|
|
2018-04-26 20:15:09 +00:00
|
|
|
/* (2) Check for string match */
|
|
|
|
clearURI, match := s.matchString(str)
|
|
|
|
if !match {
|
2018-04-26 13:01:18 +00:00
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2018-04-26 20:15:09 +00:00
|
|
|
/* (3) Check for non-string match (wildcards) */
|
|
|
|
match = s.matchWildcards(clearURI)
|
|
|
|
if !match {
|
|
|
|
return false
|
2018-04-26 13:01:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// GetMatch returns the indexed match (excluding string matchers)
|
|
|
|
func (s Scheme) GetMatch(n uint8) ([]string, error) {
|
|
|
|
|
|
|
|
/* (1) Index out of range */
|
|
|
|
if n > uint8(len(s)) {
|
|
|
|
return nil, fmt.Errorf("Index out of range")
|
|
|
|
}
|
|
|
|
|
|
|
|
/* (2) Iterate to find index (exclude strings) */
|
|
|
|
ni := -1
|
2018-04-26 21:39:10 +00:00
|
|
|
for _, m := range s {
|
2018-04-26 13:01:18 +00:00
|
|
|
|
|
|
|
// ignore strings
|
2018-04-26 21:39:10 +00:00
|
|
|
if len(m.pat) > 0 { continue }
|
2018-04-26 13:01:18 +00:00
|
|
|
|
|
|
|
// increment match counter : ni
|
|
|
|
ni++
|
|
|
|
|
|
|
|
// if expected index -> return matches
|
|
|
|
if uint8(ni) == n {
|
2018-04-26 21:39:10 +00:00
|
|
|
return m.buf, nil
|
2018-04-26 13:01:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
/* (3) If nothing found -> return empty set */
|
|
|
|
return nil, fmt.Errorf("Index out of range (max: %d)", ni)
|
|
|
|
|
2018-04-26 21:39:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// GetAllMatch returns all the indexed match (excluding string matchers)
|
|
|
|
func (s Scheme) GetAllMatch() [][]string {
|
|
|
|
|
|
|
|
match := make([][]string, 0, len(s))
|
|
|
|
|
|
|
|
for _, m := range s {
|
|
|
|
|
|
|
|
// ignore strings
|
|
|
|
if len(m.pat) > 0 { continue }
|
|
|
|
|
|
|
|
match = append(match, m.buf)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return match
|
|
|
|
|
2018-04-26 11:03:44 +00:00
|
|
|
}
|