All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.dromara.jpom.controller.user.UserBasicInfoController Maven / Gradle / Ivy

The newest version!
/*
 * Copyright (c) 2019 Of Him Code Technology Studio
 * Jpom is licensed under Mulan PSL v2.
 * You can use this software according to the terms and conditions of the Mulan PSL v2.
 * You may obtain a copy of Mulan PSL v2 at:
 * 			http://license.coscl.org.cn/MulanPSL2
 * THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND, EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT, MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE.
 * See the Mulan PSL v2 for more details.
 */
package org.dromara.jpom.controller.user;

import cn.hutool.cache.impl.TimedCache;
import cn.hutool.core.collection.CollStreamUtil;
import cn.hutool.core.lang.RegexPool;
import cn.hutool.core.lang.Validator;
import cn.hutool.core.util.RandomUtil;
import cn.hutool.core.util.StrUtil;
import cn.hutool.db.Entity;
import cn.hutool.db.sql.Direction;
import cn.hutool.db.sql.Order;
import cn.keepbx.jpom.IJsonMessage;
import cn.keepbx.jpom.model.JsonMessage;
import com.alibaba.fastjson2.JSONObject;
import lombok.extern.slf4j.Slf4j;
import org.dromara.jpom.common.BaseServerController;
import org.dromara.jpom.common.JpomManifest;
import org.dromara.jpom.common.i18n.I18nMessageUtil;
import org.dromara.jpom.common.interceptor.PermissionInterceptor;
import org.dromara.jpom.common.validator.ValidatorItem;
import org.dromara.jpom.common.validator.ValidatorRule;
import org.dromara.jpom.configuration.UserConfig;
import org.dromara.jpom.func.system.model.ClusterInfoModel;
import org.dromara.jpom.func.system.service.ClusterInfoService;
import org.dromara.jpom.func.user.model.UserLoginLogModel;
import org.dromara.jpom.func.user.server.UserLoginLogServer;
import org.dromara.jpom.model.PageResultDto;
import org.dromara.jpom.model.data.MailAccountModel;
import org.dromara.jpom.model.log.BuildHistoryLog;
import org.dromara.jpom.model.log.UserOperateLogV1;
import org.dromara.jpom.model.user.UserModel;
import org.dromara.jpom.monitor.EmailUtil;
import org.dromara.jpom.service.dblog.DbBuildHistoryLogService;
import org.dromara.jpom.service.dblog.DbUserOperateLogService;
import org.dromara.jpom.service.system.SystemParametersServer;
import org.dromara.jpom.service.user.UserBindWorkspaceService;
import org.dromara.jpom.service.user.UserService;
import org.dromara.jpom.system.ServerConfig;
import org.dromara.jpom.util.TwoFactorAuthUtils;
import org.springframework.http.MediaType;
import org.springframework.util.Assert;
import org.springframework.web.bind.annotation.*;

import javax.servlet.http.HttpServletRequest;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.concurrent.TimeUnit;
import java.util.stream.Collectors;

/**
 * @author bwcx_jzy
 * @since 2019/8/10
 */
@RestController
@RequestMapping(value = "/user")
@Slf4j
public class UserBasicInfoController extends BaseServerController {

    private static final TimedCache CACHE = new TimedCache<>(TimeUnit.MINUTES.toMillis(30));

    private final SystemParametersServer systemParametersServer;
    private final UserBindWorkspaceService userBindWorkspaceService;
    private final UserService userService;
    private final UserConfig userConfig;
    private final UserLoginLogServer userLoginLogServer;
    private final DbUserOperateLogService dbUserOperateLogService;
    private final ClusterInfoService clusterInfoService;
    private final DbBuildHistoryLogService dbBuildHistoryLogService;

    public UserBasicInfoController(SystemParametersServer systemParametersServer,
                                   UserBindWorkspaceService userBindWorkspaceService,
                                   UserService userService,
                                   ServerConfig serverConfig,
                                   UserLoginLogServer userLoginLogServer,
                                   DbUserOperateLogService dbUserOperateLogService,
                                   ClusterInfoService clusterInfoService,
                                   DbBuildHistoryLogService dbBuildHistoryLogService) {
        this.systemParametersServer = systemParametersServer;
        this.userBindWorkspaceService = userBindWorkspaceService;
        this.userService = userService;
        this.userConfig = serverConfig.getUser();
        this.userLoginLogServer = userLoginLogServer;
        this.dbUserOperateLogService = dbUserOperateLogService;
        this.clusterInfoService = clusterInfoService;
        this.dbBuildHistoryLogService = dbBuildHistoryLogService;
    }


    /**
     * get user basic info
     * 获取管理员基本信息接口
     *
     * @return json
     * @author Hotstrip
     */
    @RequestMapping(value = "user-basic-info", method = RequestMethod.POST, produces = MediaType.APPLICATION_JSON_VALUE)
    public IJsonMessage> getUserBasicInfo() {
        UserModel userModel = getUser();
        userModel = userService.getByKey(userModel.getId(), false);
        // return basic info
        Map map = new HashMap<>(10);
        map.put("id", userModel.getId());
        map.put("name", userModel.getName());
        map.put("systemUser", userModel.isSystemUser());
        map.put("superSystemUser", userModel.isSuperSystemUser());
        map.put("demoUser", userModel.isDemoUser());
        map.put("email", userModel.getEmail());
        map.put("dingDing", userModel.getDingDing());
        map.put("workWx", userModel.getWorkWx());
        map.put("md5Token", userModel.getPassword());
        boolean bindMfa = userService.hasBindMfa(userModel.getId());
        map.put("bindMfa", bindMfa);
        map.put("forceMfa", userConfig.isForceMfa());
        return JsonMessage.success("", map);
    }

    @RequestMapping(value = "save_basicInfo.json", method = RequestMethod.POST, produces = MediaType.APPLICATION_JSON_VALUE)
    public IJsonMessage saveBasicInfo(String email,
                                              String dingDing, String workWx, String code,
                                              @ValidatorItem(value = ValidatorRule.NOT_BLANK, range = "2:10", msg = "i18n.nickname_length_limit.6312") String name) {
        UserModel user = getUser();
        UserModel userModel = userService.getByKey(user.getId());
        UserModel updateModel = new UserModel(user.getId());
        // 判断是否一样
        if (StrUtil.isNotEmpty(email) && !StrUtil.equals(email, userModel.getEmail())) {
            Validator.validateEmail(email, I18nMessageUtil.get("i18n.invalid_email_format.7526"));
            Integer cacheCode = CACHE.get(email);
            if (cacheCode == null || !Objects.equals(cacheCode.toString(), code)) {
                return new JsonMessage<>(405, I18nMessageUtil.get("i18n.correct_verification_code2_required.df13"));
            }
            updateModel.setEmail(email);
        }

        updateModel.setName(name);
        //
        if (StrUtil.isNotEmpty(dingDing) && !Validator.isUrl(dingDing)) {
            Validator.validateMatchRegex(RegexPool.URL_HTTP, dingDing, I18nMessageUtil.get("i18n.correct_dingtalk_address_required.2b4a"));
        }
        updateModel.setDingDing(dingDing);
        if (StrUtil.isNotEmpty(workWx)) {
            Validator.validateMatchRegex(RegexPool.URL_HTTP, workWx, I18nMessageUtil.get("i18n.correct_enterprise_wechat_address_required.5f2d"));
        }
        updateModel.setWorkWx(workWx);
        userService.updateById(updateModel);
        return JsonMessage.success(I18nMessageUtil.get("i18n.modify_success.69be"));
    }

    /**
     * 发送邮箱验证
     *
     * @param email 邮箱
     * @return msg
     */
    @RequestMapping(value = "sendCode.json", method = RequestMethod.POST, produces = MediaType.APPLICATION_JSON_VALUE)
    public IJsonMessage sendCode(@ValidatorItem(value = ValidatorRule.EMAIL, msg = "i18n.invalid_email_format.7526") String email) {
        MailAccountModel config = systemParametersServer.getConfig(MailAccountModel.ID, MailAccountModel.class);
        Assert.notNull(config, I18nMessageUtil.get("i18n.admin_email_not_configured.ecb8"));
        int randomInt = RandomUtil.randomInt(1000, 9999);
        try {
            String title = I18nMessageUtil.get("i18n.jpom_verification_code.5b5b");
            EmailUtil.send(email, title, StrUtil.format(I18nMessageUtil.get("i18n.verification_code_is.5af5"), randomInt));
        } catch (Exception e) {
            log.error(I18nMessageUtil.get("i18n.send_failed.9ca6"), e);
            return new JsonMessage<>(500, I18nMessageUtil.get("i18n.send_email_failure.1ab3") + e.getMessage());
        }
        CACHE.put(email, randomInt);
        return JsonMessage.success(I18nMessageUtil.get("i18n.send_success.9db9"));
    }

    /**
     * 查询用户自己的工作空间
     *
     * @return msg
     */
    @GetMapping(value = "my-workspace", produces = MediaType.APPLICATION_JSON_VALUE)
    public IJsonMessage> myWorkspace() {
        UserModel user = getUser();
        List userWorkspaceModels = userService.myWorkspace(user);
        return JsonMessage.success("", userWorkspaceModels);
    }

    /**
     * 保存用户自己的工作空间
     *
     * @return msg
     */
    @PostMapping(value = "save-workspace", produces = MediaType.APPLICATION_JSON_VALUE)
    public IJsonMessage saveWorkspace(@RequestBody List workspaceModels) {
        Assert.notEmpty(workspaceModels, I18nMessageUtil.get("i18n.no_workspace_selected.33d5"));
        List collect = workspaceModels.stream()
            .filter(workspaceModel -> StrUtil.isNotEmpty(workspaceModel.getId()))
            .peek(userWorkspaceModel -> userWorkspaceModel.setOriginalName(null))
            .collect(Collectors.toList());
        UserModel user = getUser();
        Map map = CollStreamUtil.toMap(collect, UserWorkspaceModel::getId, workspaceModel -> workspaceModel);
        String name = "user-my-workspace-" + user.getId();
        systemParametersServer.upsert(name, map, I18nMessageUtil.get("i18n.user_custom_workspace.ef93"));
        return JsonMessage.success(I18nMessageUtil.get("i18n.save_succeeded.3b10"));
    }

    /**
     * 关闭自己到 mfa 相关信息
     *
     * @return json
     */
    @GetMapping(value = "close_mfa", produces = MediaType.APPLICATION_JSON_VALUE)
    public IJsonMessage closeMfa(@ValidatorItem String code) {
        UserModel user = getUser();
        boolean mfaCode = userService.verifyMfaCode(user.getId(), code);
        Assert.state(mfaCode, I18nMessageUtil.get("i18n.verification_code_incorrect.d8c0"));
        UserModel userModel = new UserModel(user.getId());
        userModel.setTwoFactorAuthKey(StrUtil.EMPTY);
        userService.updateById(userModel);
        return JsonMessage.success(I18nMessageUtil.get("i18n.close_success.8a31"));
    }

    @GetMapping(value = "generate_mfa", produces = MediaType.APPLICATION_JSON_VALUE)
    public IJsonMessage generateMfa() {
        UserModel user = getUser();
        JSONObject jsonObject = new JSONObject();
        String tfaKey = TwoFactorAuthUtils.generateTFAKey();
        jsonObject.put("mfaKey", tfaKey);
        jsonObject.put("url", TwoFactorAuthUtils.generateOtpAuthUrl(user.getId(), tfaKey));
        return JsonMessage.success("", jsonObject);
    }

    /**
     * 绑定 mfa
     *
     * @param mfa     mfa key
     * @param twoCode 验证码
     * @return json
     */
    @GetMapping(value = "bind_mfa", produces = MediaType.APPLICATION_JSON_VALUE)
    public IJsonMessage bindMfa(String mfa, String twoCode) {
        //
        UserModel user = getUser();
        boolean bindMfa = userService.hasBindMfa(user.getId());
        Assert.state(!bindMfa, I18nMessageUtil.get("i18n.account_already_bound_to_mfa.5122"));
        // demo
        Assert.state(!user.isDemoUser(), PermissionInterceptor.DEMO_TIP);
        //
        boolean tfaCode = TwoFactorAuthUtils.validateTFACode(mfa, twoCode);
        Assert.state(tfaCode, I18nMessageUtil.get("i18n.mfa_incorrect_code.8783"));
        userService.bindMfa(user.getId(), mfa);
        return JsonMessage.success(I18nMessageUtil.get("i18n.binding_success.1974"));
    }

    /**
     * 登录日志列表
     *
     * @return json
     */
    @RequestMapping(value = "list-login-log-data", method = RequestMethod.POST, produces = MediaType.APPLICATION_JSON_VALUE)
    public IJsonMessage> listLoginLogData(HttpServletRequest request) {
        UserModel user = getUser();
        PageResultDto pageResult = userLoginLogServer.listPageByUserId(request, user.getId());
        return JsonMessage.success("", pageResult);
    }

    /**
     * 操作日志
     *
     * @return json
     */
    @RequestMapping(value = "list-operate-log-data", method = RequestMethod.POST, produces = MediaType.APPLICATION_JSON_VALUE)
    public IJsonMessage> listOperateLogData(HttpServletRequest request) {
        UserModel user = getUser();
        PageResultDto pageResult = dbUserOperateLogService.listPageByUserId(request, user.getId());
        return JsonMessage.success("", pageResult);
    }

    @RequestMapping(value = "recent-log-data", method = RequestMethod.POST, produces = MediaType.APPLICATION_JSON_VALUE)
    public IJsonMessage recentData(HttpServletRequest request) {
        UserModel user = getUser();
        JSONObject jsonObject = new JSONObject();
        {
            Entity entity = Entity.create();
            entity.set("userId", user.getId());
            List operateLog = dbUserOperateLogService.queryList(entity, 10, new Order("createTimeMillis", Direction.DESC));
            jsonObject.put("operateLog", operateLog);
        }
        {
            Entity entity = Entity.create();
            entity.set("modifyUser", user.getId());
            List loginLog = userLoginLogServer.queryList(entity, 10, new Order("createTimeMillis", Direction.DESC));

            jsonObject.put("loginLog", loginLog);
        }
        {
            String workspaceId = dbBuildHistoryLogService.getCheckUserWorkspace(request);
            Entity entity = Entity.create();
            entity.set("workspaceId", workspaceId);
            entity.set("modifyUser", user.getId());
            List loginLog = dbBuildHistoryLogService.queryList(entity, 10, new Order("createTimeMillis", Direction.DESC));
            jsonObject.put("buildLog", loginLog);
        }
        return JsonMessage.success("", jsonObject);
    }

    /**
     * 查询集群列表
     *
     * @return json
     */
    @GetMapping(value = "cluster-list")
    public IJsonMessage clusterList() {
        List list = clusterInfoService.list();
        JSONObject jsonObject = new JSONObject();
        jsonObject.put("list", list);
        jsonObject.put("currentId", JpomManifest.getInstance().getInstallId());
        return JsonMessage.success("", jsonObject);
    }
}