package macro import ( "strconv" "strings" "github.com/kataras/iris/v12/macro/interpreter/ast" ) var ( // String type // Allows anything (single path segment, as everything except the `Path`). // Its functions can be used by the rest of the macros and param types whenever not available function by name is used. // Because of its "master" boolean value to true (third parameter). String = NewMacro("string", "", true, false, nil). RegisterFunc("regexp", MustRegexp). // checks if param value starts with the 'prefix' arg RegisterFunc("prefix", func(prefix string) func(string) bool { return func(paramValue string) bool { return strings.HasPrefix(paramValue, prefix) } }). // checks if param value ends with the 'suffix' arg RegisterFunc("suffix", func(suffix string) func(string) bool { return func(paramValue string) bool { return strings.HasSuffix(paramValue, suffix) } }). // checks if param value contains the 's' arg RegisterFunc("contains", func(s string) func(string) bool { return func(paramValue string) bool { return strings.Contains(paramValue, s) } }). // checks if param value's length is at least 'min' RegisterFunc("min", func(min int) func(string) bool { return func(paramValue string) bool { return len(paramValue) >= min } }). // checks if param value's length is not bigger than 'max' RegisterFunc("max", func(max int) func(string) bool { return func(paramValue string) bool { return max >= len(paramValue) } }) simpleNumberEval = MustRegexp("^-?[0-9]+$") // Int or number type // both positive and negative numbers, actual value can be min-max int64 or min-max int32 depends on the arch. // If x64: -9223372036854775808 to 9223372036854775807. // If x32: -2147483648 to 2147483647 and etc.. Int = NewMacro("int", "number", false, false, func(paramValue string) (interface{}, bool) { if !simpleNumberEval(paramValue) { return nil, false } v, err := strconv.Atoi(paramValue) if err != nil { return nil, false } return v, true }). // checks if the param value's int representation is // bigger or equal than 'min' RegisterFunc("min", func(min int) func(int) bool { return func(paramValue int) bool { return paramValue >= min } }). // checks if the param value's int representation is // smaller or equal than 'max'. RegisterFunc("max", func(max int) func(int) bool { return func(paramValue int) bool { return paramValue <= max } }). // checks if the param value's int representation is // between min and max, including 'min' and 'max'. RegisterFunc("range", func(min, max int) func(int) bool { return func(paramValue int) bool { return !(paramValue < min || paramValue > max) } }) // Int8 type // -128 to 127. Int8 = NewMacro("int8", "", false, false, func(paramValue string) (interface{}, bool) { if !simpleNumberEval(paramValue) { return nil, false } v, err := strconv.ParseInt(paramValue, 10, 8) if err != nil { return nil, false } return int8(v), true }). RegisterFunc("min", func(min int8) func(int8) bool { return func(paramValue int8) bool { return paramValue >= min } }). RegisterFunc("max", func(max int8) func(int8) bool { return func(paramValue int8) bool { return paramValue <= max } }). RegisterFunc("range", func(min, max int8) func(int8) bool { return func(paramValue int8) bool { return !(paramValue < min || paramValue > max) } }) // Int16 type // -32768 to 32767. Int16 = NewMacro("int16", "", false, false, func(paramValue string) (interface{}, bool) { if !simpleNumberEval(paramValue) { return nil, false } v, err := strconv.ParseInt(paramValue, 10, 16) if err != nil { return nil, false } return int16(v), true }). RegisterFunc("min", func(min int16) func(int16) bool { return func(paramValue int16) bool { return paramValue >= min } }). RegisterFunc("max", func(max int16) func(int16) bool { return func(paramValue int16) bool { return paramValue <= max } }). RegisterFunc("range", func(min, max int16) func(int16) bool { return func(paramValue int16) bool { return !(paramValue < min || paramValue > max) } }) // Int32 type // -2147483648 to 2147483647. Int32 = NewMacro("int32", "", false, false, func(paramValue string) (interface{}, bool) { if !simpleNumberEval(paramValue) { return nil, false } v, err := strconv.ParseInt(paramValue, 10, 32) if err != nil { return nil, false } return int32(v), true }). RegisterFunc("min", func(min int32) func(int32) bool { return func(paramValue int32) bool { return paramValue >= min } }). RegisterFunc("max", func(max int32) func(int32) bool { return func(paramValue int32) bool { return paramValue <= max } }). RegisterFunc("range", func(min, max int32) func(int32) bool { return func(paramValue int32) bool { return !(paramValue < min || paramValue > max) } }) // Int64 as int64 type // -9223372036854775808 to 9223372036854775807. Int64 = NewMacro("int64", "long", false, false, func(paramValue string) (interface{}, bool) { if !simpleNumberEval(paramValue) { return nil, false } v, err := strconv.ParseInt(paramValue, 10, 64) if err != nil { // if err == strconv.ErrRange... return nil, false } return v, true }). // checks if the param value's int64 representation is // bigger or equal than 'min'. RegisterFunc("min", func(min int64) func(int64) bool { return func(paramValue int64) bool { return paramValue >= min } }). // checks if the param value's int64 representation is // smaller or equal than 'max'. RegisterFunc("max", func(max int64) func(int64) bool { return func(paramValue int64) bool { return paramValue <= max } }). // checks if the param value's int64 representation is // between min and max, including 'min' and 'max'. RegisterFunc("range", func(min, max int64) func(int64) bool { return func(paramValue int64) bool { return !(paramValue < min || paramValue > max) } }) // Uint as uint type // actual value can be min-max uint64 or min-max uint32 depends on the arch. // If x64: 0 to 18446744073709551615. // If x32: 0 to 4294967295 and etc. Uint = NewMacro("uint", "", false, false, func(paramValue string) (interface{}, bool) { v, err := strconv.ParseUint(paramValue, 10, strconv.IntSize) // 32,64... if err != nil { return nil, false } return uint(v), true }). // checks if the param value's int representation is // bigger or equal than 'min' RegisterFunc("min", func(min uint) func(uint) bool { return func(paramValue uint) bool { return paramValue >= min } }). // checks if the param value's int representation is // smaller or equal than 'max'. RegisterFunc("max", func(max uint) func(uint) bool { return func(paramValue uint) bool { return paramValue <= max } }). // checks if the param value's int representation is // between min and max, including 'min' and 'max'. RegisterFunc("range", func(min, max uint) func(uint) bool { return func(paramValue uint) bool { return !(paramValue < min || paramValue > max) } }) uint8Eval = MustRegexp("^([0-9]|[1-8][0-9]|9[0-9]|1[0-9]{2}|2[0-4][0-9]|25[0-5])$") // Uint8 as uint8 type // 0 to 255. Uint8 = NewMacro("uint8", "", false, false, func(paramValue string) (interface{}, bool) { if !uint8Eval(paramValue) { return nil, false } v, err := strconv.ParseUint(paramValue, 10, 8) if err != nil { return nil, false } return uint8(v), true }). // checks if the param value's uint8 representation is // bigger or equal than 'min'. RegisterFunc("min", func(min uint8) func(uint8) bool { return func(paramValue uint8) bool { return paramValue >= min } }). // checks if the param value's uint8 representation is // smaller or equal than 'max'. RegisterFunc("max", func(max uint8) func(uint8) bool { return func(paramValue uint8) bool { return paramValue <= max } }). // checks if the param value's uint8 representation is // between min and max, including 'min' and 'max'. RegisterFunc("range", func(min, max uint8) func(uint8) bool { return func(paramValue uint8) bool { return !(paramValue < min || paramValue > max) } }) // Uint16 as uint16 type // 0 to 65535. Uint16 = NewMacro("uint16", "", false, false, func(paramValue string) (interface{}, bool) { v, err := strconv.ParseUint(paramValue, 10, 16) if err != nil { return nil, false } return uint16(v), true }). RegisterFunc("min", func(min uint16) func(uint16) bool { return func(paramValue uint16) bool { return paramValue >= min } }). RegisterFunc("max", func(max uint16) func(uint16) bool { return func(paramValue uint16) bool { return paramValue <= max } }). RegisterFunc("range", func(min, max uint16) func(uint16) bool { return func(paramValue uint16) bool { return !(paramValue < min || paramValue > max) } }) // Uint32 as uint32 type // 0 to 4294967295. Uint32 = NewMacro("uint32", "", false, false, func(paramValue string) (interface{}, bool) { v, err := strconv.ParseUint(paramValue, 10, 32) if err != nil { return nil, false } return uint32(v), true }). RegisterFunc("min", func(min uint32) func(uint32) bool { return func(paramValue uint32) bool { return paramValue >= min } }). RegisterFunc("max", func(max uint32) func(uint32) bool { return func(paramValue uint32) bool { return paramValue <= max } }). RegisterFunc("range", func(min, max uint32) func(uint32) bool { return func(paramValue uint32) bool { return !(paramValue < min || paramValue > max) } }) // Uint64 as uint64 type // 0 to 18446744073709551615. Uint64 = NewMacro("uint64", "", false, false, func(paramValue string) (interface{}, bool) { v, err := strconv.ParseUint(paramValue, 10, 64) if err != nil { return nil, false } return v, true }). // checks if the param value's uint64 representation is // bigger or equal than 'min'. RegisterFunc("min", func(min uint64) func(uint64) bool { return func(paramValue uint64) bool { return paramValue >= min } }). // checks if the param value's uint64 representation is // smaller or equal than 'max'. RegisterFunc("max", func(max uint64) func(uint64) bool { return func(paramValue uint64) bool { return paramValue <= max } }). // checks if the param value's uint64 representation is // between min and max, including 'min' and 'max'. RegisterFunc("range", func(min, max uint64) func(uint64) bool { return func(paramValue uint64) bool { return !(paramValue < min || paramValue > max) } }) // Bool or boolean as bool type // a string which is "1" or "t" or "T" or "TRUE" or "true" or "True" // or "0" or "f" or "F" or "FALSE" or "false" or "False". Bool = NewMacro("bool", "boolean", false, false, func(paramValue string) (interface{}, bool) { // a simple if statement is faster than regex ^(true|false|True|False|t|0|f|FALSE|TRUE)$ // in this case. v, err := strconv.ParseBool(paramValue) if err != nil { return nil, false } return v, true }) alphabeticalEval = MustRegexp("^[a-zA-Z ]+$") // Alphabetical letter type // letters only (upper or lowercase) Alphabetical = NewMacro("alphabetical", "", false, false, func(paramValue string) (interface{}, bool) { if !alphabeticalEval(paramValue) { return nil, false } return paramValue, true }) fileEval = MustRegexp("^[a-zA-Z0-9_.-]*$") // File type // letters (upper or lowercase) // numbers (0-9) // underscore (_) // dash (-) // point (.) // no spaces! or other character File = NewMacro("file", "", false, false, func(paramValue string) (interface{}, bool) { if !fileEval(paramValue) { return nil, false } return paramValue, true }) // Path type // anything, should be the last part // // It allows everything, we have String and Path as different // types because I want to give the opportunity to the user // to organise the macro functions based on wildcard or single dynamic named path parameter. // Should be living in the latest path segment of a route path. Path = NewMacro("path", "", false, true, nil) // Defaults contains the defaults macro and parameters types for the router. // // Read https://github.com/kataras/iris/tree/master/_examples/routing/macros for more details. Defaults = &Macros{ String, Int, Int8, Int16, Int32, Int64, Uint, Uint8, Uint16, Uint32, Uint64, Bool, Alphabetical, File, Path, } ) // Macros is just a type of a slice of *Macro // which is responsible to register and search for macros based on the indent(parameter type). type Macros []*Macro // Register registers a custom Macro. // The "indent" should not be empty and should be unique, it is the parameter type's name, i.e "string". // The "alias" is optionally and it should be unique, it is the alias of the parameter type. // "isMaster" and "isTrailing" is for default parameter type and wildcard respectfully. // The "evaluator" is the function that is converted to an Iris handler which is executed every time // before the main chain of a route's handlers that contains this macro of the specific parameter type. // // Read https://github.com/kataras/iris/tree/master/_examples/routing/macros for more details. func (ms *Macros) Register(indent, alias string, isMaster, isTrailing bool, evaluator ParamEvaluator) *Macro { macro := NewMacro(indent, alias, isMaster, isTrailing, evaluator) if ms.register(macro) { return macro } return nil } func (ms *Macros) register(macro *Macro) bool { if macro.Indent() == "" { return false } cp := *ms for _, m := range cp { // can't add more than one with the same ast characteristics. if macro.Indent() == m.Indent() { return false } if alias := macro.Alias(); alias != "" { if alias == m.Alias() || alias == m.Indent() { return false } } if macro.Master() && m.Master() { return false } } cp = append(cp, macro) *ms = cp return true } // Unregister removes a macro and its parameter type from the list. func (ms *Macros) Unregister(indent string) bool { cp := *ms for i, m := range cp { if m.Indent() == indent { copy(cp[i:], cp[i+1:]) cp[len(cp)-1] = nil cp = cp[:len(cp)-1] *ms = cp return true } } return false } // Lookup returns the responsible macro for a parameter type, it can return nil. func (ms *Macros) Lookup(pt ast.ParamType) *Macro { if m := ms.Get(pt.Indent()); m != nil { return m } if alias, has := ast.HasAlias(pt); has { if m := ms.Get(alias); m != nil { return m } } return nil } // Get returns the responsible macro for a parameter type, it can return nil. func (ms *Macros) Get(indentOrAlias string) *Macro { if indentOrAlias == "" { return nil } for _, m := range *ms { if m.Indent() == indentOrAlias { return m } if m.Alias() == indentOrAlias { return m } } return nil } // GetMaster returns the default macro and its parameter type, // by default it will return the `String` macro which is responsible for the "string" parameter type. func (ms *Macros) GetMaster() *Macro { for _, m := range *ms { if m.Master() { return m } } return nil } // GetTrailings returns the macros that have support for wildcards parameter types. // By default it will return the `Path` macro which is responsible for the "path" parameter type. func (ms *Macros) GetTrailings() (macros []*Macro) { for _, m := range *ms { if m.Trailing() { macros = append(macros, m) } } return }