2019-12-18 11:26:20 +01:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2019 Alex aka mailoman <alex@webz.asia>
|
|
|
|
*
|
|
|
|
* Licensed 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.
|
|
|
|
*
|
|
|
|
* @author Alex aka mailoman <alex@webz.asia>
|
|
|
|
* @copyright Copyright (c) 2019 Alex aka mailoman <alex@webz.asia>
|
|
|
|
* @since 18.12.2019
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2018-09-01 17:18:38 +02:00
|
|
|
package echoswagger
|
|
|
|
|
|
|
|
import (
|
|
|
|
"net/url"
|
|
|
|
"reflect"
|
|
|
|
"regexp"
|
|
|
|
"time"
|
|
|
|
)
|
|
|
|
|
|
|
|
var emailRegexp = regexp.MustCompile("^[a-zA-Z0-9.!#$%&'*+/=?^_`{|}~-]+@[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?(?:\\.[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?)*$")
|
|
|
|
|
|
|
|
func isValidEmail(str string) bool {
|
|
|
|
return emailRegexp.MatchString(str)
|
|
|
|
}
|
|
|
|
|
|
|
|
// See: https://github.com/swagger-api/swagger-js/blob/7414ad062ba9b6d9cc397c72e7561ec775b35a9f/lib/shred/parseUri.js#L28
|
|
|
|
func isValidURL(str string) bool {
|
|
|
|
if _, err := url.ParseRequestURI(str); err != nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
func isValidParam(t reflect.Type, nest, inner bool) bool {
|
|
|
|
if t == nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
switch t.Kind() {
|
|
|
|
case reflect.Bool, reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64,
|
|
|
|
reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr,
|
|
|
|
reflect.Float32, reflect.Float64, reflect.String:
|
|
|
|
if !nest || (nest && inner) {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
case reflect.Array, reflect.Slice:
|
|
|
|
return isValidParam(t.Elem(), nest, true)
|
|
|
|
case reflect.Ptr:
|
|
|
|
return isValidParam(t.Elem(), nest, inner)
|
|
|
|
case reflect.Struct:
|
2018-09-03 16:52:30 +02:00
|
|
|
if t == reflect.TypeOf(time.Time{}) && (!nest || nest && inner) {
|
2018-09-01 17:18:38 +02:00
|
|
|
return true
|
|
|
|
} else if !inner {
|
|
|
|
for i := 0; i < t.NumField(); i++ {
|
|
|
|
if !isValidParam(t.Field(i).Type, nest, true) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
// isValidSchema reports a type is valid for body param.
|
|
|
|
// valid case:
|
|
|
|
// 1. Struct
|
|
|
|
// 2. Struct{ A int64 }
|
|
|
|
// 3. *[]Struct
|
|
|
|
// 4. [][]Struct
|
|
|
|
// 5. []Struct{ A []Struct }
|
|
|
|
// 6. []Struct{ A Map[string]string }
|
|
|
|
// 7. *Struct{ A []Map[int64]Struct }
|
|
|
|
// 8. Map[string]string
|
|
|
|
// 9. []int64
|
|
|
|
// invalid case:
|
|
|
|
// 1. interface{}
|
|
|
|
// 2. Map[Struct]string
|
2018-09-18 05:59:50 +02:00
|
|
|
func isValidSchema(t reflect.Type, inner bool, pres ...reflect.Type) bool {
|
2018-09-01 17:18:38 +02:00
|
|
|
if t == nil {
|
|
|
|
return false
|
|
|
|
}
|
2018-09-18 05:59:50 +02:00
|
|
|
for _, pre := range pres {
|
|
|
|
if t == pre {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-01 17:18:38 +02:00
|
|
|
switch t.Kind() {
|
|
|
|
case reflect.Bool, reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64,
|
|
|
|
reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr,
|
2019-01-07 06:14:57 +01:00
|
|
|
reflect.Float32, reflect.Float64, reflect.String, reflect.Interface:
|
2018-09-01 17:18:38 +02:00
|
|
|
return true
|
|
|
|
case reflect.Array, reflect.Slice:
|
2018-09-18 05:59:50 +02:00
|
|
|
return isValidSchema(t.Elem(), inner, pres...)
|
2018-09-01 17:18:38 +02:00
|
|
|
case reflect.Map:
|
2018-09-18 05:59:50 +02:00
|
|
|
return isBasicType(t.Key()) && isValidSchema(t.Elem(), true, pres...)
|
2018-09-01 17:18:38 +02:00
|
|
|
case reflect.Ptr:
|
2018-09-18 05:59:50 +02:00
|
|
|
return isValidSchema(t.Elem(), inner, pres...)
|
2018-09-01 17:18:38 +02:00
|
|
|
case reflect.Struct:
|
2018-09-18 05:59:50 +02:00
|
|
|
pres = append(pres, t)
|
2018-09-01 17:18:38 +02:00
|
|
|
if t == reflect.TypeOf(time.Time{}) {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
for i := 0; i < t.NumField(); i++ {
|
2018-09-18 05:59:50 +02:00
|
|
|
if !isValidSchema(t.Field(i).Type, true, pres...) {
|
2018-09-01 17:18:38 +02:00
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
func isBasicType(t reflect.Type) bool {
|
|
|
|
if t == nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
switch t.Kind() {
|
|
|
|
case reflect.Bool, reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64,
|
|
|
|
reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr,
|
|
|
|
reflect.Float32, reflect.Float64, reflect.String:
|
|
|
|
return true
|
|
|
|
case reflect.Ptr:
|
|
|
|
return isBasicType(t.Elem())
|
|
|
|
case reflect.Struct:
|
|
|
|
if t == reflect.TypeOf(time.Time{}) {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
2018-09-16 10:35:29 +02:00
|
|
|
|
|
|
|
func isValidScheme(s string) bool {
|
|
|
|
if s == "http" || s == "https" || s == "ws" || s == "wss" {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|