added cookie v1 and v2 decoding support
This commit is contained in:
parent
df3ac447ff
commit
b8f03d60e4
@ -21,11 +21,11 @@
|
|||||||
*/
|
*/
|
||||||
package org.jboss.netty.handler.codec.http;
|
package org.jboss.netty.handler.codec.http;
|
||||||
|
|
||||||
|
import org.jboss.netty.util.CaseIgnoringComparator;
|
||||||
|
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.TreeMap;
|
import java.util.TreeMap;
|
||||||
|
|
||||||
import org.jboss.netty.util.CaseIgnoringComparator;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @author The Netty Project (netty-dev@lists.jboss.org)
|
* @author The Netty Project (netty-dev@lists.jboss.org)
|
||||||
* @author Andy Taylor (andy.taylor@jboss.org)
|
* @author Andy Taylor (andy.taylor@jboss.org)
|
||||||
@ -33,8 +33,11 @@ import org.jboss.netty.util.CaseIgnoringComparator;
|
|||||||
*/
|
*/
|
||||||
public class CookieDecoder {
|
public class CookieDecoder {
|
||||||
|
|
||||||
private final static String semicolon = ";";
|
private final static String SEMICOLON = ";";
|
||||||
private final static String equals = "=";
|
|
||||||
|
private final static String EQUALS = "=";
|
||||||
|
|
||||||
|
private final static String COMMA = ",";
|
||||||
|
|
||||||
private final String charset;
|
private final String charset;
|
||||||
|
|
||||||
@ -50,19 +53,88 @@ public class CookieDecoder {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public Map<String, Cookie> decode(String header) {
|
public Map<String, Cookie> decode(String header) {
|
||||||
// FIXME: Support both version 0 and 1 cookies
|
|
||||||
// FIXME: Decode all cookie fields, including domain, path, maxAge, secure, and comment.
|
|
||||||
// FIXME: CookieDecoder cannot assume that the first field is always the name-value pair.
|
|
||||||
// FIXME: Check RFC 2109 - http://www.ietf.org/rfc/rfc2109.txt
|
|
||||||
Map<String, Cookie> cookies = new TreeMap<String, Cookie>(CaseIgnoringComparator.INSTANCE);
|
Map<String, Cookie> cookies = new TreeMap<String, Cookie>(CaseIgnoringComparator.INSTANCE);
|
||||||
String[] split = header.split(semicolon);
|
String[] split = header.split(SEMICOLON);
|
||||||
for (String s : split) {
|
for (int i = 0; i < split.length; i++) {
|
||||||
String[] cookie = s.split(equals);
|
DefaultCookie theCookie;
|
||||||
if(cookie != null && cookie.length == 2) {
|
String s = split[i];
|
||||||
|
String[] cookie = s.split(EQUALS, 2);
|
||||||
|
if (cookie != null && cookie.length == 2) {
|
||||||
String name = cookie[0].trim();
|
String name = cookie[0].trim();
|
||||||
String value = QueryStringDecoder.decodeComponent(cookie[1], charset);
|
String value = QueryStringDecoder.decodeComponent(cookie[1], charset);
|
||||||
cookies.put(name, new DefaultCookie(name, value));
|
theCookie = new DefaultCookie(name, value);
|
||||||
|
cookies.put(name, theCookie);
|
||||||
|
boolean discard = false;
|
||||||
|
boolean secure = false;
|
||||||
|
String comment = null;
|
||||||
|
String commentURL = null;
|
||||||
|
String domain = null;
|
||||||
|
String path = null;
|
||||||
|
int version = 0;
|
||||||
|
int maxAge = 0;
|
||||||
|
int[] ports = null;
|
||||||
|
for (int j = i + 1; j < split.length; j++, i++) {
|
||||||
|
String[] val = split[j].split(EQUALS, 2);
|
||||||
|
if (val != null && val.length == 1) {
|
||||||
|
if (CookieHeaderNames.DISCARD.equalsIgnoreCase(val[0])) {
|
||||||
|
discard = true;
|
||||||
}
|
}
|
||||||
|
else if (CookieHeaderNames.SECURE.equalsIgnoreCase(val[0])) {
|
||||||
|
secure = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else if (val != null && val.length == 2) {
|
||||||
|
name = val[0].trim();
|
||||||
|
value = val[1].trim();
|
||||||
|
if (CookieHeaderNames.COMMENT.equalsIgnoreCase(name)) {
|
||||||
|
comment = QueryStringDecoder.decodeComponent(value, charset);
|
||||||
|
}
|
||||||
|
else if (CookieHeaderNames.COMMENTURL.equalsIgnoreCase(name)) {
|
||||||
|
commentURL = QueryStringDecoder.decodeComponent(value, charset);
|
||||||
|
}
|
||||||
|
else if (CookieHeaderNames.DOMAIN.equalsIgnoreCase(name)) {
|
||||||
|
domain = QueryStringDecoder.decodeComponent(value, charset);
|
||||||
|
}
|
||||||
|
else if (CookieHeaderNames.PATH.equalsIgnoreCase(name)) {
|
||||||
|
path = QueryStringDecoder.decodeComponent(value, charset);
|
||||||
|
}
|
||||||
|
else if (CookieHeaderNames.EXPIRES.equalsIgnoreCase(name)) {
|
||||||
|
maxAge = Integer.valueOf(value);
|
||||||
|
}
|
||||||
|
else if (CookieHeaderNames.MAX_AGE.equalsIgnoreCase(name)) {
|
||||||
|
maxAge = Integer.valueOf(value);
|
||||||
|
}
|
||||||
|
else if (CookieHeaderNames.VERSION.equalsIgnoreCase(name)) {
|
||||||
|
version = Integer.valueOf(value);
|
||||||
|
}
|
||||||
|
else if (CookieHeaderNames.PORTLIST.equalsIgnoreCase(name)) {
|
||||||
|
String[] portList = value.split(COMMA);
|
||||||
|
ports = new int[portList.length];
|
||||||
|
for (int i1 = 0; i1 < portList.length; i1++) {
|
||||||
|
String s1 = portList[i1];
|
||||||
|
ports[i1] = Integer.valueOf(s1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
theCookie.setVersion(version);
|
||||||
|
theCookie.setMaxAge(maxAge);
|
||||||
|
theCookie.setPath(path);
|
||||||
|
theCookie.setDomain(domain);
|
||||||
|
theCookie.setSecure(secure);
|
||||||
|
if (version > 0) {
|
||||||
|
theCookie.setComment(comment);
|
||||||
|
}
|
||||||
|
if (version > 1) {
|
||||||
|
theCookie.setCommentURL(commentURL);
|
||||||
|
theCookie.setPortList(ports);
|
||||||
|
theCookie.setDiscard(discard);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
return cookies;
|
return cookies;
|
||||||
}
|
}
|
||||||
|
@ -27,9 +27,9 @@ package org.jboss.netty.handler.codec.http;
|
|||||||
public class CookieHeaderNames {
|
public class CookieHeaderNames {
|
||||||
public static final String PATH = "path";
|
public static final String PATH = "path";
|
||||||
|
|
||||||
private static final String EXPIRES = "expires";
|
public static final String EXPIRES = "expires";
|
||||||
|
|
||||||
private static final String MAX_AGE = "max-age";
|
public static final String MAX_AGE = "max-age";
|
||||||
|
|
||||||
public static final String DOMAIN = "domain";
|
public static final String DOMAIN = "domain";
|
||||||
|
|
||||||
|
@ -0,0 +1,188 @@
|
|||||||
|
/*
|
||||||
|
* JBoss, Home of Professional Open Source
|
||||||
|
* Copyright 2005-2008, Red Hat Middleware LLC, and individual contributors
|
||||||
|
* by the @authors tag. See the copyright.txt in the distribution for a
|
||||||
|
* full listing of individual contributors.
|
||||||
|
*
|
||||||
|
* This is free software; you can redistribute it and/or modify it
|
||||||
|
* under the terms of the GNU Lesser General Public License as
|
||||||
|
* published by the Free Software Foundation; either version 2.1 of
|
||||||
|
* the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This software is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
* Lesser General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU Lesser General Public
|
||||||
|
* License along with this software; if not, write to the Free
|
||||||
|
* Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
|
||||||
|
* 02110-1301 USA, or see the FSF site: http://www.fsf.org.
|
||||||
|
*/
|
||||||
|
package org.jboss.netty.handler.codec.http;
|
||||||
|
|
||||||
|
import static org.junit.Assert.assertEquals;
|
||||||
|
import static org.junit.Assert.assertFalse;
|
||||||
|
import static org.junit.Assert.assertNotNull;
|
||||||
|
import static org.junit.Assert.assertNull;
|
||||||
|
import static org.junit.Assert.assertTrue;
|
||||||
|
import org.junit.Test;
|
||||||
|
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author <a href="mailto:andy.taylor@jboss.org">Andy Taylor</a>
|
||||||
|
*/
|
||||||
|
public class CookieDecoderTest {
|
||||||
|
@Test
|
||||||
|
public void testDecodingSingleCookieV0() {
|
||||||
|
String cookieString = "myCookie=myValue;expires=50;path=%2Fapathsomewhere;domain=%2Fadomainsomewhere;secure;";
|
||||||
|
CookieDecoder cookieDecoder = new CookieDecoder();
|
||||||
|
Map<String, Cookie> cookieMap = cookieDecoder.decode(cookieString);
|
||||||
|
assertEquals(1, cookieMap.size());
|
||||||
|
Cookie cookie = cookieMap.get("MyCookie");
|
||||||
|
assertNotNull(cookie);
|
||||||
|
assertEquals("myValue", cookie.getValue());
|
||||||
|
assertNull(cookie.getComment());
|
||||||
|
assertNull(cookie.getCommentURL());
|
||||||
|
assertEquals("/adomainsomewhere", cookie.getDomain());
|
||||||
|
assertFalse(cookie.isDiscard());
|
||||||
|
assertEquals(50, cookie.getMaxAge());
|
||||||
|
assertEquals("/apathsomewhere", cookie.getPath());
|
||||||
|
assertNull(cookie.getPortList());
|
||||||
|
assertTrue(cookie.isSecure());
|
||||||
|
assertEquals(0, cookie.getVersion());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testDecodingSingleCookieV0ExtraParamsIgnored() {
|
||||||
|
String cookieString = "myCookie=myValue;max-age=50;path=%2Fapathsomewhere;domain=%2Fadomainsomewhere;secure;comment=this%20is%20a%20comment;version=0;commentURL=http%2F%3Aaurl.com;port=80,8080;discard;";
|
||||||
|
CookieDecoder cookieDecoder = new CookieDecoder();
|
||||||
|
Map<String, Cookie> cookieMap = cookieDecoder.decode(cookieString);
|
||||||
|
assertEquals(1, cookieMap.size());
|
||||||
|
Cookie cookie = cookieMap.get("MyCookie");
|
||||||
|
assertNotNull(cookie);
|
||||||
|
assertEquals("myValue", cookie.getValue());
|
||||||
|
assertNull(cookie.getComment());
|
||||||
|
assertNull(cookie.getCommentURL());
|
||||||
|
assertEquals("/adomainsomewhere", cookie.getDomain());
|
||||||
|
assertFalse(cookie.isDiscard());
|
||||||
|
assertEquals(50, cookie.getMaxAge());
|
||||||
|
assertEquals("/apathsomewhere", cookie.getPath());
|
||||||
|
assertNull(cookie.getPortList());
|
||||||
|
assertTrue(cookie.isSecure());
|
||||||
|
assertEquals(0, cookie.getVersion());
|
||||||
|
}
|
||||||
|
@Test
|
||||||
|
public void testDecodingSingleCookieV1() {
|
||||||
|
String cookieString = "myCookie=myValue;max-age=50;path=%2Fapathsomewhere;domain=%2Fadomainsomewhere;secure;comment=this%20is%20a%20comment;version=1;";
|
||||||
|
CookieDecoder cookieDecoder = new CookieDecoder();
|
||||||
|
Map<String, Cookie> cookieMap = cookieDecoder.decode(cookieString);
|
||||||
|
assertEquals(1, cookieMap.size());
|
||||||
|
Cookie cookie = cookieMap.get("MyCookie");
|
||||||
|
assertEquals("myValue", cookie.getValue());
|
||||||
|
assertNotNull(cookie);
|
||||||
|
assertEquals("this is a comment", cookie.getComment());
|
||||||
|
assertNull(cookie.getCommentURL());
|
||||||
|
assertEquals("/adomainsomewhere", cookie.getDomain());
|
||||||
|
assertFalse(cookie.isDiscard());
|
||||||
|
assertEquals(50, cookie.getMaxAge());
|
||||||
|
assertEquals("/apathsomewhere", cookie.getPath());
|
||||||
|
assertNull(cookie.getPortList());
|
||||||
|
assertTrue(cookie.isSecure());
|
||||||
|
assertEquals(1, cookie.getVersion());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testDecodingSingleCookieV1ExtraParamsIgnored() {
|
||||||
|
String cookieString = "myCookie=myValue;max-age=50;path=%2Fapathsomewhere;domain=%2Fadomainsomewhere;secure;comment=this%20is%20a%20comment;version=1;commentURL=http%2F%3Aaurl.com;port=80,8080;discard;";
|
||||||
|
CookieDecoder cookieDecoder = new CookieDecoder();
|
||||||
|
Map<String, Cookie> cookieMap = cookieDecoder.decode(cookieString);
|
||||||
|
assertEquals(1, cookieMap.size());
|
||||||
|
Cookie cookie = cookieMap.get("MyCookie");
|
||||||
|
assertNotNull(cookie);
|
||||||
|
assertEquals("myValue", cookie.getValue());
|
||||||
|
assertEquals("this is a comment", cookie.getComment());
|
||||||
|
assertNull(cookie.getCommentURL());
|
||||||
|
assertEquals("/adomainsomewhere", cookie.getDomain());
|
||||||
|
assertFalse(cookie.isDiscard());
|
||||||
|
assertEquals(50, cookie.getMaxAge());
|
||||||
|
assertEquals("/apathsomewhere", cookie.getPath());
|
||||||
|
assertNull(cookie.getPortList());
|
||||||
|
assertTrue(cookie.isSecure());
|
||||||
|
assertEquals(1, cookie.getVersion());
|
||||||
|
}
|
||||||
|
@Test
|
||||||
|
public void testDecodingSingleCookieV2() {
|
||||||
|
String cookieString = "myCookie=myValue;max-age=50;path=%2Fapathsomewhere;domain=%2Fadomainsomewhere;secure;comment=this%20is%20a%20comment;version=2;commentURL=http%2F%3Aaurl.com;port=80,8080;discard;";
|
||||||
|
CookieDecoder cookieDecoder = new CookieDecoder();
|
||||||
|
Map<String, Cookie> cookieMap = cookieDecoder.decode(cookieString);
|
||||||
|
assertEquals(1, cookieMap.size());
|
||||||
|
Cookie cookie = cookieMap.get("MyCookie");
|
||||||
|
assertNotNull(cookie);
|
||||||
|
assertEquals("myValue", cookie.getValue());
|
||||||
|
assertEquals("this is a comment", cookie.getComment());
|
||||||
|
assertEquals("http/:aurl.com", cookie.getCommentURL());
|
||||||
|
assertEquals("/adomainsomewhere", cookie.getDomain());
|
||||||
|
assertTrue(cookie.isDiscard());
|
||||||
|
assertEquals(50, cookie.getMaxAge());
|
||||||
|
assertEquals("/apathsomewhere", cookie.getPath());
|
||||||
|
assertNotNull(cookie.getPortList());
|
||||||
|
assertEquals(2, cookie.getPortList().length);
|
||||||
|
assertEquals(80, cookie.getPortList()[0]);
|
||||||
|
assertEquals(8080, cookie.getPortList()[1]);
|
||||||
|
assertTrue(cookie.isSecure());
|
||||||
|
assertEquals(2, cookie.getVersion());
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testDecodingMultipleCookies() {
|
||||||
|
String c1 = "myCookie=myValue;max-age=50;path=%2Fapathsomewhere;domain=%2Fadomainsomewhere;secure;comment=this%20is%20a%20comment;version=2;commentURL=http%2F%3Aaurl.com;port=80,8080;discard;";
|
||||||
|
String c2 = "myCookie2=myValue2;max-age=0;path=%2Fanotherpathsomewhere;domain=%2Fanotherdomainsomewhere;comment=this%20is%20another%20comment;version=2;commentURL=http%2F%3Aanotherurl.com;";
|
||||||
|
String c3 = "myCookie3=myValue3;max-age=0;version=2;";
|
||||||
|
CookieDecoder decoder = new CookieDecoder();
|
||||||
|
|
||||||
|
Map<String, Cookie> cookieMap = decoder.decode(c1 + c2 + c3);
|
||||||
|
assertEquals(3, cookieMap.size());
|
||||||
|
Cookie cookie = cookieMap.get("MyCookie");
|
||||||
|
assertNotNull(cookie);
|
||||||
|
assertEquals("myValue", cookie.getValue());
|
||||||
|
assertEquals("this is a comment", cookie.getComment());
|
||||||
|
assertEquals("http/:aurl.com", cookie.getCommentURL());
|
||||||
|
assertEquals("/adomainsomewhere", cookie.getDomain());
|
||||||
|
assertTrue(cookie.isDiscard());
|
||||||
|
assertEquals(50, cookie.getMaxAge());
|
||||||
|
assertEquals("/apathsomewhere", cookie.getPath());
|
||||||
|
assertNotNull(cookie.getPortList());
|
||||||
|
assertEquals(2, cookie.getPortList().length);
|
||||||
|
assertEquals(80, cookie.getPortList()[0]);
|
||||||
|
assertEquals(8080, cookie.getPortList()[1]);
|
||||||
|
assertTrue(cookie.isSecure());
|
||||||
|
assertEquals(2, cookie.getVersion());
|
||||||
|
cookie = cookieMap.get("MyCookie2");
|
||||||
|
assertNotNull(cookie);
|
||||||
|
assertEquals("myValue2", cookie.getValue());
|
||||||
|
assertEquals("this is another comment", cookie.getComment());
|
||||||
|
assertEquals("http/:anotherurl.com", cookie.getCommentURL());
|
||||||
|
assertEquals("/anotherdomainsomewhere", cookie.getDomain());
|
||||||
|
assertFalse(cookie.isDiscard());
|
||||||
|
assertEquals(0, cookie.getMaxAge());
|
||||||
|
assertEquals("/anotherpathsomewhere", cookie.getPath());
|
||||||
|
assertNull(cookie.getPortList());
|
||||||
|
assertFalse(cookie.isSecure());
|
||||||
|
assertEquals(2, cookie.getVersion());
|
||||||
|
cookie = cookieMap.get("MyCookie3");
|
||||||
|
assertNotNull(cookie);
|
||||||
|
assertEquals("myValue3", cookie.getValue());
|
||||||
|
assertNull( cookie.getComment());
|
||||||
|
assertNull(cookie.getCommentURL());
|
||||||
|
assertNull(cookie.getDomain());
|
||||||
|
assertFalse(cookie.isDiscard());
|
||||||
|
assertEquals(0, cookie.getMaxAge());
|
||||||
|
assertNull(cookie.getPath());
|
||||||
|
assertNull(cookie.getPortList());
|
||||||
|
assertFalse(cookie.isSecure());
|
||||||
|
assertEquals(2, cookie.getVersion());
|
||||||
|
}
|
||||||
|
}
|
@ -28,7 +28,7 @@ import org.junit.Test;
|
|||||||
/**
|
/**
|
||||||
* @author <a href="mailto:andy.taylor@jboss.org">Andy Taylor</a>
|
* @author <a href="mailto:andy.taylor@jboss.org">Andy Taylor</a>
|
||||||
*/
|
*/
|
||||||
public class HttpCookieEncoderTest {
|
public class CookieEncoderTest {
|
||||||
@Test
|
@Test
|
||||||
public void testEncodingSingleCookieV0() {
|
public void testEncodingSingleCookieV0() {
|
||||||
String result = "myCookie=myValue;expires=50;path=%2Fapathsomewhere;domain=%2Fadomainsomewhere;secure;";
|
String result = "myCookie=myValue;expires=50;path=%2Fapathsomewhere;domain=%2Fadomainsomewhere;secure;";
|
Loading…
x
Reference in New Issue
Block a user