From 7677aafc1e00a4718a84a983ba576ea1a4baa187 Mon Sep 17 00:00:00 2001 From: Trustin Lee Date: Fri, 13 Mar 2009 15:06:13 +0000 Subject: [PATCH] * CookieDecoder is more tolerant to quoted values * CookieDecoder now understands the first $Version attribute --- .../handler/codec/http/CookieDecoder.java | 41 +++++++++++++++---- .../handler/codec/http/DefaultCookie.java | 2 + 2 files changed, 34 insertions(+), 9 deletions(-) diff --git a/src/main/java/org/jboss/netty/handler/codec/http/CookieDecoder.java b/src/main/java/org/jboss/netty/handler/codec/http/CookieDecoder.java index e7035f11d7..ac696a8233 100644 --- a/src/main/java/org/jboss/netty/handler/codec/http/CookieDecoder.java +++ b/src/main/java/org/jboss/netty/handler/codec/http/CookieDecoder.java @@ -57,13 +57,28 @@ public class CookieDecoder { public Map decode(String header) { Map cookies = new TreeMap(CaseIgnoringComparator.INSTANCE); String[] split = header.split(SEMICOLON); + int version = 0; for (int i = 0; i < split.length; i++) { DefaultCookie theCookie; String s = split[i]; String[] cookie = s.split(EQUALS, 2); if (cookie != null && cookie.length == 2) { - String name = cookie[0].trim(); - String value = QueryStringDecoder.decodeComponent(cookie[1], charset); + String name = trimName(cookie[0]); + String value; + + // $Version is the only attribute that can come before the + // actual cookie name-value pair. + if (name.equalsIgnoreCase(CookieHeaderNames.VERSION)) { + try { + version = Integer.parseInt(trimValue(cookie[1])); + } catch (NumberFormatException e) { + // Ignore. + } + continue; + } + + // If it's not a version attribute, it's the name-value pair. + value = QueryStringDecoder.decodeComponent(trimValue(cookie[1]), charset); theCookie = new DefaultCookie(name, value); cookies.put(name, theCookie); boolean discard = false; @@ -72,8 +87,7 @@ public class CookieDecoder { String commentURL = null; String domain = null; String path = null; - int version = 0; - int maxAge = 0; + int maxAge = -1; int[] ports = null; loop: for (int j = i + 1; j < split.length; j++, i++) { @@ -91,13 +105,13 @@ public class CookieDecoder { } break; case 2: - name = val[0].trim(); - value = val[1].trim(); + name = trimName(val[0]); + value = trimValue(val[1]); if (CookieHeaderNames.COMMENT.equalsIgnoreCase(name)) { comment = value; } else if (CookieHeaderNames.COMMENTURL.equalsIgnoreCase(name)) { - value = trimSurroundingQuotes(value); + value = trimValue(value); commentURL = value; } else if (CookieHeaderNames.DOMAIN.equalsIgnoreCase(name)) { @@ -127,7 +141,7 @@ public class CookieDecoder { version = Integer.valueOf(value); } else if (CookieHeaderNames.PORT.equalsIgnoreCase(name)) { - value = trimSurroundingQuotes(value); + value = trimValue(value); String[] portList = value.split(COMMA); ports = new int[portList.length]; for (int i1 = 0; i1 < portList.length; i1++) { @@ -160,7 +174,16 @@ public class CookieDecoder { return cookies; } - private String trimSurroundingQuotes(String value) { + private String trimName(String name) { + if (name.startsWith("$")) { + return name.substring(1); + } else { + return name; + } + } + + private String trimValue(String value) { + value = value.trim(); if (value.length() >= 2) { char firstChar = value.charAt(0); char lastChar = value.charAt(value.length() - 1); diff --git a/src/main/java/org/jboss/netty/handler/codec/http/DefaultCookie.java b/src/main/java/org/jboss/netty/handler/codec/http/DefaultCookie.java index a994a76127..39735b7aa9 100644 --- a/src/main/java/org/jboss/netty/handler/codec/http/DefaultCookie.java +++ b/src/main/java/org/jboss/netty/handler/codec/http/DefaultCookie.java @@ -73,6 +73,8 @@ public class DefaultCookie implements Cookie { "=,; \\t\\r\\n\\v\\f: " + name); } } + + // FIXME: Refuse known attribute names. this.name = name; setValue(value); }