Python requests 301/302/303重定向(跨域/本域)cookie、Location問題

  今天使用request的get方法獲取一個網站的登陸頁信息,結果使用charles抓包的時候發現該網站登陸頁303(重定向的問題),網上查了不少資料,緣由以下:json

1、cookie跨域

  緣由:利用requests模擬登陸時,直接使用request.get(url),容易形成 301/302/303 重定向,由於cookie不持久,形成重定向以後的cookie等信息獲取不到cookie

  1>.同一個Host下邊重定向session

    解決方法:使用requests.Session()方法,會使該鏈接持久化,而且保存請求的狀態(session、cookie等),這樣在重定向以後,咱們使用requests.cookies就能獲取到cookie了網站

    代碼:  url

result = requests.Session()
header = {
    'Host' : 'xxxxx',
    'Accept' : 'xxxx',
    'User-Agen': 'xxxx' # 這個最重要,建議手動設置(有時候沒有設置,形成獲取數據失敗,本人被坑過......)
}
response = result.get(url, headers=header)

  注意上面講的重定向是在同一個host下重定向,跨域的重定向用上邊的方法就獲取不到cookie spa

  2>.跨域重定向code

    示例:A(http://a.kuyu.com/login)->B(http://b.baidu.com/xxxxx)->C(http://b.baidu.com/xxxx/xxxx),相似這種形式,那麼使用第一種解決辦法就沒法獲取到cookie,這時候就須要變換一種方式來進行獲取blog

  

# 獲取請求url域名
def getHost(url):
    pattern = re.compile(r'(.*?)://(.*?)/', re.S)
    response = re.search(pattern, url)
    if response:
        return {'header':str(response.group(1)).strip(), 'host': str(response.group(2)).strip()}
    else:
        return None
cookieType = {} # 保存域名對應的cookie
def getRedirectCookie(url):
    locationList = set()
    cookie = ''
    resUrl = url  # 獲取最後請求url
    header = {
        'User-Agen': 'Mozilla/5.0 (Windows NT 6.1; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/61.0.3163.100 Safari/537.36'
    }
    try:
        while True:
            # 請求
            response = requests.get(url, headers=header, allow_redirects=False)
            # 獲取host
            hostObj = getHost(resUrl)
            if hostObj is None:
                return None
            # 處理cookie
            cookie = response.cookies.get_dict()
            if cookie == {}:
                pass
            else:
                cookieType[str(hostObj['host']).strip()] = json.dumps(cookie)  # 保存cookie
            # 獲取跳轉的url
            if 'Location' in response.headers.keys():
                url = response.headers.get('Location')
                if not 'http' in url:
                    url = hostObj['header'] + '://' + hostObj['host'] + url  # 拼接host域名

                resUrl = url
                if url in locationList: break
                locationList.add(url)
            else:
                break

        return {'url': str(resUrl), 'content': response.content, 'header': response.headers}
    except urllib2.URLError, e:
        if hasattr(e, 'reason'):
            print '請求失敗,緣由:' + e.reason
        return None

  注意:有時候執行帶有重定向的時候會出現:"requests.exceptions.TooManyRedirects: Exceeded 30 redirects."的狀況,那麼這時候是說重定向次數上限了,也可使用上邊的這種方式來解決ip

2、Location(重定向url)

  有時候遇到重定向,咱們只須要獲取重定向的url便可,這個時候使用上述方法開啓一個保持狀態的長鏈接,並請求時獲取不到的請求的headers的,那麼咱們該怎麼辦呢?

  解決:  

result = requests.Session()
header = {
    'Host' : 'xxxxx',
    'Accept' : 'xxxx',
    'User-Agen': 'xxxx' # 這個最重要,建議手動設置(有時候沒有設置,形成獲取數據失敗,本人被坑過......)
}
response = result.get(url, headers=header, allow_redirects=False)
location = response.headers['Location'] # 注意有些header返回的Location中的url是不帶host部分的,須要注意一下

  allow_redirects=False的意義爲拒絕默認的301/302/303重定向從而能夠經過response.headers['Location']拿到重定向的URL。

相關文章
相關標籤/搜索