2016-02-01 12:59:10 +01:00
|
|
|
/*
|
|
|
|
Copyright (c) 2016 VMware, Inc. All Rights Reserved.
|
|
|
|
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.
|
|
|
|
*/
|
2016-02-26 11:54:14 +01:00
|
|
|
|
2016-02-01 12:59:10 +01:00
|
|
|
package api
|
|
|
|
|
|
|
|
import (
|
2016-05-25 08:21:01 +02:00
|
|
|
"fmt"
|
2016-02-24 07:31:52 +01:00
|
|
|
"net/http"
|
2016-04-08 09:25:52 +02:00
|
|
|
"os"
|
2016-05-25 08:21:01 +02:00
|
|
|
"regexp"
|
2016-02-01 12:59:10 +01:00
|
|
|
"strconv"
|
2016-04-08 09:25:52 +02:00
|
|
|
"strings"
|
2016-02-01 12:59:10 +01:00
|
|
|
|
|
|
|
"github.com/vmware/harbor/dao"
|
|
|
|
"github.com/vmware/harbor/models"
|
2016-03-28 02:50:09 +02:00
|
|
|
"github.com/vmware/harbor/utils/log"
|
2016-02-01 12:59:10 +01:00
|
|
|
)
|
|
|
|
|
2016-02-26 11:35:55 +01:00
|
|
|
// UserAPI handles request to /api/users/{}
|
2016-02-01 12:59:10 +01:00
|
|
|
type UserAPI struct {
|
|
|
|
BaseAPI
|
2016-04-15 10:55:33 +02:00
|
|
|
currentUserID int
|
|
|
|
userID int
|
|
|
|
SelfRegistration bool
|
|
|
|
IsAdmin bool
|
|
|
|
AuthMode string
|
2016-02-01 12:59:10 +01:00
|
|
|
}
|
|
|
|
|
2016-04-21 07:27:47 +02:00
|
|
|
type passwordReq struct {
|
|
|
|
OldPassword string `json:"old_password"`
|
|
|
|
NewPassword string `json:"new_password"`
|
|
|
|
}
|
|
|
|
|
2016-02-26 11:35:55 +01:00
|
|
|
// Prepare validates the URL and parms
|
2016-02-01 12:59:10 +01:00
|
|
|
func (ua *UserAPI) Prepare() {
|
|
|
|
|
2016-04-08 09:25:52 +02:00
|
|
|
authMode := strings.ToLower(os.Getenv("AUTH_MODE"))
|
|
|
|
if authMode == "" {
|
|
|
|
authMode = "db_auth"
|
|
|
|
}
|
|
|
|
ua.AuthMode = authMode
|
|
|
|
|
|
|
|
selfRegistration := strings.ToLower(os.Getenv("SELF_REGISTRATION"))
|
|
|
|
if selfRegistration == "on" {
|
|
|
|
ua.SelfRegistration = true
|
|
|
|
}
|
|
|
|
|
2016-04-01 13:42:13 +02:00
|
|
|
if ua.Ctx.Input.IsPost() {
|
2016-04-08 09:25:52 +02:00
|
|
|
sessionUserID := ua.GetSession("userId")
|
2016-04-15 10:55:33 +02:00
|
|
|
_, _, ok := ua.Ctx.Request.BasicAuth()
|
|
|
|
if sessionUserID == nil && !ok {
|
2016-04-08 09:25:52 +02:00
|
|
|
return
|
|
|
|
}
|
2016-04-01 13:42:13 +02:00
|
|
|
}
|
|
|
|
|
2016-02-25 06:40:08 +01:00
|
|
|
ua.currentUserID = ua.ValidateUser()
|
2016-02-01 12:59:10 +01:00
|
|
|
id := ua.Ctx.Input.Param(":id")
|
|
|
|
if id == "current" {
|
2016-02-25 06:40:08 +01:00
|
|
|
ua.userID = ua.currentUserID
|
2016-02-01 12:59:10 +01:00
|
|
|
} else if len(id) > 0 {
|
|
|
|
var err error
|
2016-02-25 06:40:08 +01:00
|
|
|
ua.userID, err = strconv.Atoi(id)
|
2016-02-01 12:59:10 +01:00
|
|
|
if err != nil {
|
2016-03-28 02:50:09 +02:00
|
|
|
log.Errorf("Invalid user id, error: %v", err)
|
2016-02-24 07:31:52 +01:00
|
|
|
ua.CustomAbort(http.StatusBadRequest, "Invalid user Id")
|
2016-02-01 12:59:10 +01:00
|
|
|
}
|
2016-02-26 03:15:01 +01:00
|
|
|
userQuery := models.User{UserID: ua.userID}
|
2016-02-01 12:59:10 +01:00
|
|
|
u, err := dao.GetUser(userQuery)
|
|
|
|
if err != nil {
|
2016-03-28 02:50:09 +02:00
|
|
|
log.Errorf("Error occurred in GetUser, error: %v", err)
|
2016-02-24 07:31:52 +01:00
|
|
|
ua.CustomAbort(http.StatusInternalServerError, "Internal error.")
|
2016-02-01 12:59:10 +01:00
|
|
|
}
|
|
|
|
if u == nil {
|
2016-03-28 02:50:09 +02:00
|
|
|
log.Errorf("User with Id: %d does not exist", ua.userID)
|
2016-02-24 07:31:52 +01:00
|
|
|
ua.CustomAbort(http.StatusNotFound, "")
|
2016-02-01 12:59:10 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-04-08 09:25:52 +02:00
|
|
|
var err error
|
|
|
|
ua.IsAdmin, err = dao.IsAdminRole(ua.currentUserID)
|
2016-02-01 12:59:10 +01:00
|
|
|
if err != nil {
|
2016-04-08 09:25:52 +02:00
|
|
|
log.Errorf("Error occurred in IsAdminRole:%v", err)
|
2016-02-24 07:31:52 +01:00
|
|
|
ua.CustomAbort(http.StatusInternalServerError, "Internal error.")
|
2016-02-01 12:59:10 +01:00
|
|
|
}
|
2016-04-15 10:55:33 +02:00
|
|
|
|
2016-04-08 09:25:52 +02:00
|
|
|
}
|
2016-02-01 12:59:10 +01:00
|
|
|
|
2016-04-08 09:25:52 +02:00
|
|
|
// Get ...
|
|
|
|
func (ua *UserAPI) Get() {
|
2016-02-25 06:40:08 +01:00
|
|
|
if ua.userID == 0 { //list users
|
2016-04-08 09:25:52 +02:00
|
|
|
if !ua.IsAdmin {
|
2016-03-28 02:50:09 +02:00
|
|
|
log.Errorf("Current user, id: %d does not have admin role, can not list users", ua.currentUserID)
|
2016-02-24 07:31:52 +01:00
|
|
|
ua.RenderError(http.StatusForbidden, "User does not have admin role")
|
2016-02-01 12:59:10 +01:00
|
|
|
return
|
|
|
|
}
|
|
|
|
username := ua.GetString("username")
|
|
|
|
userQuery := models.User{}
|
|
|
|
if len(username) > 0 {
|
|
|
|
userQuery.Username = "%" + username + "%"
|
|
|
|
}
|
|
|
|
userList, err := dao.ListUsers(userQuery)
|
|
|
|
if err != nil {
|
2016-03-28 02:50:09 +02:00
|
|
|
log.Errorf("Failed to get data from database, error: %v", err)
|
2016-02-24 07:31:52 +01:00
|
|
|
ua.RenderError(http.StatusInternalServerError, "Failed to query from database")
|
2016-02-01 12:59:10 +01:00
|
|
|
return
|
|
|
|
}
|
|
|
|
ua.Data["json"] = userList
|
|
|
|
|
2016-04-08 09:25:52 +02:00
|
|
|
} else if ua.userID == ua.currentUserID || ua.IsAdmin {
|
2016-02-26 03:15:01 +01:00
|
|
|
userQuery := models.User{UserID: ua.userID}
|
2016-02-01 12:59:10 +01:00
|
|
|
u, err := dao.GetUser(userQuery)
|
|
|
|
if err != nil {
|
2016-03-28 02:50:09 +02:00
|
|
|
log.Errorf("Error occurred in GetUser, error: %v", err)
|
2016-02-24 07:31:52 +01:00
|
|
|
ua.CustomAbort(http.StatusInternalServerError, "Internal error.")
|
2016-02-01 12:59:10 +01:00
|
|
|
}
|
|
|
|
ua.Data["json"] = u
|
|
|
|
} else {
|
2016-03-28 02:50:09 +02:00
|
|
|
log.Errorf("Current user, id: %d does not have admin role, can not view other user's detail", ua.currentUserID)
|
2016-02-24 07:31:52 +01:00
|
|
|
ua.RenderError(http.StatusForbidden, "User does not have admin role")
|
2016-02-01 12:59:10 +01:00
|
|
|
return
|
|
|
|
}
|
|
|
|
ua.ServeJSON()
|
|
|
|
}
|
|
|
|
|
2016-02-26 11:35:55 +01:00
|
|
|
// Put ...
|
2016-05-25 08:21:01 +02:00
|
|
|
func (ua *UserAPI) Put() {
|
|
|
|
ldapAdminUser := (ua.AuthMode == "ldap_auth" && ua.userID == 1 && ua.userID == ua.currentUserID)
|
|
|
|
|
|
|
|
if !(ua.AuthMode == "db_auth" || ldapAdminUser) {
|
|
|
|
ua.CustomAbort(http.StatusForbidden, "")
|
|
|
|
}
|
2016-04-08 09:25:52 +02:00
|
|
|
if !ua.IsAdmin {
|
2016-05-25 08:21:01 +02:00
|
|
|
if ua.userID != ua.currentUserID {
|
2016-06-01 06:27:26 +02:00
|
|
|
log.Warning("Guests can only change their own account.")
|
2016-05-25 08:21:01 +02:00
|
|
|
ua.CustomAbort(http.StatusForbidden, "Guests can only change their own account.")
|
|
|
|
}
|
|
|
|
}
|
2016-06-06 17:07:17 +02:00
|
|
|
user := models.User{UserID: ua.userID}
|
|
|
|
ua.DecodeJSONReq(&user)
|
2016-06-06 17:47:27 +02:00
|
|
|
err := commonValidate(user)
|
2016-06-06 17:07:17 +02:00
|
|
|
if err != nil {
|
|
|
|
log.Warning("Bad request in change user profile: %v", err)
|
|
|
|
ua.RenderError(http.StatusBadRequest, "change user profile error:"+err.Error())
|
|
|
|
return
|
|
|
|
}
|
2016-06-06 11:53:36 +02:00
|
|
|
userQuery := models.User{UserID: ua.userID}
|
|
|
|
u, err := dao.GetUser(userQuery)
|
|
|
|
if err != nil {
|
|
|
|
log.Errorf("Error occurred in GetUser, error: %v", err)
|
|
|
|
ua.CustomAbort(http.StatusInternalServerError, "Internal error.")
|
|
|
|
}
|
|
|
|
if u == nil {
|
|
|
|
log.Errorf("User with Id: %d does not exist", ua.userID)
|
|
|
|
ua.CustomAbort(http.StatusNotFound, "")
|
|
|
|
}
|
|
|
|
if u.Email != user.Email {
|
|
|
|
emailExist, err := dao.UserExists(user, "email")
|
|
|
|
if err != nil {
|
|
|
|
log.Errorf("Error occurred in change user profile: %v", err)
|
|
|
|
ua.CustomAbort(http.StatusInternalServerError, "Internal error.")
|
|
|
|
}
|
|
|
|
if emailExist {
|
|
|
|
log.Warning("email has already been used!")
|
|
|
|
ua.RenderError(http.StatusConflict, "email has already been used!")
|
|
|
|
return
|
|
|
|
}
|
2016-05-25 08:21:01 +02:00
|
|
|
}
|
|
|
|
if err := dao.ChangeUserProfile(user); err != nil {
|
|
|
|
log.Errorf("Failed to update user profile, error: %v", err)
|
|
|
|
ua.CustomAbort(http.StatusInternalServerError, err.Error())
|
|
|
|
}
|
2016-02-01 12:59:10 +01:00
|
|
|
}
|
|
|
|
|
2016-04-01 13:42:13 +02:00
|
|
|
// Post ...
|
|
|
|
func (ua *UserAPI) Post() {
|
2016-04-08 09:25:52 +02:00
|
|
|
|
|
|
|
if !(ua.AuthMode == "db_auth") {
|
|
|
|
ua.CustomAbort(http.StatusForbidden, "")
|
|
|
|
}
|
|
|
|
|
|
|
|
if !(ua.SelfRegistration || ua.IsAdmin) {
|
|
|
|
log.Warning("Registration can only be used by admin role user when self-registration is off.")
|
|
|
|
ua.CustomAbort(http.StatusForbidden, "")
|
|
|
|
}
|
|
|
|
|
2016-04-06 05:26:24 +02:00
|
|
|
user := models.User{}
|
|
|
|
ua.DecodeJSONReq(&user)
|
2016-05-25 08:21:01 +02:00
|
|
|
err := validate(user)
|
|
|
|
if err != nil {
|
2016-06-01 06:27:26 +02:00
|
|
|
log.Warning("Bad request in Register: %v", err)
|
2016-05-25 08:21:01 +02:00
|
|
|
ua.RenderError(http.StatusBadRequest, "register error:"+err.Error())
|
|
|
|
return
|
|
|
|
}
|
|
|
|
userExist, err := dao.UserExists(user, "username")
|
|
|
|
if err != nil {
|
|
|
|
log.Errorf("Error occurred in Register: %v", err)
|
2016-06-01 06:27:26 +02:00
|
|
|
ua.CustomAbort(http.StatusInternalServerError, "Internal error.")
|
2016-05-25 08:21:01 +02:00
|
|
|
}
|
|
|
|
if userExist {
|
|
|
|
log.Warning("username has already been used!")
|
2016-06-01 06:27:26 +02:00
|
|
|
ua.RenderError(http.StatusConflict, "username has already been used!")
|
|
|
|
return
|
2016-05-25 08:21:01 +02:00
|
|
|
}
|
2016-06-01 11:51:06 +02:00
|
|
|
emailExist, err := dao.UserExists(user, "email")
|
|
|
|
if err != nil {
|
|
|
|
log.Errorf("Error occurred in change user profile: %v", err)
|
|
|
|
ua.CustomAbort(http.StatusInternalServerError, "Internal error.")
|
|
|
|
}
|
|
|
|
if emailExist {
|
|
|
|
log.Warning("email has already been used!")
|
|
|
|
ua.RenderError(http.StatusConflict, "email has already been used!")
|
|
|
|
return
|
|
|
|
}
|
2016-04-22 11:57:39 +02:00
|
|
|
userID, err := dao.Register(user)
|
2016-04-01 13:42:13 +02:00
|
|
|
if err != nil {
|
2016-04-05 11:45:47 +02:00
|
|
|
log.Errorf("Error occurred in Register: %v", err)
|
2016-06-01 06:27:26 +02:00
|
|
|
ua.CustomAbort(http.StatusInternalServerError, "Internal error.")
|
2016-04-01 13:42:13 +02:00
|
|
|
}
|
2016-04-15 06:33:48 +02:00
|
|
|
|
2016-04-25 06:10:15 +02:00
|
|
|
ua.Redirect(http.StatusCreated, strconv.FormatInt(userID, 10))
|
2016-04-01 13:42:13 +02:00
|
|
|
}
|
|
|
|
|
2016-02-26 11:35:55 +01:00
|
|
|
// Delete ...
|
2016-02-01 12:59:10 +01:00
|
|
|
func (ua *UserAPI) Delete() {
|
2016-04-08 09:25:52 +02:00
|
|
|
if !ua.IsAdmin {
|
2016-03-28 02:50:09 +02:00
|
|
|
log.Warningf("current user, id: %d does not have admin role, can not remove user", ua.currentUserID)
|
2016-02-24 07:31:52 +01:00
|
|
|
ua.RenderError(http.StatusForbidden, "User does not have admin role")
|
2016-02-01 12:59:10 +01:00
|
|
|
return
|
|
|
|
}
|
2016-04-08 09:25:52 +02:00
|
|
|
var err error
|
2016-02-25 06:40:08 +01:00
|
|
|
err = dao.DeleteUser(ua.userID)
|
2016-02-01 12:59:10 +01:00
|
|
|
if err != nil {
|
2016-03-28 02:50:09 +02:00
|
|
|
log.Errorf("Failed to delete data from database, error: %v", err)
|
2016-02-24 07:31:52 +01:00
|
|
|
ua.RenderError(http.StatusInternalServerError, "Failed to delete User")
|
2016-02-01 12:59:10 +01:00
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
2016-04-21 07:27:47 +02:00
|
|
|
|
|
|
|
// ChangePassword handles PUT to /api/users/{}/password
|
|
|
|
func (ua *UserAPI) ChangePassword() {
|
2016-05-09 14:57:41 +02:00
|
|
|
ldapAdminUser := (ua.AuthMode == "ldap_auth" && ua.userID == 1 && ua.userID == ua.currentUserID)
|
2016-05-20 10:56:12 +02:00
|
|
|
|
2016-05-09 14:57:41 +02:00
|
|
|
if !(ua.AuthMode == "db_auth" || ldapAdminUser) {
|
2016-04-21 07:27:47 +02:00
|
|
|
ua.CustomAbort(http.StatusForbidden, "")
|
|
|
|
}
|
|
|
|
|
|
|
|
if !ua.IsAdmin {
|
|
|
|
if ua.userID != ua.currentUserID {
|
|
|
|
log.Error("Guests can only change their own account.")
|
|
|
|
ua.CustomAbort(http.StatusForbidden, "Guests can only change their own account.")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
var req passwordReq
|
|
|
|
ua.DecodeJSONReq(&req)
|
|
|
|
if req.OldPassword == "" {
|
|
|
|
log.Error("Old password is blank")
|
|
|
|
ua.CustomAbort(http.StatusBadRequest, "Old password is blank")
|
|
|
|
}
|
|
|
|
|
|
|
|
queryUser := models.User{UserID: ua.userID, Password: req.OldPassword}
|
|
|
|
user, err := dao.CheckUserPassword(queryUser)
|
|
|
|
if err != nil {
|
|
|
|
log.Errorf("Error occurred in CheckUserPassword: %v", err)
|
|
|
|
ua.CustomAbort(http.StatusInternalServerError, "Internal error.")
|
|
|
|
}
|
|
|
|
if user == nil {
|
|
|
|
log.Warning("Password input is not correct")
|
|
|
|
ua.CustomAbort(http.StatusForbidden, "old_password_is_not_correct")
|
|
|
|
}
|
|
|
|
|
|
|
|
if req.NewPassword == "" {
|
|
|
|
ua.CustomAbort(http.StatusBadRequest, "please_input_new_password")
|
|
|
|
}
|
|
|
|
updateUser := models.User{UserID: ua.userID, Password: req.NewPassword, Salt: user.Salt}
|
|
|
|
err = dao.ChangeUserPassword(updateUser, req.OldPassword)
|
|
|
|
if err != nil {
|
|
|
|
log.Errorf("Error occurred in ChangeUserPassword: %v", err)
|
|
|
|
ua.CustomAbort(http.StatusInternalServerError, "Internal error.")
|
|
|
|
}
|
|
|
|
}
|
2016-05-20 10:56:12 +02:00
|
|
|
|
2016-06-17 09:22:58 +02:00
|
|
|
// ToggleUserAdminRole handles PUT api/users/{}/sysadmin
|
2016-05-25 08:21:01 +02:00
|
|
|
func (ua *UserAPI) ToggleUserAdminRole() {
|
|
|
|
if !ua.IsAdmin {
|
|
|
|
log.Warningf("current user, id: %d does not have admin role, can not update other user's role", ua.currentUserID)
|
|
|
|
ua.RenderError(http.StatusForbidden, "User does not have admin role")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
userQuery := models.User{UserID: ua.userID}
|
2016-06-02 11:33:10 +02:00
|
|
|
ua.DecodeJSONReq(&userQuery)
|
2016-06-06 11:53:36 +02:00
|
|
|
if err := dao.ToggleUserAdminRole(userQuery.UserID, userQuery.HasAdminRole); err != nil {
|
2016-05-25 08:21:01 +02:00
|
|
|
log.Errorf("Error occurred in ToggleUserAdminRole: %v", err)
|
|
|
|
ua.CustomAbort(http.StatusInternalServerError, "Internal error.")
|
|
|
|
}
|
|
|
|
}
|
2016-06-06 11:53:36 +02:00
|
|
|
|
2016-06-02 11:33:10 +02:00
|
|
|
// validate only validate when user register
|
2016-05-25 08:21:01 +02:00
|
|
|
func validate(user models.User) error {
|
|
|
|
|
|
|
|
if isIllegalLength(user.Username, 0, 20) {
|
|
|
|
return fmt.Errorf("Username with illegal length.")
|
2016-05-20 10:56:12 +02:00
|
|
|
}
|
2016-05-25 08:21:01 +02:00
|
|
|
if isContainIllegalChar(user.Username, []string{",", "~", "#", "$", "%"}) {
|
|
|
|
return fmt.Errorf("Username contains illegal characters.")
|
|
|
|
}
|
|
|
|
if isIllegalLength(user.Password, 0, 20) {
|
|
|
|
return fmt.Errorf("Password with illegal length.")
|
|
|
|
}
|
|
|
|
if err := commonValidate(user); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2016-06-02 11:33:10 +02:00
|
|
|
//commonValidate validates email, realname, comment information when user register or change their profile
|
2016-05-25 08:21:01 +02:00
|
|
|
func commonValidate(user models.User) error {
|
|
|
|
|
|
|
|
if len(user.Email) > 0 {
|
|
|
|
if m, _ := regexp.MatchString(`^(([^<>()[\]\\.,;:\s@\"]+(\.[^<>()[\]\\.,;:\s@\"]+)*)|(\".+\"))@((\[[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\])|(([a-zA-Z\-0-9]+\.)+[a-zA-Z]{2,}))$`, user.Email); !m {
|
|
|
|
return fmt.Errorf("Email with illegal format.")
|
2016-05-20 10:56:12 +02:00
|
|
|
}
|
2016-06-01 11:51:06 +02:00
|
|
|
} else {
|
|
|
|
return fmt.Errorf("Email can't be empty")
|
2016-05-20 10:56:12 +02:00
|
|
|
}
|
2016-05-25 08:21:01 +02:00
|
|
|
|
|
|
|
if isIllegalLength(user.Realname, 0, 20) {
|
|
|
|
return fmt.Errorf("Realname with illegal length.")
|
|
|
|
}
|
|
|
|
|
|
|
|
if isContainIllegalChar(user.Realname, []string{",", "~", "#", "$", "%"}) {
|
|
|
|
return fmt.Errorf("Realname contains illegal characters.")
|
|
|
|
}
|
|
|
|
if isIllegalLength(user.Comment, -1, 30) {
|
|
|
|
return fmt.Errorf("Comment with illegal length.")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
func isIllegalLength(s string, min int, max int) bool {
|
|
|
|
if min == -1 {
|
|
|
|
return (len(s) > max)
|
|
|
|
}
|
|
|
|
if max == -1 {
|
|
|
|
return (len(s) <= min)
|
|
|
|
}
|
|
|
|
return (len(s) < min || len(s) > max)
|
|
|
|
}
|
|
|
|
|
|
|
|
func isContainIllegalChar(s string, illegalChar []string) bool {
|
|
|
|
for _, c := range illegalChar {
|
|
|
|
if strings.Index(s, c) >= 0 {
|
|
|
|
return true
|
|
|
|
}
|
2016-05-20 10:56:12 +02:00
|
|
|
}
|
2016-05-25 08:21:01 +02:00
|
|
|
return false
|
2016-05-20 10:56:12 +02:00
|
|
|
}
|