SpringSecurity+钉钉免登

问题汇总

1、钉钉免登时,要通过dingtalk.js获取一个只能使用一次的code,用次code在服务端获取用户userid;若需要实现自动登录,就不能用传统的用户名密码登录。
2、登录过程中需要从后端获取 accesstoken、jsticket、用户userid,用userid获取userinfo,当accesstoken和jsticket都过期时(7200秒),四个参数皆需要通过网络获取,耗时较长。
3、需要使用spring security保护RESTApi,防止匿名访问。
4、spring security默认表单登录模式,用ajax 增加复杂度。

解决方案

解决思路:

1、前端获取到code时,用corpid作为username,code作为password,重写userDetail,用code能获取到userid时,表明code有效,并用词userid创建新用户,corpid作为密码,添加User权限并返回;
2、accesstoken和jsticket使用独立的程序定时获取,登录时只从数据库取出使用,不从网络获取以减少时间;前端加入loading动画以提升用户体验;开启remeber-me功能,减少登录。
3、配置springSecurity即可,同时将登录事件放置与dd.ready中,防止在钉钉之外的浏览器打开访问。
4、创建form,并使用jQuery填充用户名及密码,使用jQuery模拟click登录。安全性方面,由于corpid公开无妨,code使用次数仅有1此且5分钟有效期,目前暂未使用加密传递;

解决步骤

1、引入springsecurity依赖

compile('org.springframework.boot:spring-boot-starter-security')

2、配置websecurityConfigure

@Configuration
@EnableWebSecurity
public class WebSecurityConfig extends WebSecurityConfigurerAdapter {

    private final String KEY="TECHASER";

    //注入userBean
    @Bean
    UserDetailsService customUserService(){
        return new DingtalkUserService();
    }

    //重写登录验证
    @Override
    protected void configure(AuthenticationManagerBuilder auth)throws Exception{
        auth.userDetailsService(customUserService());
    }

    @Override
    protected void configure(HttpSecurity http) throws Exception {
        http
                .authorizeRequests()
                .antMatchers(
                        HttpMethod.GET,
                        "/",
                        "/*.html",
                        "/favicon.ico",
                        "/**/*.html",
                        "/**/*.css",
                        "/**/*.js",
                        "/**/*.map",
                        "/**/*.png",
                        "/**/*.jpg"
                ).permitAll()
                .anyRequest().authenticated()
                .and()
                .formLogin()
                .loginPage("/login").defaultSuccessUrl("/app").failureUrl("/login/failed").permitAll()
                .and().logout().logoutUrl("/logout").logoutSuccessUrl("/login").permitAll()
                .and().exceptionHandling().accessDeniedPage("/403")
                .and().csrf().disable()
                .rememberMe().rememberMeParameter("rememberMe").tokenValiditySeconds(5000).key(KEY);
        http.headers().cacheControl();
    }
}

登录验证方法:重写 loadUserByUsername即可

@Service
public class DingtalkUserService implements UserDetailsService {

    @Autowired
    private AuthService authService;

    /**
     * 使用前台传来的 免登code,从后端获取userId,从error_code=0判断code是否有效,corpId作为密码,并添加ROLE_USER权限
     * corpId 前面添加 {noop} ,不使用密码加密
     * @param username
     * @return
     * @throws UsernameNotFoundException
     */
    @Override
    public UserDetails loadUserByUsername(String username) throws UsernameNotFoundException {
        String code=username;
        String password="{noop}"+ Company.CORP_ID;
        UserResponse userResponse= null;
        try {
            userResponse = AuthHelper.getUserCode(authService.autoGetAccessToken(),code);
        } catch (IOException e) {
            e.printStackTrace();
        }
        System.out.println("errorCode:"+userResponse.getErrcode());
        if(userResponse.getErrcode().equals("0")){
            List<GrantedAuthority> authorities=new ArrayList<GrantedAuthority>();
            authorities.add(new SimpleGrantedAuthority("ROLE_USER"));
            System.out.println("password is "+password);
            return  new User(userResponse.getUserid(),password,authorities);
        }else {
            return new User("user","{noop}not",new ArrayList<>());
        }
    }
}

登录页面

<!DOCTYPE html>
<html xmlns:th="www.thymeleaf.org">
<head>
    <meta charset="utf-8">
    <title>Login</title>
    <script th:inline="javascript" type="text/javascript">
        /*<![CDATA[*/
        //在此拿到corpid
        var corpid = [[${corpid}]];
        /*]]>*/
    </script>
</head>
<body>

<div style="top:200px;">
    <form th:action="@{/login}" th:method="post">
        <input type="text" name="username" id="username" hidden>
        <input type="password" name="password" id="password" hidden>
        <input type="text" name="remember-me" checked value="TECHASER" hidden>
        <div class='btn-container'>
            <button class='btn btn--shockwave is-active' type="submit" style="color: #0c5460" id="submit-btn">
                点击用钉钉登录
            </button>
        </div>
    </form>
</div>

<script type="text/javascript" th:src="@{http://g.alicdn.com/dingding/open-develop/1.6.9/dingtalk.js}"></script>
<script type="text/javascript" th:src="@{/jquery/jquery-3.3.1.min.js}"></script>

<script type="text/javascript">
    $(document).ready(function () {
        dd.ready(function () {
            dd.runtime.permission.requestAuthCode({
                corpId: corpid,
                onSuccess: function (info) {
                    $("#username").val(info.code);
                    $("#password").val(corpid);
                    $("#submit-btn").click().delay(2000);
                },
                onFail: function (err) {
                    alert('fail: ' + JSON.stringify(err));
                }
            });
        });
    });
</script>
</body>
</html>

需要通过网络获取的参数诸如accessToken jsTicket Userid等,可参考 http://open.dingtalk.com/

后续需要解决的问题:

1、登录速度需要进一步优化
2、remember-me开启后,需要在页面内提供可供退出登录的连接
3、加密传出数据,启用https
4、可以看到 crsf处于disable状态,需开启

绕过的问题

使用 ajax方式登录时,可以从控制台看出已经登录,并返回了需要打开的文档,但未能跳转页面,不明原因。


2018年8月10日补充更新:

解决了登录缓慢的问题

原先登录速度方面,由于使用了spring security,当webApp中请求过多时,造成认证授权的内容过多,容易发生阻塞:

2018-08-10 15:27:47.665 DEBUG 18456 --- [nio-8089-exec-1] o.s.s.w.u.matcher.AntPathRequestMatcher  : Checking match of request : '/app'; against '/'
2018-08-10 15:27:47.666 DEBUG 18456 --- [nio-8089-exec-1] o.s.s.w.u.matcher.AntPathRequestMatcher  : Checking match of request : '/app'; against '/img/*'
2018-08-10 15:27:47.666 DEBUG 18456 --- [nio-8089-exec-1] o.s.s.w.u.matcher.AntPathRequestMatcher  : Checking match of request : '/app'; against '/*.html'
2018-08-10 15:27:47.666 DEBUG 18456 --- [nio-8089-exec-1] o.s.s.w.u.matcher.AntPathRequestMatcher  : Checking match of request : '/app'; against '/favicon.ico'
2018-08-10 15:27:47.666 DEBUG 18456 --- [nio-8089-exec-1] o.s.s.w.u.matcher.AntPathRequestMatcher  : Checking match of request : '/app'; against '/**/*.html'
2018-08-10 15:27:47.666 DEBUG 18456 --- [nio-8089-exec-1] o.s.s.w.u.matcher.AntPathRequestMatcher  : Checking match of request : '/app'; against '/**/*.css'
2018-08-10 15:27:47.666 DEBUG 18456 --- [nio-8089-exec-1] o.s.s.w.u.matcher.AntPathRequestMatcher  : Checking match of request : '/app'; against '/**/*.js'
2018-08-10 15:27:47.666 DEBUG 18456 --- [nio-8089-exec-1] o.s.s.w.u.matcher.AntPathRequestMatcher  : Checking match of request : '/app'; against '/**/*.map'
2018-08-10 15:27:47.666 DEBUG 18456 --- [nio-8089-exec-1] o.s.s.w.u.matcher.AntPathRequestMatcher  : Checking match of request : '/app'; against '/**/*.png'
2018-08-10 15:27:47.666 DEBUG 18456 --- [nio-8089-exec-1] o.s.s.w.u.matcher.AntPathRequestMatcher  : Checking match of request : '/app'; against '/**/*.jpg'
2018-08-10 15:27:47.666 DEBUG 18456 --- [nio-8089-exec-1] o.s.s.w.u.matcher.AntPathRequestMatcher  : Checking match of request : '/app'; against '/**/*.eot'
2018-08-10 15:27:47.666 DEBUG 18456 --- [nio-8089-exec-1] o.s.s.w.u.matcher.AntPathRequestMatcher  : Checking match of request : '/app'; against '/**/*.ttf'
2018-08-10 15:27:47.666 DEBUG 18456 --- [nio-8089-exec-1] o.s.s.w.u.matcher.AntPathRequestMatcher  : Checking match of request : '/app'; against '/**/*.woff'
2018-08-10 15:27:47.666 DEBUG 18456 --- [nio-8089-exec-1] o.s.s.w.u.matcher.AntPathRequestMatcher  : Checking match of request : '/app'; against '/**/*.woff2'
2018-08-10 15:27:47.666 DEBUG 18456 --- [nio-8089-exec-1] o.s.s.w.u.matcher.AntPathRequestMatcher  : Checking match of request : '/app'; against '/**/*.svg'
2018-08-10 15:27:47.666 DEBUG 18456 --- [nio-8089-exec-1] o.s.s.w.u.matcher.AntPathRequestMatcher  : Checking match of request : '/app'; against '/pages/*'

在请求静态资源时,也会做验证,虽然WebSecurityConfigure中配置忽略:

@Override
    public void configure(WebSecurity web) throws Exception {
        web
                .ignoring()
                .antMatchers("/",
                        "/img/*",
                        "/*.html",
                        "/favicon.ico",
                        "/**/*.html",
                        "/**/*.css",
                        "/**/*.js",
                        "/**/*.map",
                        "/**/*.png",
                        "/**/*.jpg",
                        "/**/*.eot",
                        "/**/*.ttf",
                        "/**/*.woff",
                        "/**/*.woff2",
                        "/**/*.svg",
                        "/pages/*",
                        "/test4"
                );
    }

总是在某些请求是卡住五六秒,导致从登陆页面跳转至应用首页时发生白屏现象。

解决办法是,尽量减少请求的次数:
1、合并css、js文件
2、开源框架使用cdn 国内推荐使用:https://www.bootcdn.cn
3、页面中的ajax请求放到页面底部 window.onload函数中,待页面框架加载并渲染完成后开始执行。

此时页面访问速度明细提升,从最初的5~10秒到现在1秒内加载完成。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,189评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,577评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,857评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,703评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,705评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,620评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,995评论 3 396
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,656评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,898评论 1 298
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,639评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,720评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,395评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,982评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,953评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,195评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,907评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,472评论 2 342

推荐阅读更多精彩内容