【解决方案】跨域共享cookie


层级关系的域名的cookie共享

首先,我们需要知道cookie的作用的域名是domain本身及domain下的所有子域名。

test.lalogin.test.laarticle.test.la为例,login.test.laarticle.test.la作为test.la的子域名,在跨域共享信息的时候,我们只需要将需要共享的信息放置于test.la域名下即可。

譬如共享登录信息,在通过login.test.la域名登录后,将登录信息置于test.la下,访问article.test.la,前端通过从test.la域下拿cookie信息,从而达到共享的目的。

/**
     *
     * @param name  cookie的key
     * @param value cookie的value
     */
    public static Cookie buildCookie(String name, String value, String domain) {
        Cookie cookie = new Cookie(name, value);
        // 设置时间
        cookie.setMaxAge();
        cookie.setDomain(domain);
        // 设置路径
        cookie.setPath("/");
        return cookie;
    }

这里需要注意一下cookie时间和路径的设置,cookie存储在本地内存里,maxAge默认值为-1,此时只要关闭浏览器,cookie就会消失。设置为0,表示从内存中删除此cookie。

浏览器会将maxAge为正数的Cookie持久化,即写到对应的Cookie文件中。无论客户关闭了浏览器还是电脑,只要还在maxAge秒之前,登录网站时该Cookie仍然有效。(其设置的最大保存时长也是有限制的)

第二个注意的地方是路径,cookie默认设置的路径是当前访问的路径,如果不指明设置在根路径下,有时会出现明明设置成功了,但却一直读取不到的情况。


无层级关系的域名的cookie共享

两个完全不同的域名,如test.latest.com,如何共享cookie?这里有本人实践过的两种解决方案。

  • 譬如在test.la登陆后,后端设置cookie,前端将cookie保存至Storage中,然后访问test.com时,以jsonp的方式(test.la域名)进行请求接口A,后端的接口A将会拿到test.la域下的cookie,并以jsonp形式返回,实现test.com域名下也能共享该cookie。
 @RequestMapping("/getLoginInfo")
    public void getLoginInfo(HttpServletRequest request, HttpServletResponse response) {
	   // 这里是封装的一个cookie获取类,可以自己获取对应的cookie
       String accessToken = CookieUtil.getValue("access_token", request);
       response.setHeader("Access-Control-Allow-Origin", "*");
       response.setHeader("Access-Control-Allow-Methods", "POST,GET,OPTIONS,DELETE,HEAD,PUT,PATCH");
       response.setHeader("Access-Control-Max-Age", "36000");
       response.setHeader("Access-Control-Allow-Headers", "Origin, X-Requested-With, Content-Type, Accept,Authorization,authorization");
       response.setHeader("Access-Control-Allow-Credentials","true");

       JsonObject jsonObject = new JsonObject();
       jsonObject.addProperty("access_token", accessToken);
   
       String callBack = request.getParameter("callback");
       // jsonp形式返回
        try {
            if (StringUtils.isBlank(callBack)) {
                response.getWriter().append(new Gson().toJson(jsonObject));
            } else {
                response.getWriter().append(callBack + "(" + new Gson().toJson(jsonObject) + ")");
            }
            return;
        } catch (IOException e) {
           LOGGER.error("跨域失败 :" + e);
        }
    }
  • 另一种方法,在test.la登陆后,设置cookie到test.la下,然后由前端发起一个js请求接口A(https://test.la/A),后端接口A做redirect重定向到后端接口B(https://test.com/B),在B接口设置cookie到test.com
 @RequestMapping("/redirect")
 // 参数t为域名, 如例子中的test.com
    public String redirect(String t, HttpServletRequest request) {
        if (StringUtils.isBlank(t)) {
            throw new BusinessException("域名参数不能为空");
        }
        // 截取域名后3段, 判断是否合法
        String target = getRealDomain(t);
        if (!defaultDomain(target)) {
            throw new BusinessException("域名不合法" + t);
        }
        // 结合例子,这里我们就从`test.la`获取cookie, 然后作为参数
        String value = CookieUtil.getValue("access_token", request);
        LOGGER.info("浏览器跨域请求 : {}, {}", t, value);
        return "redirect:https://youli." + t + "/api/user/web/set?t=" + value;
    }

    private String getRealDomain(String domain) {
        String[] array = domain.split("\\.");
        if (array.length <= 3) {
            return domain;
        }
        StringBuilder builder = new StringBuilder();
        for (int i = 1; i < array.length; i++) {
            builder.append(".").append(array[i]);
        }
        return builder.substring(1);
    }

    private boolean defaultDomain(String target) {
        String[] array = domains.split(",");
        for (String domain : array) {
            if (domain.equals(target)) {
                return true;
            }
        }
        return false;
    }

    @RequestMapping("/set")
    @ResponseBody
    public ResultData<?> set(String t, HttpServletRequest request, HttpServletResponse response) throws URISyntaxException {
        if (StringUtils.isBlank(t)) {
            throw new BusinessException("cookie参数不能为空");
        }
        // 截取域名后3段, 判断是否合法
        URI uri = new URI(request.getRequestURL().toString());
        String domain = StringUtils.substring(uri.getHost(),StringUtils.indexOf(uri.getHost(),".") + 1);
        LOGGER.info("设置的域名:" + domain);
        String target = getRealDomain(domain);
        LOGGER.info("设置的target:" + domain);
        if (!defaultDomain(target)) {
            throw new BusinessException("域名不合法" + t);
        }
        response.setHeader("Access-Control-Allow-Credentials","true");
        response.setHeader("p3p","CP=\"CURa ADMa DEVa PSAo PSDo OUR BUS UNI PUR INT DEM STA PRE COM NAV OTC NOI DSP COR\"");
        response.setContentType("text/javascript;charset=UTF-8");


        ResponseCookie cookie = ResponseCookie.from("access_token", target) // key & value
                .httpOnly(true)         // 禁止js读取
                .secure(true)           // 在http下也传输
                .domain(domain)         // 域名
                .path("/")              // path
                .maxAge(DateTimeUtil.getSecondTimestamp() + 3600 * 24 * 360 * 10)  
                .sameSite("None")
                .build()
                ;
        response.addHeader("Set-Cookie", cookie.toString());
        return new ResultData<>();
    }
  • 1
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
在Web开发中,Cookie是常用的一种状态管理方式,它可以在客户端浏览器和服务器之间传递信息,并且具有跨域的能力。但是在跨域场景下,不同域名之间的Cookie是不能共享的。本文将介绍如何通过Java实现跨域Cookie共享的方法。 一、什么是跨域Cookie共享 在同源策略下,浏览器只允许当前域名下的页面访问该域名下的Cookie,而不允许其他域名的页面访问该域名下的Cookie。这就是所谓的Cookie跨域问题。 但是在实际开发中,我们经常需要在不同域名之间共享Cookie。例如,在OA系统中,用户登录后可能会跳转到其他子系统中,此时需要保持用户登录状态,就需要在不同域名之间共享Cookie。这就是所谓的跨域Cookie共享。 二、实现跨域Cookie共享的方法 在Java中,实现跨域Cookie共享的方法有两种: 1、使用第三方Cookie共享方案,例如:CAS、OAuth等。 2、手动实现Cookie共享,即在跨域场景下手动设置Cookie。 下面将介绍手动实现Cookie共享的方法。 三、手动实现Cookie共享 手动实现Cookie共享的原理很简单:在A域名下设置Cookie时,同时在B域名下也设置一个相同名称的Cookie。这样,在B域名下访问时就可以获取到A域名下的Cookie信息了。 具体实现步骤如下: 1、在A域名下设置Cookie时,同时在B域名下也设置一个相同名称的Cookie。 代码示例: ```java // 在A域名下设置Cookie Cookie cookie = new Cookie("userId", "123456"); cookie.setDomain(".domaina.com"); // 设置Cookie域名为.domaina.com cookie.setPath("/"); // 设置Cookie的路径为根路径 cookie.setMaxAge(60 * 60 * 24 * 7); // 设置Cookie的生命周期为7天 response.addCookie(cookie); // 将Cookie添加到响应中 // 在B域名下设置相同名称的Cookie Cookie cookie = new Cookie("userId", "123456"); cookie.setDomain(".domainb.com"); // 设置Cookie域名为.domainb.com cookie.setPath("/"); // 设置Cookie的路径为根路径 cookie.setMaxAge(60 * 60 * 24 * 7); // 设置Cookie的生命周期为7天 response.addCookie(cookie); // 将Cookie添加到响应中 ``` 2、在B域名下访问时,获取A域名下的Cookie信息。 代码示例: ```java // 获取A域名下的Cookie信息 Cookie[] cookies = request.getCookies(); if (cookies != null) { for (Cookie cookie : cookies) { if (cookie.getName().equals("userId")) { String userId = cookie.getValue(); // TODO: 处理业务逻辑 break; } } } ``` 需要注意的是,在设置Cookie时,域名要以点号开头,例如:.domaina.com。这样设置后,所有以domaina.com结尾的子域名都可以共享Cookie。 四、总结 本文介绍了如何通过Java实现跨域Cookie共享的方法,即手动实现Cookie共享。在跨域场景下,手动实现Cookie共享可以很好地解决Cookie跨域问题。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

legendaryhaha

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值