Skip to content
This repository was archived by the owner on Feb 8, 2023. It is now read-only.

Fix #17: JSON generates cookie that it cannot read #23

Open
wants to merge 5 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
39 changes: 21 additions & 18 deletions src/main/java/com/github/jscookie/javacookie/Cookies.java
Original file line number Diff line number Diff line change
Expand Up @@ -296,27 +296,31 @@ private String encode( String decoded, Set<Integer> exceptions ) {
String character = new String( Character.toChars( codePoint ) );
CharArrayWriter hexSequence = new CharArrayWriter();
byte[] bytes = character.getBytes( UTF_8 );
for ( int bytesIndex = 0; bytesIndex < bytes.length; bytesIndex++ ) {
char left = Character.forDigit( bytes[ bytesIndex ] >> 4 & 0xF, 16 );
char right = Character.forDigit( bytes[ bytesIndex ] & 0xF, 16 );
for (byte aByte : bytes) {
char left = Character.forDigit(aByte >> 4 & 0xF, 16);
char right = Character.forDigit(aByte & 0xF, 16);
hexSequence
.append( '%' )
.append( left )
.append( right );
.append('%')
.append(left)
.append(right);
}
String target = character.toString();
String sequence = hexSequence.toString().toUpperCase();
encoded = encoded.replace( target, sequence );
encoded = encoded.replace(character, sequence );
} catch ( UnsupportedEncodingException e ) {
e.printStackTrace();
}
}
return encoded;
}

private String decode( String encoded ) {
private String decode(String encoded) {
// Decode characters with 3 bytes first, then with 1 byte to fix https://github.com/js-cookie/java-cookie/issues/14
return decode( decode( encoded, 3 ), 1 );
}

private String decode( String encoded, Integer bytesPerCharacter ) {
String decoded = encoded;
Pattern pattern = Pattern.compile( "(%[0-9A-Z]{2})+" );
Pattern pattern = Pattern.compile( "(%[0-9A-Z]{2}){" + bytesPerCharacter + "}" );
Matcher matcher = pattern.matcher( encoded );
while ( matcher.find() ) {
String encodedChar = matcher.group();
Expand Down Expand Up @@ -392,14 +396,13 @@ private String decodeValue( String encodedValue, String decodedName ) {
private Map<String, String> getCookies( String cookieHeader ) {
Map<String, String> result = new HashMap<String, String>();
String[] cookies = cookieHeader.split( "; " );
for ( int i = 0; i < cookies.length; i++ ) {
String cookie = cookies[ i ];
String encodedName = cookie.split( "=" )[ 0 ];
String decodedName = decode( encodedName );

String encodedValue = cookie.substring( cookie.indexOf( '=' ) + 1, cookie.length() );
String decodedValue = decodeValue( encodedValue, decodedName );
result.put( decodedName, decodedValue );
for (String cookie : cookies) {
String encodedName = cookie.split("=")[0];
String decodedName = decode(encodedName);

String encodedValue = cookie.substring(cookie.indexOf('=') + 1);
String decodedValue = decodeValue(encodedValue, decodedName);
result.put(decodedName, decodedValue);
}
return result;
}
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,86 @@
package com.github.jscookie.javacookie.test.integration;

import com.github.jscookie.javacookie.test.integration.test.utils.Debug;
import com.github.jscookie.javacookie.test.unit.utils.IntegrationUtils;
import org.jboss.arquillian.container.test.api.Deployment;
import org.jboss.arquillian.container.test.api.RunAsClient;
import org.jboss.arquillian.junit.Arquillian;
import org.jboss.arquillian.test.api.ArquillianResource;
import org.jboss.shrinkwrap.api.spec.WebArchive;
import org.junit.Assert;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.openqa.selenium.By;
import org.openqa.selenium.Cookie;
import org.openqa.selenium.WebDriver;
import org.openqa.selenium.firefox.FirefoxDriver;

import java.io.File;
import java.net.URL;
import java.util.Set;
import java.util.concurrent.TimeUnit;

@RunWith(Arquillian.class)
public class CookiesWriteReadIT {
private static Debug debug = Debug.FALSE;

@Deployment
public static WebArchive createDeployment() {
WebArchive war = IntegrationUtils.createCommonDeployment()
.addAsWebInfResource(
new File("src/test/resources/web.xml"),
"web.xml"
);

System.out.println(" ----- LOGGING THE FILES ADDED TO JBOSS");
System.out.println(war.toString(true));
System.out.println(" ----- END OF LOGGING THE FILES ADDED TO JBOSS");

return war;
}

@RunAsClient
@Test
public void write_read(@ArquillianResource URL baseURL) {
WebDriver driver = new FirefoxDriver();
driver.manage().timeouts().implicitlyWait(20, TimeUnit.SECONDS);

// Set simple cookie
driver.navigate().to(baseURL + "write?name=c&value=v");
Set<Cookie> cookies = driver.manage().getCookies();
Assert.assertEquals(1, cookies.size());
Assert.assertEquals("c", cookies.iterator().next().getName());
Assert.assertEquals("v", cookies.iterator().next().getValue());

// Read cookie
driver.navigate().to(baseURL + "read?name=c");
Assert.assertEquals("{\"name\":\"c\",\"value\":\"v\"}", driver.findElement(By.id("json")).getText());

if (debug.is(false)) {
driver.quit();
}
}

@RunAsClient
@Test
public void write_read_json(@ArquillianResource URL baseURL) {
WebDriver driver = new FirefoxDriver();
driver.manage().timeouts().implicitlyWait(20, TimeUnit.SECONDS);

// Set JSON cookie
driver.navigate().to(baseURL + "write_json?name=c&value=v");
Set<Cookie> cookies = driver.manage().getCookies();
Assert.assertEquals(1, cookies.size());
Assert.assertEquals("c", cookies.iterator().next().getName());
Assert.assertEquals("{%22attr1%22:%22v%22%2C%22attr2%22:%22v%22}", cookies.iterator().next().getValue());

// Read cookie
driver.navigate().to(baseURL + "read_json?name=c");
Assert.assertEquals("{\"attr1\":\"v\",\"attr2\":\"v\"}", driver.findElement(By.id("json")).getText());

if (debug.is(false)) {
driver.quit();
}
}

}
Original file line number Diff line number Diff line change
@@ -0,0 +1,27 @@
package com.github.jscookie.javacookie.test.integration;

import com.fasterxml.jackson.databind.ObjectMapper;
import com.github.jscookie.javacookie.Cookies;
import com.github.jscookie.javacookie.test.integration.test.utils.IntegrationTestUtils;
import com.github.jscookie.javacookie.test.integration.test.utils.Result;

import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;

public class ReadCookieServlet extends HttpServlet {

@Override
public void doGet(HttpServletRequest request, HttpServletResponse response) throws IOException {
Cookies cookies = Cookies.initFromServlet(request, response);

String name = IntegrationTestUtils.getUTF8Param("name", request);
String value = cookies.get(name);

response.setContentType("application/json");
new ObjectMapper()
.writeValue(response.getOutputStream(), new Result(name, value));
}

}
Original file line number Diff line number Diff line change
@@ -0,0 +1,34 @@
package com.github.jscookie.javacookie.test.integration;

import com.fasterxml.jackson.databind.ObjectMapper;
import com.github.jscookie.javacookie.CookieParseException;
import com.github.jscookie.javacookie.Cookies;
import com.github.jscookie.javacookie.test.integration.test.utils.CustomJSONCookie;
import com.github.jscookie.javacookie.test.integration.test.utils.IntegrationTestUtils;

import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;

public class ReadJSONCookieServlet extends HttpServlet {

@Override
public void doGet(HttpServletRequest request, HttpServletResponse response) throws IOException {
Cookies cookies = Cookies.initFromServlet(request, response);

String name = IntegrationTestUtils.getUTF8Param("name", request);

CustomJSONCookie customJSONCookie;
try {
customJSONCookie = cookies.get(name, CustomJSONCookie.class);
if (customJSONCookie != null) {
response.setContentType("application/json");
new ObjectMapper().writeValue(response.getOutputStream(), customJSONCookie);
}
} catch (CookieParseException e) {
e.printStackTrace();
}

}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
package com.github.jscookie.javacookie.test.integration;

import com.fasterxml.jackson.databind.ObjectMapper;
import com.github.jscookie.javacookie.Cookies;
import com.github.jscookie.javacookie.test.integration.test.utils.IntegrationTestUtils;
import com.github.jscookie.javacookie.test.integration.test.utils.Result;

import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;

public class WriteCookieServlet extends HttpServlet {

@Override
public void doGet(HttpServletRequest request, HttpServletResponse response) throws IOException {
Cookies cookies = Cookies.initFromServlet(request, response);

String name = IntegrationTestUtils.getUTF8Param("name", request);
String value = IntegrationTestUtils.getUTF8Param("value", request);

cookies.set(name, value);

response.setContentType("application/json");
new ObjectMapper()
.writeValue(response.getOutputStream(), new Result(name, value));
}

}
Original file line number Diff line number Diff line change
@@ -0,0 +1,35 @@
package com.github.jscookie.javacookie.test.integration;

import com.fasterxml.jackson.databind.ObjectMapper;
import com.github.jscookie.javacookie.CookieSerializationException;
import com.github.jscookie.javacookie.Cookies;
import com.github.jscookie.javacookie.test.integration.test.utils.CustomJSONCookie;
import com.github.jscookie.javacookie.test.integration.test.utils.IntegrationTestUtils;

import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;

public class WriteJSONCookieServlet extends HttpServlet {

@Override
public void doGet(HttpServletRequest request, HttpServletResponse response) throws IOException {
Cookies cookies = Cookies.initFromServlet(request, response);

String name = IntegrationTestUtils.getUTF8Param("name", request);
String value = IntegrationTestUtils.getUTF8Param("value", request);

CustomJSONCookie customJSONCookie = new CustomJSONCookie(value, value);

try {
cookies.set(name, customJSONCookie);
} catch (CookieSerializationException e) {
e.printStackTrace();
}

response.setContentType("application/json");
new ObjectMapper()
.writeValue(response.getOutputStream(), customJSONCookie);
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -11,13 +11,15 @@

import com.fasterxml.jackson.databind.ObjectMapper;
import com.github.jscookie.javacookie.Cookies;
import com.github.jscookie.javacookie.test.integration.test.utils.IntegrationTestUtils;
import com.github.jscookie.javacookie.test.integration.test.utils.Result;

public class EncodingServlet extends HttpServlet {
private static final long serialVersionUID = 1;
@Override
public void doGet( HttpServletRequest request, HttpServletResponse response )
throws ServletException, IOException {
String name = getUTF8Param( "name", request );
String name = IntegrationTestUtils.getUTF8Param( "name", request );

System.out.println( "Testing: " + name );

Expand All @@ -35,31 +37,4 @@ public void doGet( HttpServletRequest request, HttpServletResponse response )
.writeValue( response.getOutputStream(), new Result( name, value ) );
}

/**
* Retrieves the parameter using UTF-8 charset since the server default is ISO-8859-1
*/
private String getUTF8Param( String name, HttpServletRequest request ) throws UnsupportedEncodingException {
String query = request.getQueryString();
for ( String pair : query.split( "&" ) ) {
if ( name.equals( pair.split( "=" )[ 0 ] ) ) {
return URLDecoder.decode( pair.split( "=" )[ 1 ], "UTF-8" );
}
}
return null;
}
}

class Result {
private String name;
private String value;
Result( String name, String value ) {
this.name = name;
this.value = value;
}
public String getName() {
return name;
}
public String getValue() {
return value;
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
package com.github.jscookie.javacookie.test.integration.test.utils;

import com.fasterxml.jackson.annotation.JsonCreator;
import com.fasterxml.jackson.annotation.JsonProperty;
import com.github.jscookie.javacookie.CookieValue;

public class CustomJSONCookie implements CookieValue {
private String attr1;
private String attr2;

@JsonCreator
public CustomJSONCookie(@JsonProperty("attr1") String attr1, @JsonProperty("attr2") String attr2) {
this.attr1 = attr1;
this.attr2 = attr2;
}

public String getAttr1() {
return attr1;
}

public String getAttr2() {
return attr2;
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
package com.github.jscookie.javacookie.test.integration.test.utils;

import javax.servlet.http.HttpServletRequest;
import java.io.UnsupportedEncodingException;
import java.net.URLDecoder;

public class IntegrationTestUtils {
/**
* Retrieves the parameter using UTF-8 charset since the server default is ISO-8859-1
*/
public static String getUTF8Param( String name, HttpServletRequest request ) throws UnsupportedEncodingException {
String query = request.getQueryString();
for ( String pair : query.split( "&" ) ) {
if ( name.equals( pair.split( "=" )[ 0 ] ) ) {
return URLDecoder.decode( pair.split( "=" )[ 1 ], "UTF-8" );
}
}
return null;
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
package com.github.jscookie.javacookie.test.integration.test.utils;

public class Result {
private String name;
private String value;
public Result( String name, String value ) {
this.name = name;
this.value = value;
}
public String getName() {
return name;
}
public String getValue() {
return value;
}
}
Loading