HostOnly Cookie和HttpOnly Cookie


怎么使用Cookie?

  通常我們有兩種方式給瀏覽器設置或獲取Cookie,分別是HTTP Response Headers中的Set-Cookie Header和HTTP Request Headers中的Cookie Header,以及通過JavaScript對document.cookie進行賦值或取值。

rfc6265第5.2節定義的Set-Cookie Header,除了必須包含Cookie正文,還可以選擇性包含6個屬性path、domain、max-age、expires、secure、httponly,它們之間用英文分號和空格("; ")連接。

Cookie的正文部分,是由&連接的key=value鍵值對字符串,類似於url中的查詢字符串。下面是一個標准的Set-Cookie Header:

-收縮 HTML代碼   運行代碼   [如果運行無效果,請自行將源代碼保存為html文件運行]
Set-Cookie: key=value; path=path; domain=domain; max-age=max-age-in-seconds; expires=date-in-GMTString-format; secure; httponly

  在瀏覽器端,通過document.cookie也可以設置Cookie,以MDC文檔為例,Cookie的內容除了必須包含正文之外,還可選5個屬性:path、domain、max-age、expires、secure。下面是簡單的示例:

-收縮 HTML代碼   運行代碼   [如果運行無效果,請自行將源代碼保存為html文件運行]
document.cookie = "key=value; path=path; domain=domain; max-age=max-age-in-seconds; expires=date-in-GMTString-format; secure";

有兩點需要說明:

  • max-age作為對expires的補充,現階段有兼容性問題(IE低版本不支持),所以一般不單獨使用;
  • JS中設置Cookie和HTTP方式相比較,少了對HttpOnly的控制,是因為JS不能讀寫HttpOnly Cookie;

我在各瀏覽器測試后發現, 如果不考慮HttpOnly,通過JS或HTTP方式設置的Cookie沒有區別;通過JS或HTTP方式獲取到的Cookie內容也是一樣的。所以本文后續測試中,不需要注明是通過何種方式操作的Cookie。

什么是HostOnly Cookie?

  rfc6265第5.3節定 義了瀏覽器存放每個Cookie時應該包括這些字段:name、value、expiry-time、domain、path、creation- time、last-access-time、persistent-flag,、host-only-flag、secure-only-flag和 http-only-flag。

其中:

  • name、value:由Cookie正文指定;
  • expiry-time:根據Cookie中的expires和max-age產生;
  • domain、path:分別由Cookie中的domain和path指定;
  • creation-time、last-access-time:由瀏覽器自行獲得;
  • persistent-flag:持久化標記,在expiry-time未知的情況下為false,表示這是個session cookie;
  • secure-only-flag:在Cookie中包含secure屬性時為true,表示這個cookie僅在https環境下才能使用;
  • http-only-flag:在Cookie中包含httponly屬性時為true,表示這個cookie不允許通過JS來讀寫;
  • host-only-flag:在Cookie中不包含Domain屬性,或者Domain屬性為空,或者Domain屬性不合法(不等於頁面url中的Domain部分、也不是頁面Domain的大域)時為true。此時,我們把這個Cookie稱之為HostOnly Cookie;

那么host-only-flag如果為true會怎樣呢?rfc6265里有這么一段:

Either: The cookie's host-only-flag is true and the canonicalized request-host is identical to the cookie's domain.

Or:  The cookie's host-only-flag is false and the canonicalized request-host domain-matches the cookie's domain.

  獲取Cookie時,首先要檢查Domain匹配性,其次才檢查path、secure、httponly等屬性的匹配性。如果host- only-flag為true時,只有當前域名與該Cookie的Domain屬性完全相等才可以進入后續流程;host-only-flag為 false時,符合域規則(domain-matches)的域名都可以進入后續流程。

  舉個例子,host-only-flag為true時,Domain屬性為example.com的Cookie只有在example.com才有 可能獲取到;host-only-flag為false時,Domain屬性為example.com的Cookie,在example.com、 www.example.com、sub.example.com等等都可能獲取到。

  下面,我們來研究下各瀏覽器對HostOnly Cookie,也就是Cookie的host-only-flag屬性的支持情況。

支持度測試

qgy18.com,設置如下HostOnly Cookie:

-收縮 HTML代碼   運行代碼   [如果運行無效果,請自行將源代碼保存為html文件運行]
name=ququ; expires=Tue, 10-Jul-2013 08:30:18 GMT; path=/

訪問www.qgy18.com,獲取Cookie,結果如下:

瀏覽器 在www.qgy18.com獲取到的Cookie
Chrome 29.0.1547.3 dev  
Firefox 22.0  
Chrome 27.0.1453.116 m  
IE 6.0.2900.5512 name=ququ
IE 10.0.9200.16438 name=ququ
Opera 12.15(Presto內核,非Webkit)  
iOS Safari 6.1.3  
Safari 7.0  

小結:

  IE系列(表中只列了IE6和10,實際上IE6-IE10都測過)不支持HostOnly Cookie。在qgy18.com設置的Cookie,www.qgy18.com可以直接獲取到。

  其它瀏覽器支持HostOnly Cookie。本測試中,對於非IE:獲取Cookie的頁面Domain是www.qgy18.com,由於設置Cookie時沒有指定Domain, 按照前面的規則,host-only-flag為true,Cookie的Domain屬性是qgy18.com,二者不完全匹配,所以獲取不到這個 Cookie。

  對於非HostOnly Cookie,例如在qgy18.com設置Cookie時指定Domain為qgy18.com,在www.qgy18.com可以獲取到這個Cookie,這時候host-only-flag為false。

Cookie覆蓋測試

www.qgy18.com,設置以下3條Cookie:

-收縮 HTML代碼   運行代碼   [如果運行無效果,請自行將源代碼保存為html文件運行]
、name=ququ1; expires=Tue, 10-Jul-2013 08:30:18 GMT; path=/
、name=ququ2; expires=Tue, 10-Jul-2013 08:30:18 GMT; path=/; domain=.www.qgy18.com
、name=ququ3; expires=Tue, 10-Jul-2013 08:30:18 GMT; path=/; domain=www.qgy18.com

訪問www.qgy18.com,獲取Cookie,結果如下:

瀏覽器 在www.qgy18.com獲取到的Cookie
Chrome 29.0.1547.3 dev name=ququ1; name=ququ3
Firefox 22.0 name=ququ1; name=ququ3
Chrome 27.0.1453.116 m name=ququ1; name=ququ3
IE 6.0.2900.5512 name=ququ3
IE 10.0.9200.16438 name=ququ3
Opera 12.15(Presto內核,非Webkit) name=ququ3
iOS Safari 6.1.3 name=ququ3; name=ququ1
Safari 7.0 name=ququ3; name=ququ1

  規范里有兩點規定需要先說明下:

  1. 設置Cookie時,Domain屬性值如果是.a.com,前面的.會被去掉,變成a.com(rfc6265第5.2.3節);
  2. 對於name、path和domain均相同的Cookie,后面的覆蓋前面的(rfc6265第5.3節第10段);

小結:

  由於IE系列不支持HostOnly Cookie,三個語句對於IE來說是完全一樣的(1沒有指定Domain,自動使用請求頭中的Host或者頁面url中的Domain部分作為 Cookie的Domain屬性,都是www.qgy18.com),后面覆蓋前面,只剩下name=ququ3;

  分歧出在Presto內核的Opera與Chrome、Safari和Firefox之間:Opera認為三個語句的name、path和 domain均相同,產生了跟IE一樣的結果;其它瀏覽器認為host-only-flag為true的Domain和其它兩個不同,所以只有語句3可以 覆蓋2,剩下1和3;

  從各自控制台展示的Cookie信息印證了這一點:

  Chrome、Safari和Firefox都把HostOnly Cookie的Domain顯示為真正的Domain,非HostOnly Cookie的Domain前面多加了一個英文句號.。這樣,前面的結果似乎也解釋得過去:對於Chrome、Safari和Firefox,由於.的存 在,第1條語句的Domain和其它兩個確實不一樣,不會被覆蓋。

chrome29 (Chrome 29.0.1547.3 dev)

firefox22 (Firefox 22.0 )

safari7 (Safari 7.0)

opera12 (Opera 12.15 )

ie10 (IE 10.0.9200.16438 )

  大家應該注意到了:本節測試中,Safari獲取到的Cookie順序與其它瀏覽器不一樣。這是個大隱患,無論是用JS還是HTTP獲取這個Demo里的Cookie,都會在Safari下得到不一樣的結果。Cookie的優先級問題我打算找時間詳細測試下,再單獨討論。

結論

在我測試過的瀏覽器中:

  • 除開IE,Chrome/Firefox/Safari/Opera都支持HostOnly Cookie,可以限制Cookie只有在Domain完全匹配時才可用;
  • Opera的HostOnly Cookie會被domain、path和name相同的非HostOnly Cookie覆蓋;
  • Chrome/Firefox/Safari中,非HostOnly Cookie的Domain屬性前面多了一個.,與沒有.的HostOnly Cookie不存在覆蓋的可能;
  • 同名Cookie優先級存在瀏覽器差異,實際項目中應該避免出現同名Cookie;

  之前有同學問過Chrome開發工具看到的Cookie為什么有些前面有.,有些沒有?經過前面的分析,大家應該都知道原因了吧。

  另外,想不通為什么host-only-flag,沒有像http-only-flag或secure-only-flag一樣,有對應的屬性可以直接設置呢?

來源:http://ju.outofmemory.cn/entry/36099


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM