package service

import (
	"crypto/sha1"
	"fmt"
	"github.com/linmadan/egglib-go/core/application"
	"github.com/tiptok/gocomm/common"
	"github.com/tiptok/gocomm/pkg/log"
	"gitlab.fjmaimaimai.com/mmm-go/godevp/pkg/application/factory"
	"gitlab.fjmaimaimai.com/mmm-go/godevp/pkg/application/user/command"
	"gitlab.fjmaimaimai.com/mmm-go/godevp/pkg/application/user/query"
	"gitlab.fjmaimaimai.com/mmm-go/godevp/pkg/domain"
	"gitlab.fjmaimaimai.com/mmm-go/godevp/pkg/infrastructure/utils"
	"gitlab.fjmaimaimai.com/mmm-go/godevp/pkg/protocol"
	"strings"
	"time"
)

type UserService struct {
}

func (svr *UserService) CreateUser(header *protocol.RequestHeader, request *command.CreateUserRequest) (rsp interface{}, err error) {
	var (
		transactionContext, _ = factory.CreateTransactionContext(nil)
	)
	rsp = &command.CreateUserResponse{}
	if err = request.ValidateCommand(); err != nil {
		err = application.ThrowError(application.ARG_ERROR, err.Error())
		return
	}
	if err = transactionContext.StartTransaction(); err != nil {
		err = application.ThrowError(application.TRANSACTION_ERROR, err.Error())
		return
	}
	defer func() {
		transactionContext.RollbackTransaction()
	}()
	newUser := &domain.Users{
		Name:       request.Name,
		Phone:      request.Phone,
		Roles:      request.Roles,
		Status:     1,
		AdminType:  request.AdminType,
		CreateTime: time.Now(),
		UpdateTime: time.Now(),
	}

	var UserRepository, _ = factory.CreateUserRepository(transactionContext)
	if request.Phone != "" {
		if _, err = UserRepository.FindOne(map[string]interface{}{"phone": request.Phone}); err == nil {
			err = protocol.NewCustomMessage(1, "手机号已存在")
			return
		}
	}
	if len(newUser.Passwd) == 0 {
		newUser.Passwd = fmt.Sprintf("%x", sha1.Sum([]byte("mmm123456")))
	}
	if m, e := UserRepository.Save(newUser); e != nil {
		err = application.ThrowError(application.INTERNAL_SERVER_ERROR, e.Error())
		return
	} else {
		rsp = m
	}
	if err = transactionContext.CommitTransaction(); err != nil {
		err = application.ThrowError(application.TRANSACTION_ERROR, err.Error())
		return
	}
	return
}

func (svr *UserService) UpdateUser(header *protocol.RequestHeader, request *command.UpdateUserRequest) (rsp interface{}, err error) {
	var (
		transactionContext, _ = factory.CreateTransactionContext(nil)
	)
	rsp = &command.UpdateUserResponse{}
	if err = request.ValidateCommand(); err != nil {
		err = application.ThrowError(application.ARG_ERROR, err.Error())
		return
	}
	if err = transactionContext.StartTransaction(); err != nil {
		log.Error(err)
		return nil, err
	}
	defer func() {
		transactionContext.RollbackTransaction()
	}()

	var UserRepository, _ = factory.CreateUserRepository(transactionContext)
	var user *domain.Users
	if user, err = UserRepository.FindOne(map[string]interface{}{"id": request.Id}); err != nil {
		err = protocol.NewCustomMessage(1, "用户不存在")
		return
	}
	if request.Phone != "" && request.Phone != user.Phone {
		if _, err = UserRepository.FindOne(map[string]interface{}{"phone": request.Phone}); err == nil {
			err = protocol.NewCustomMessage(1, "手机号已存在")
			return
		}
	}
	//  common.ObjectToMap(request)
	if err = user.Update(utils.LoadCustomFieldToMap(request, header.BodyKeys...)); err != nil {
		err = application.ThrowError(application.INTERNAL_SERVER_ERROR, err.Error())
		return
	}
	if user, err = UserRepository.Save(user); err != nil {
		err = application.ThrowError(application.INTERNAL_SERVER_ERROR, err.Error())
		return
	}
	if err = transactionContext.CommitTransaction(); err != nil {
		err = application.ThrowError(application.TRANSACTION_ERROR, err.Error())
		return
	}
	return
}

func (svr *UserService) GetUser(header *protocol.RequestHeader, request *query.GetUserRequest) (rsp interface{}, err error) {
	var (
		transactionContext, _ = factory.CreateTransactionContext(nil)
		RoleRepository, _     = factory.CreateRoleRepository(transactionContext)
	)
	rsp = &query.GetUserResponse{}
	if err = request.ValidateCommand(); err != nil {
		err = application.ThrowError(application.ARG_ERROR, err.Error())
		return
	}
	if err = transactionContext.StartTransaction(); err != nil {
		err = application.ThrowError(application.TRANSACTION_ERROR, err.Error())
		return
	}
	defer func() {
		transactionContext.RollbackTransaction()
	}()

	var UserRepository, _ = factory.CreateUserRepository(transactionContext)
	var user *domain.Users
	if user, err = UserRepository.FindOne(common.ObjectToMap(request)); err != nil {
		err = application.ThrowError(application.INTERNAL_SERVER_ERROR, err.Error())
		return
	}
	retMap := map[string]interface{}{"id": user.Id, "name": user.Name, "phone": user.Phone, "adminType": user.AdminType, "status": user.Status}
	var roles []*domain.Role
	for _, v := range user.Roles {
		if role, e := RoleRepository.FindOne(map[string]interface{}{"id": v}); e == nil {
			roles = append(roles, role)
		}
	}
	retMap["roles"] = utils.LoadCustomField(roles, "Id", "RoleName")
	rsp = map[string]interface{}{"user": retMap}
	if err = transactionContext.CommitTransaction(); err != nil {
		err = application.ThrowError(application.TRANSACTION_ERROR, err.Error())
		return
	}
	return
}

func (svr *UserService) DeleteUser(header *protocol.RequestHeader, request *command.DeleteUserRequest) (rsp interface{}, err error) {
	var (
		transactionContext, _ = factory.CreateTransactionContext(nil)
	)
	rsp = &command.DeleteUserResponse{}
	if err = request.ValidateCommand(); err != nil {
		err = application.ThrowError(application.ARG_ERROR, err.Error())
		return
	}
	if err = transactionContext.StartTransaction(); err != nil {
		err = application.ThrowError(application.TRANSACTION_ERROR, err.Error())
		return
	}
	defer func() {
		transactionContext.RollbackTransaction()
	}()

	var UserRepository, _ = factory.CreateUserRepository(transactionContext)
	var user *domain.Users
	if user, err = UserRepository.FindOne(common.ObjectToMap(request)); err != nil {
		err = application.ThrowError(application.INTERNAL_SERVER_ERROR, err.Error())
		return
	}
	if user, err = UserRepository.Remove(user); err != nil {
		err = application.ThrowError(application.INTERNAL_SERVER_ERROR, err.Error())
		return
	}
	rsp = user
	if err = transactionContext.CommitTransaction(); err != nil {
		err = application.ThrowError(application.TRANSACTION_ERROR, err.Error())
		return
	}
	return
}

func (svr *UserService) ListUser(header *protocol.RequestHeader, request *query.ListUserRequest) (rsp interface{}, err error) {
	var (
		transactionContext, _ = factory.CreateTransactionContext(nil)
		RoleRepository, _     = factory.CreateRoleRepository(transactionContext)
	)
	rsp = &query.ListUserResponse{}
	if err = request.ValidateCommand(); err != nil {
		err = application.ThrowError(application.TRANSACTION_ERROR, err.Error())
		return
	}
	if err = transactionContext.StartTransaction(); err != nil {
		log.Error(err)
		return nil, err
	}
	defer func() {
		transactionContext.RollbackTransaction()
	}()

	getRoles := func(roleIds []int64) string {
		if len(roleIds) == 0 {
			return ""
		}
		var roles []string
		var mapRoles = make(map[int64]*domain.Role)
		for _, id := range roleIds {
			if _, ok := mapRoles[id]; ok {
				continue
			}
			if roleItem, _ := RoleRepository.FindOne(map[string]interface{}{"id": id}); roleItem != nil {
				roles = append(roles, roleItem.RoleName)
			} else {
				mapRoles[id] = nil
			}
			continue
		}
		return strings.Join(roles, ",")
	}

	// TODO:可优化,每次只查询 user.id 列表 ,通过缓存查询user对象
	var UserRepository, _ = factory.CreateUserRepository(transactionContext)
	var user []*domain.Users
	var total int64
	if total, user, err = UserRepository.Find(common.ObjectToMap(request)); err != nil {
		err = application.ThrowError(application.INTERNAL_SERVER_ERROR, err.Error())
		return
	}
	userList := make([]map[string]interface{}, 0)
	for _, v := range user {
		item := map[string]interface{}{"id": v.Id, "name": v.Name, "phone": v.Phone, "adminType": v.AdminType, "status": v.Status, "createTime": v.CreateTime.Local().Format("2006-01-02 15:04:05")}
		item["roles"] = getRoles(v.Roles)
		userList = append(userList, item)
	}

	rsp = map[string]interface{}{
		"totalRow":   total,
		"pageNumber": request.PageNumber,
		"lists":      userList,
	}
	if err = transactionContext.CommitTransaction(); err != nil {
		err = application.ThrowError(application.TRANSACTION_ERROR, err.Error())
		return
	}
	return
}

func NewUserService(options map[string]interface{}) *UserService {
	svr := &UserService{}
	return svr
}