微信公众号授权与Shiro权限框架整合

2017-12-07  本文已影响0人  捕猎者

微信公众号授权与Shiro权限框架整合

背景

微信公众号授权之后,想访问本地服务器的资源。

实现

  1. 创建新的Token类型。Shiro默认用的Token类型是UsernamePasswordToken,仿照UsernamePasswordToken,建一个属于微信用户的Token类型,叫MockToke

     private String username; //用户名
    
     public String getUsername() {
         return this.username;
     }
    
     public void setUsername(String username) {
         this.username = username;
     }
    
     @Override
     public Object getPrincipal() {
         return this.username;
     }
    
     @Override
     public Object getCredentials() {
         return null;  //密码返回null,不进行认证
     }
    
  2. 添加filter类,来处理不同类型的请求到不同的登陆页面

     //表明是访问微信资源的请求,省略set方法
     private String urlPattern;
     //微信请求对应跳转页面,省略set方法
     private String wxloginUrl;
     //未登录重定向到登陆页
     @Override
     protected void redirectToLogin(ServletRequest req, ServletResponse resp)
             throws IOException {
         HttpServletRequest request = (HttpServletRequest) req;
         HttpServletResponse response = (HttpServletResponse) resp;
         String loginUrl;
         //后台地址跳转到后台登录地址,前台需要登录的跳转到shiro配置的登录地址
         //若请求里包含urlPattern,设置loginUrl为微信登录授权页入口
         if (request.getRequestURI().indexOf(getUrlPattern())!=-1) {
             loginUrl = getWxloginUrl();
         } else {
             loginUrl = getLoginUrl();
         }
         WebUtils.issueRedirect(request, response, loginUrl);
     }
    
  3. 修改Shiro配置文件,添加Filters

    //添加filter的bean,初始化参数值。
    <bean id="userFilter" class="com.smallchill.core.filter.WltxUserFilter">
    <property name="urlPattern" value="/wx"/>
    <property name="wxloginUrl" value="https://open.weixin.qq.com/connect/oauth2/authorize?appid=YOU_APP_ID&amp;redirect_uri=http%3A%2F%2Fwltx.wx.tunnel.echomod.cn%2FWxAuthServlet&amp;response_type=code&amp;scope=snsapi_userinfo&amp;state=STATE&amp;connect_redirect=1#wechat_redirect"/>
</bean>


shiroFilter配置中添加filter

    <bean id="shiroFilter" class="org.apache.shiro.spring.web.ShiroFilterFactoryBean">
    <!-- 安全管理器 -->
    <property name="securityManager" ref="securityManager" />
    <!-- 默认的登陆访问url -->
    <property name="loginUrl" value="/login" />
    <!-- 登陆成功后跳转的url -->
    <property name="successUrl" value="/" />
    <!-- 没有权限跳转的url -->
    <property name="unauthorizedUrl" value="/unauth" />
    <!-- 作用于rest无状态的api过滤器 -->
    <property name="filters">
       <map>
           <!--<entry key="rest" >
                <bean id="restShiroFilter" class="com.smallchill.core.shiro.RestAuthorizationFilter"></bean>  
           </entry>-->
           <entry key="user" value-ref="userFilter" />
       </map>    
    </property>
    <property name="filterChainDefinitions">
        <value>
            <!-- 
                anon  不需要认证
                authc 需要认证
                user  验证通过或RememberMe登录的都可以
                rest  用于rest无状态服务
            -->
            /static/** = anon
            /wx/** = user
            /WxAuthServlet = anon  <!-- 微信授权完,redirect跳转路径,因此时还没拿到微信用户信息,设置为不需要认证 -->
            /MP_verify_LAdRmyUraSqTJfHH.txt = anon
            /login = anon
            /WxServlet = anon
            /captcha = anon
            /** = user 
        </value>
    </property>
</bean>
  1. 修改Realm类用户验证方法,若为微信用户,免密码验证

     /**
      * 登录认证
      */
     @Override
     protected AuthenticationInfo doGetAuthenticationInfo(AuthenticationToken authcToken) throws AuthenticationException {
         log.info("Shiro登录认证启动");
         //若为微信用户token
         if(authcToken instanceof MockToken){
             SimpleAuthenticationInfo info = new SimpleAuthenticationInfo(authcToken.getPrincipal(),null,getName());
             return info;
         }
         
         IShiro shiroFactory = ShiroManager.me().getDefaultShiroFactory();
         UsernamePasswordToken token = (UsernamePasswordToken) authcToken;
         
         User user = shiroFactory.user(token.getUsername());
    
         ShiroUser shiroUser = shiroFactory.shiroUser(user);
    
         SimpleAuthenticationInfo info = shiroFactory.info(shiroUser, user, getName());
    
         log.info("Shiro登录认证完毕");
         return info;
     }
    

    重写Realm的supports方法,使之支持MockToken

     @Override
     public boolean supports(AuthenticationToken token) {
    
         return token instanceof UsernamePasswordToken || token instanceof MockToken;
     }
    
  2. 微信授权通过后,redirect到配置的路径,获取微信用户信息,缓存用户信息,登陆。 以下为WxAuthServlet的部分代码

     WxUserInfo userInfo = wxAdapter.getWxUserInfo(accessToken,openId);
     // 设置要传递的参数
     WxUserCatch.catchUserInfo(userInfo);
    
     MockToken token = new MockToken();
     token.setRememberMe(true);
     token.setUsername(userInfo.getOpenId());
     Subject currentUser = ShiroKit.getSubject();
     currentUser.login(token);
    
    
     // 跳转到index.html
     request.getRequestDispatcher("/wx/index.html").forward(request, response);
    
上一篇 下一篇

猜你喜欢

热点阅读