2014-03-06 02:21:44 -05:00
// Copyright 2014 The Gogs Authors. All rights reserved.
2019-04-25 18:42:50 -04:00
// Copyright 2019 The Gitea Authors. All rights reserved.
2014-03-06 02:21:44 -05:00
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file.
package auth
import (
"reflect"
2014-07-26 00:24:27 -04:00
"strings"
2014-03-06 02:21:44 -05:00
2016-11-10 17:24:48 +01:00
"code.gitea.io/gitea/models"
2019-11-23 01:33:31 +02:00
"code.gitea.io/gitea/modules/auth/sso"
2017-04-19 06:02:20 +03:00
"code.gitea.io/gitea/modules/validation"
2019-08-23 09:40:30 -07:00
"gitea.com/macaron/binding"
"gitea.com/macaron/macaron"
"gitea.com/macaron/session"
"github.com/unknwon/com"
2014-03-06 02:21:44 -05:00
)
2016-11-27 14:03:59 +08:00
// IsAPIPath if URL is an api path
2015-07-15 19:17:57 +08:00
func IsAPIPath ( url string ) bool {
return strings . HasPrefix ( url , "/api/" )
}
2014-07-26 00:24:27 -04:00
// SignedInUser returns the user object of signed user.
2014-12-05 17:54:57 -05:00
// It returns a bool value to indicate whether user uses basic auth or not.
2015-09-02 02:40:15 -04:00
func SignedInUser ( ctx * macaron . Context , sess session . Store ) ( * models . User , bool ) {
2014-11-10 05:30:07 -05:00
if ! models . HasEngine {
2014-11-18 11:07:16 -05:00
return nil , false
2014-11-10 05:30:07 -05:00
}
2019-11-23 01:33:31 +02:00
// Try to sign in with each of the enabled plugins
for _ , ssoMethod := range sso . Methods ( ) {
if ! ssoMethod . IsEnabled ( ) {
continue
2017-02-05 08:10:46 -05:00
}
2019-11-23 01:33:31 +02:00
user := ssoMethod . VerifyAuthData ( ctx , sess )
if user != nil {
_ , isBasic := ssoMethod . ( * sso . Basic )
return user , isBasic
2014-11-10 05:30:07 -05:00
}
2017-02-05 08:10:46 -05:00
}
2014-11-10 05:30:07 -05:00
2017-02-05 08:10:46 -05:00
return nil , false
2014-03-28 18:40:31 -04:00
}
2016-11-27 14:03:59 +08:00
// Form form binding interface
2014-10-15 11:19:20 -04:00
type Form interface {
binding . Validator
}
2015-02-01 12:41:03 -05:00
func init ( ) {
binding . SetNameMapper ( com . ToSnakeCase )
}
2014-07-26 00:24:27 -04:00
// AssignForm assign form values back to the template data.
func AssignForm ( form interface { } , data map [ string ] interface { } ) {
typ := reflect . TypeOf ( form )
val := reflect . ValueOf ( form )
if typ . Kind ( ) == reflect . Ptr {
typ = typ . Elem ( )
val = val . Elem ( )
}
for i := 0 ; i < typ . NumField ( ) ; i ++ {
field := typ . Field ( i )
fieldName := field . Tag . Get ( "form" )
// Allow ignored fields in the struct
if fieldName == "-" {
continue
2015-02-01 12:41:03 -05:00
} else if len ( fieldName ) == 0 {
fieldName = com . ToSnakeCase ( field . Name )
2014-07-26 00:24:27 -04:00
}
data [ fieldName ] = val . Field ( i ) . Interface ( )
}
}
2015-10-29 21:09:48 -04:00
func getRuleBody ( field reflect . StructField , prefix string ) string {
2014-07-26 00:24:27 -04:00
for _ , rule := range strings . Split ( field . Tag . Get ( "binding" ) , ";" ) {
2014-08-01 06:12:14 -04:00
if strings . HasPrefix ( rule , prefix ) {
2017-03-03 16:21:31 +08:00
return rule [ len ( prefix ) : len ( rule ) - 1 ]
2014-07-26 00:24:27 -04:00
}
}
return ""
}
2016-11-27 14:03:59 +08:00
// GetSize get size int form tag
2015-07-24 21:02:49 +08:00
func GetSize ( field reflect . StructField ) string {
2015-10-29 21:09:48 -04:00
return getRuleBody ( field , "Size(" )
2015-07-24 21:02:49 +08:00
}
2016-11-27 14:03:59 +08:00
// GetMinSize get minimal size in form tag
2014-08-01 06:12:14 -04:00
func GetMinSize ( field reflect . StructField ) string {
2015-10-29 21:09:48 -04:00
return getRuleBody ( field , "MinSize(" )
2014-08-01 06:12:14 -04:00
}
2016-11-27 14:03:59 +08:00
// GetMaxSize get max size in form tag
2014-08-01 06:12:14 -04:00
func GetMaxSize ( field reflect . StructField ) string {
2015-10-29 21:09:48 -04:00
return getRuleBody ( field , "MaxSize(" )
}
2016-11-27 14:03:59 +08:00
// GetInclude get include in form tag
2015-10-29 21:09:48 -04:00
func GetInclude ( field reflect . StructField ) string {
return getRuleBody ( field , "Include(" )
2014-08-01 06:12:14 -04:00
}
2014-10-15 11:19:20 -04:00
func validate ( errs binding . Errors , data map [ string ] interface { } , f Form , l macaron . Locale ) binding . Errors {
if errs . Len ( ) == 0 {
return errs
2014-07-26 00:24:27 -04:00
}
data [ "HasError" ] = true
2019-09-09 08:48:21 +03:00
// If the field with name errs[0].FieldNames[0] is not found in form
// somehow, some code later on will panic on Data["ErrorMsg"].(string).
// So initialize it to some default.
data [ "ErrorMsg" ] = l . Tr ( "form.unknown_error" )
2014-07-26 00:24:27 -04:00
AssignForm ( f , data )
typ := reflect . TypeOf ( f )
val := reflect . ValueOf ( f )
if typ . Kind ( ) == reflect . Ptr {
typ = typ . Elem ( )
val = val . Elem ( )
}
2019-09-09 08:48:21 +03:00
if field , ok := typ . FieldByName ( errs [ 0 ] . FieldNames [ 0 ] ) ; ok {
2014-07-26 00:24:27 -04:00
fieldName := field . Tag . Get ( "form" )
2019-09-09 08:48:21 +03:00
if fieldName != "-" {
2017-03-03 16:21:31 +08:00
data [ "Err_" + field . Name ] = true
2015-07-08 19:47:56 +08:00
trName := field . Tag . Get ( "locale" )
if len ( trName ) == 0 {
trName = l . Tr ( "form." + field . Name )
} else {
trName = l . Tr ( trName )
}
2014-10-15 11:19:20 -04:00
switch errs [ 0 ] . Classification {
2014-12-15 01:49:59 -05:00
case binding . ERR_REQUIRED :
2014-07-26 00:24:27 -04:00
data [ "ErrorMsg" ] = trName + l . Tr ( "form.require_error" )
2014-12-15 01:49:59 -05:00
case binding . ERR_ALPHA_DASH :
2014-07-26 00:24:27 -04:00
data [ "ErrorMsg" ] = trName + l . Tr ( "form.alpha_dash_error" )
2014-12-15 01:49:59 -05:00
case binding . ERR_ALPHA_DASH_DOT :
2014-07-26 00:24:27 -04:00
data [ "ErrorMsg" ] = trName + l . Tr ( "form.alpha_dash_dot_error" )
2017-04-19 06:02:20 +03:00
case validation . ErrGitRefName :
data [ "ErrorMsg" ] = trName + l . Tr ( "form.git_ref_name_error" )
2015-07-24 21:02:49 +08:00
case binding . ERR_SIZE :
data [ "ErrorMsg" ] = trName + l . Tr ( "form.size_error" , GetSize ( field ) )
2014-12-15 01:49:59 -05:00
case binding . ERR_MIN_SIZE :
2014-08-01 06:12:14 -04:00
data [ "ErrorMsg" ] = trName + l . Tr ( "form.min_size_error" , GetMinSize ( field ) )
2014-12-15 01:49:59 -05:00
case binding . ERR_MAX_SIZE :
2014-08-01 06:12:14 -04:00
data [ "ErrorMsg" ] = trName + l . Tr ( "form.max_size_error" , GetMaxSize ( field ) )
2014-12-15 01:49:59 -05:00
case binding . ERR_EMAIL :
2014-07-26 00:24:27 -04:00
data [ "ErrorMsg" ] = trName + l . Tr ( "form.email_error" )
2014-12-15 01:49:59 -05:00
case binding . ERR_URL :
2014-07-26 00:24:27 -04:00
data [ "ErrorMsg" ] = trName + l . Tr ( "form.url_error" )
2015-10-29 21:09:48 -04:00
case binding . ERR_INCLUDE :
data [ "ErrorMsg" ] = trName + l . Tr ( "form.include_error" , GetInclude ( field ) )
2019-09-09 08:48:21 +03:00
case validation . ErrGlobPattern :
data [ "ErrorMsg" ] = trName + l . Tr ( "form.glob_pattern_error" , errs [ 0 ] . Message )
2014-07-26 00:24:27 -04:00
default :
2014-10-15 11:19:20 -04:00
data [ "ErrorMsg" ] = l . Tr ( "form.unknown_error" ) + " " + errs [ 0 ] . Classification
2014-07-26 00:24:27 -04:00
}
2014-10-15 11:19:20 -04:00
return errs
2014-07-26 00:24:27 -04:00
}
}
2014-10-15 11:19:20 -04:00
return errs
2014-03-28 18:40:31 -04:00
}