blob: fcca1c3f078c05264de33a3cbbbf23b39c0eed13 [file] [log] [blame]
/*
* ====================================================================
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
* ====================================================================
*
* This software consists of voluntary contributions made by many
* individuals on behalf of the Apache Software Foundation. For more
* information on the Apache Software Foundation, please see
* <http://www.apache.org/>.
*
*/
package org.apache.hc.client5.http.impl.cookie;
import java.time.Instant;
import java.time.Month;
import java.time.ZoneId;
import java.time.ZonedDateTime;
import java.util.BitSet;
import java.util.Locale;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import org.apache.hc.client5.http.cookie.CommonCookieAttributeHandler;
import org.apache.hc.client5.http.cookie.Cookie;
import org.apache.hc.client5.http.cookie.MalformedCookieException;
import org.apache.hc.client5.http.cookie.SetCookie;
import org.apache.hc.client5.http.utils.DateUtils;
import org.apache.hc.core5.annotation.Contract;
import org.apache.hc.core5.annotation.ThreadingBehavior;
import org.apache.hc.core5.util.Args;
import org.apache.hc.core5.util.TextUtils;
import org.apache.hc.core5.util.Tokenizer;
/**
* Cookie {@code expires} attribute handler conformant to the more relaxed interpretation
* of HTTP state management.
*
* @since 4.4
*/
@Contract(threading = ThreadingBehavior.STATELESS)
public class LaxExpiresHandler extends AbstractCookieAttributeHandler implements CommonCookieAttributeHandler {
private static final BitSet DELIMS;
static {
final BitSet bitSet = new BitSet();
bitSet.set(0x9);
for (int b = 0x20; b <= 0x2f; b++) {
bitSet.set(b);
}
for (int b = 0x3b; b <= 0x40; b++) {
bitSet.set(b);
}
for (int b = 0x5b; b <= 0x60; b++) {
bitSet.set(b);
}
for (int b = 0x7b; b <= 0x7e; b++) {
bitSet.set(b);
}
DELIMS = bitSet;
}
private static final Map<String, Month> MONTHS;
static {
final ConcurrentHashMap<String, Month> map = new ConcurrentHashMap<>(12);
map.put("jan", Month.JANUARY);
map.put("feb", Month.FEBRUARY);
map.put("mar", Month.MARCH);
map.put("apr", Month.APRIL);
map.put("may", Month.MAY);
map.put("jun", Month.JUNE);
map.put("jul", Month.JULY);
map.put("aug", Month.AUGUST);
map.put("sep", Month.SEPTEMBER);
map.put("oct", Month.OCTOBER);
map.put("nov", Month.NOVEMBER);
map.put("dec", Month.DECEMBER);
MONTHS = map;
}
private final static Pattern TIME_PATTERN = Pattern.compile(
"^([0-9]{1,2}):([0-9]{1,2}):([0-9]{1,2})([^0-9].*)?$");
private final static Pattern DAY_OF_MONTH_PATTERN = Pattern.compile(
"^([0-9]{1,2})([^0-9].*)?$");
private final static Pattern MONTH_PATTERN = Pattern.compile(
"^(jan|feb|mar|apr|may|jun|jul|aug|sep|oct|nov|dec)(.*)?$", Pattern.CASE_INSENSITIVE);
private final static Pattern YEAR_PATTERN = Pattern.compile(
"^([0-9]{2,4})([^0-9].*)?$");
public LaxExpiresHandler() {
super();
}
@Override
public void parse(final SetCookie cookie, final String value) throws MalformedCookieException {
Args.notNull(cookie, "Cookie");
if (TextUtils.isBlank(value)) {
return;
}
final Tokenizer.Cursor cursor = new Tokenizer.Cursor(0, value.length());
final StringBuilder content = new StringBuilder();
int second = 0, minute = 0, hour = 0, day = 0, year = 0;
Month month = Month.JANUARY;
boolean foundTime = false, foundDayOfMonth = false, foundMonth = false, foundYear = false;
try {
while (!cursor.atEnd()) {
skipDelims(value, cursor);
content.setLength(0);
copyContent(value, cursor, content);
if (content.length() == 0) {
break;
}
if (!foundTime) {
final Matcher matcher = TIME_PATTERN.matcher(content);
if (matcher.matches()) {
foundTime = true;
hour = Integer.parseInt(matcher.group(1));
minute = Integer.parseInt(matcher.group(2));
second =Integer.parseInt(matcher.group(3));
continue;
}
}
if (!foundDayOfMonth) {
final Matcher matcher = DAY_OF_MONTH_PATTERN.matcher(content);
if (matcher.matches()) {
foundDayOfMonth = true;
day = Integer.parseInt(matcher.group(1));
continue;
}
}
if (!foundMonth) {
final Matcher matcher = MONTH_PATTERN.matcher(content);
if (matcher.matches()) {
foundMonth = true;
month = MONTHS.get(matcher.group(1).toLowerCase(Locale.ROOT));
continue;
}
}
if (!foundYear) {
final Matcher matcher = YEAR_PATTERN.matcher(content);
if (matcher.matches()) {
foundYear = true;
year = Integer.parseInt(matcher.group(1));
continue;
}
}
}
} catch (final NumberFormatException ignore) {
throw new MalformedCookieException("Invalid 'expires' attribute: " + value);
}
if (!foundTime || !foundDayOfMonth || !foundMonth || !foundYear) {
throw new MalformedCookieException("Invalid 'expires' attribute: " + value);
}
if (year >= 70 && year <= 99) {
year = 1900 + year;
}
if (year >= 0 && year <= 69) {
year = 2000 + year;
}
if (day < 1 || day > 31 || year < 1601 || hour > 23 || minute > 59 || second > 59) {
throw new MalformedCookieException("Invalid 'expires' attribute: " + value);
}
final Instant expiryDate = ZonedDateTime.of(year, month.getValue(), day, hour, minute, second, 0,
ZoneId.of("UTC")).toInstant();
cookie.setExpiryDate(DateUtils.toDate(expiryDate));
}
private void skipDelims(final CharSequence buf, final Tokenizer.Cursor cursor) {
int pos = cursor.getPos();
final int indexFrom = cursor.getPos();
final int indexTo = cursor.getUpperBound();
for (int i = indexFrom; i < indexTo; i++) {
final char current = buf.charAt(i);
if (DELIMS.get(current)) {
pos++;
} else {
break;
}
}
cursor.updatePos(pos);
}
private void copyContent(final CharSequence buf, final Tokenizer.Cursor cursor, final StringBuilder dst) {
int pos = cursor.getPos();
final int indexFrom = cursor.getPos();
final int indexTo = cursor.getUpperBound();
for (int i = indexFrom; i < indexTo; i++) {
final char current = buf.charAt(i);
if (DELIMS.get(current)) {
break;
}
pos++;
dst.append(current);
}
cursor.updatePos(pos);
}
@Override
public String getAttributeName() {
return Cookie.EXPIRES_ATTR;
}
}