php 跨域 session,thinkphp中session跨域问题解决

7a2a16518b2e38195ede774f3071b6fd.png

本地使用,一切正常;后端项目和前端项目都部署到服务器,一切正常;后端项目部署到服务器,并设置允许跨域访问后,本地前端项目使用服务器上后端项目接口时,问题来了:

首先,使用postman测试获取图片验证码接口和验证图片验证码接口,正常。

然后,在html中使用获取图片验证码接口,正常;最后,在JS中使用验证图片验证码接口,出错!!!

分析

通过问题描述,我们看出,问题出现在跨域上。那么,有两种可能,一种是因为跨域设置不正确;一种是因为thinkphp本身的问题。

采用另外一种跨域配置,问题依然存在。那就是thinkphp本身的问题了,经查找资料,问题定位在thinkphp的session跨域上。

跨子域解决办法

其实不管是ThinkPHP还是php本身,在解决session跨域问题的时候都需要设置session.cookie_domain。

针对session跨域这一问题的解决方法主要有以下几种:

第一种情况:如果目录下没有.htaccess这个文件,也就是没有采取url伪静态的话,那么,在conf/config.php的第一行加上:ini_set('session.cookie_domain',".domain.com");//跨域访问Session

这时如果你开启了调试,那么可以用!但关闭了调试,就不管用了!

第二种情况:如果你目录下有.htaccess这个文件,那么你在根目录,index.php的第一行加入:<?php ini_set('session.cookie_domain',".domain.com");//跨域访问Session

// 应用入口文件

?>

这种方法不管开不开启调试都管用!

然而,我们的问题并不是跨子域的问题,而是完全跨域,所以上述方法无效。

完全跨域解决办法

获取图片验证码请求

查看获取图片验证码的请求信息,Request Headers为:Accept:image/webp,image/*,*/*;q=0.8

Accept-Encoding:gzip, deflate, sdch

Accept-Language:zh-CN,zh;q=0.8,en-US;q=0.6,en;q=0.4

Connection:keep-alive

Cookie:pma_lang=zh_CN; pma_collation_connection=utf8_unicode_ci; pma_iv-1=wnpO4gv0eQRW1AMHmGr2ww%3D%3D; pmaUser-1=weZPqS0%2BW7nzFUVHRdqcfA%3D%3D

Host:api.voidking.com

Referer:http://localhost/ajax/ajax.html

User-Agent:Mozilla/5.0 (Windows NT 6.3; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/54.0.2840.71 Safari/537.36

Response Headers为:Access-Control-Allow-Origin:*

Cache-Control:post-check=0, pre-check=0

Cache-Control:private, max-age=0, no-store, no-cache, must-revalidate

Connection:keep-alive

Content-Type:image/png

Date:Sun, 27 Nov 2016 12:10:44 GMT

Expires:Thu, 19 Nov 1981 08:52:00 GMT

Pragma:no-cache

Server:nginx

Set-Cookie:PHPSESSID=721t4sqanvsii550m1dk8gq1o3; path=/; domain=.voidking.com

Transfer-Encoding:chunked

验证验证码请求

查看验证验证码的请求信息,Request Headers为:Accept:application/json, text/javascript, */*; q=0.01

Accept-Encoding:gzip, deflate

Accept-Language:zh-CN,zh;q=0.8,en-US;q=0.6,en;q=0.4

Connection:keep-alive

Content-Length:9

Content-Type:application/x-www-form-urlencoded; charset=UTF-8

Host:api.voidking.com

Origin:http://localhost

Referer:http://localhost/ajax/ajax.html

User-Agent:Mozilla/5.0 (Windows NT 6.3; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/54.0.2840.71 Safari/537.36

Response Headers为:Access-Control-Allow-Origin:*

Cache-Control:no-store, no-cache, must-revalidate, post-check=0, pre-check=0

Connection:keep-alive

Content-Encoding:gzip

Content-Type:text/html; charset=UTF-8

Date:Sun, 27 Nov 2016 12:13:21 GMT

Expires:Thu, 19 Nov 1981 08:52:00 GMT

Pragma:no-cache

Server:nginx

Set-Cookie:PHPSESSID=149t0hhs2icqaaemvp39onkgp4; path=/; domain=.voidking.com

Transfer-Encoding:chunked

Vary:Accept-Encoding

再次获取图片验证码请求

Request Headers为:Accept:image/webp,image/*,*/*;q=0.8

Accept-Encoding:gzip, deflate, sdch

Accept-Language:zh-CN,zh;q=0.8,en-US;q=0.6,en;q=0.4

Cache-Control:max-age=0

Connection:keep-alive

Cookie:pma_lang=zh_CN; pma_collation_connection=utf8_unicode_ci; pma_iv-1=wnpO4gv0eQRW1AMHmGr2ww%3D%3D; pmaUser-1=weZPqS0%2BW7nzFUVHRdqcfA%3D%3D; PHPSESSID=721t4sqanvsii550m1dk8gq1o3

Host:api.voidking.com

Referer:http://localhost/ajax/ajax.html

User-Agent:Mozilla/5.0 (Windows NT 6.3; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/54.0.2840.71 Safari/537.36

Response Headers为:Access-Control-Allow-Origin:*

Cache-Control:private, max-age=0, no-store, no-cache, must-revalidate

Cache-Control:post-check=0, pre-check=0

Connection:keep-alive

Content-Type:image/png

Date:Sun, 27 Nov 2016 13:26:21 GMT

Expires:Thu, 19 Nov 1981 08:52:00 GMT

Pragma:no-cache

Server:nginx

Transfer-Encoding:chunked

三次请求比较

67d64014c4eff2c1bc5afecee3a0e1b2.png

第一次获取图片验证码请求,Cookie中没有PHPSESSID,所以,返回信息中有Set-Cookie。第二次获取图片验证码请求,Cookie中含有PHPSESSID,所以,返回信息中没有了Set-Cookie。

而且第一次请求返回信息Set-Cookie中的PHPSESSID,和第二次请求请求信息Cookie中的PHPSESSID是相同的。

而验证图片验证码的ajax请求,没有Cookie,自然也没有PHPSESSID,所以,返回信息中也有Set-Cookie。

可见,我们需要在前端做一些修改,使之发送请求时带着Cookie。

前端jquery设置

jquery

$(function(){

$('#send').click(function(){

//console.log(document.cookie);

$.ajax({

url: 'http://api.voidking.com/owner-bd/index.php/Home/CheckCode/checkPicCode',

type: 'POST',

crossDomain: true,

xhrFields: {

withCredentials: true

},

dataType: 'json',

data: {code: $('#picCode').val()},

success: function(data){

console.log(data);

},

error: function(xhr){

console.log(xhr);

}

});

});

});

请求时报错如下:A wildcard '*' cannot be used in the 'Access-Control-Allow-Origin' header when the credentials flag is true. Origin 'http://localhost' is therefore not allowed access. The credentials mode of an XMLHttpRequest is controlled by the withCredentials attribute.

出现了跨域报错,可见后端也需要做一些修改,使之可以接收跨域Cookie。

后端nginx设置add_header Access-Control-Allow-Origin http://localhost;

add_header Access-Control-Allow-Credentials true;

注意:

服务器端Access-Control-Allow-Credentials参数为true时,Access-Control-Allow-Origin参数的值不能为*。

后端nginx设置后,jquery的ajax请求正常了,可以携带Cookie,后端正常接收数据并返回数据。

由于angular的ajax请求不同于jquery,所以,我们还需要研究一下angular怎么发送携带Cookie的跨域请求。

前端angular设置

angular

var app = angular.module('myApp', []);

app.controller('myCtrl', function($scope, $http, $httpParamSerializer) {

$scope.send = function(){

$http({

method:'POST',

url:'http://api.voidking.com/owner-bd/index.php/Home/CheckCode/checkPicCode',

headers:{

'Content-Type':'application/x-www-form-urlencoded'

},

withCredentials: true,

dataType: 'json',

data: $httpParamSerializer({code: $scope.picCode})

}).then(function successCallback(response) {

console.log(response.data);

$scope.username = response.data.username;

}, function errorCallback(response) {

console.log(response.data);

});

}

});

推荐教程:《TP5》

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值