All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.google.gwt.query.vm.CookieManager Maven / Gradle / Ivy

The newest version!
/*
 * Copyright 2014, The gwtquery team.
 *
 * Licensed 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.
 */
package com.google.gwt.query.vm;

import java.io.IOException;
import java.net.URL;
import java.net.URLConnection;
import java.text.DateFormat;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;
import java.util.StringTokenizer;

/**
 * http://www.hccp.org/java-net-cookie-how-to.html.
 */
public class CookieManager {

  private Map>> store =
      new HashMap<>();

  private static final String SET_COOKIE = "Set-Cookie";
  private static final String COOKIE_VALUE_DELIMITER = ";";
  private static final String PATH = "path";
  private static final String EXPIRES = "expires";
  private static final String DATE_FORMAT = "EEE, dd-MMM-yyyy hh:mm:ss z";
  private static final String SET_COOKIE_SEPARATOR = "; ";
  private static final String COOKIE = "Cookie";

  private static final char NAME_VALUE_SEPARATOR = '=';
  private static final char DOT = '.';

  private DateFormat dateFormat = new SimpleDateFormat(DATE_FORMAT);;

  private static CookieManager cookieManager = new CookieManager();

  public static CookieManager getInstance() {
    return cookieManager;
  }

  public void clear() {
    store.clear();
  }

  public void removeDomainCookies(String domain) {
    store.remove(domain);
  }

  public void removeDomainCookie(String domain, String... cookies) {
    Map> domainStore = store.get(domain);
    if (domainStore != null) {
      for (String cookie : cookies) {
        domainStore.remove(cookie);
      }
    }
  }

  public void setDomcainCookie(String host, String name, String value) {
    setDomcainCookieProperty(host, name, name, value);
  }

  public void setDomcainCookieProperty(String host, String name, String prop, String value) {
    String domain = getDomainFromHost(host);
    Map> domainStore = store.get(domain);
    if (domainStore == null) {
      domainStore = new HashMap<>();
      store.put(domain, domainStore);
    }
    Map cookie = domainStore.get(name);
    if (cookie == null) {
      cookie = new HashMap<>();
      domainStore.put(name, cookie);
    }
    if (value == null) {
      cookie.remove(prop);
    } else {
      cookie.put(prop, value);
    }
  }

  /**
   * Retrieves and stores cookies returned by the host on the other side of the the open
   * java.net.URLConnection.
   * 
   * The connection MUST have been opened using the connect() method or a IOException will be
   * thrown.
   * 
   * @param conn a java.net.URLConnection - must be open, or IOException will be thrown
   * @throws java.io.IOException Thrown if conn is not open.
   */
  public void storeCookies(URLConnection conn) throws IOException {
    // let's determine the domain from where these cookies are being sent
    String domain = getDomainFromHost(conn.getURL().getHost());

    Map> domainStore; // this is where we will store cookies for this domain

    // now let's check the store to see if we have an entry for this domain
    if (store.containsKey(domain)) {
      // we do, so lets retrieve it from the store
      domainStore = store.get(domain);
    } else {
      // we don't, so let's create it and put it in the store
      domainStore = new HashMap<>();
      store.put(domain, domainStore);
    }

    // OK, now we are ready to get the cookies out of the URLConnection

    String headerName = null;
    for (int i = 1; (headerName = conn.getHeaderFieldKey(i)) != null; i++) {
      if (headerName.equalsIgnoreCase(SET_COOKIE)) {
        Map cookie = new HashMap<>();
        StringTokenizer st = new StringTokenizer(conn.getHeaderField(i), COOKIE_VALUE_DELIMITER);

        // the specification dictates that the first name/value pair
        // in the string is the cookie name and value, so let's handle
        // them as a special case:

        if (st.hasMoreTokens()) {
          String token = st.nextToken();
          String name = token.substring(0, token.indexOf(NAME_VALUE_SEPARATOR));
          String value = token.substring(token.indexOf(NAME_VALUE_SEPARATOR) + 1, token.length());
          domainStore.put(name, cookie);
          cookie.put(name, value);
        }

        while (st.hasMoreTokens()) {
          String token = st.nextToken().toLowerCase();
          int idx = token.indexOf(NAME_VALUE_SEPARATOR);
          if (idx > 0 && idx < token.length() - 1) {
            cookie.put(token.substring(0, idx).toLowerCase(), token.substring(idx + 1, token
                .length()));
          }
        }
      }
    }
  }

  /**
   * Prior to opening a URLConnection, calling this method will set all unexpired cookies that match
   * the path or subpaths for thi underlying URL
   * 
   * The connection MUST NOT have been opened method or an IOException will be thrown.
   * 
   * @param conn a java.net.URLConnection - must NOT be open, or IOException will be thrown
   * @throws java.io.IOException Thrown if conn has already been opened.
   */
  public void setCookies(URLConnection conn) throws IOException {

    // let's determine the domain and path to retrieve the appropriate cookies
    URL url = conn.getURL();
    String domain = getDomainFromHost(url.getHost());
    if (domain.equals("localhost")) {
      domain = "linkedin.com";
    }
    String path = url.getPath();

    Map> domainStore = store.get(domain);
    if (domainStore == null)
      return;
    StringBuilder cookieStringBuffer = new StringBuilder();

    Iterator cookieNames = domainStore.keySet().iterator();
    while (cookieNames.hasNext()) {
      String cookieName = cookieNames.next();
      Map cookie = domainStore.get(cookieName);
      // check cookie to ensure path matches and cookie is not expired
      // if all is cool, add cookie to header string
      if (comparePaths((String) cookie.get(PATH), path)
          && isNotExpired(cookie.get(EXPIRES))) {
        cookieStringBuffer.append(cookieName);
        cookieStringBuffer.append("=");
        cookieStringBuffer.append(cookie.get(cookieName));
        if (cookieNames.hasNext())
          cookieStringBuffer.append(SET_COOKIE_SEPARATOR);
      }
    }
    try {
      conn.setRequestProperty(COOKIE, cookieStringBuffer.toString());
    } catch (java.lang.IllegalStateException ise) {
      IOException ioe =
          new IOException(
              "Illegal State! Cookies cannot be set on a URLConnection that is already connected. "
                  + "Only call setCookies(java.net.URLConnection) AFTER calling java.net.URLConnection.connect().");
      throw ioe;
    }
  }

  private String getDomainFromHost(String host) {
    host = host.toLowerCase().replaceFirst("https?://", "").replaceAll("[/?:].*$", "");
    if (host.indexOf(DOT) != host.lastIndexOf(DOT)) {
      return host.substring(host.indexOf(DOT) + 1);
    } else {
      return host;
    }
  }

  private boolean isNotExpired(String cookieExpires) {
    if (cookieExpires == null)
      return true;
    Date now = new Date();
    try {
      return (now.compareTo(dateFormat.parse(cookieExpires))) <= 0;
    } catch (java.text.ParseException pe) {
      pe.printStackTrace();
      return false;
    }
  }

  private boolean comparePaths(String cookiePath, String targetPath) {
    if (cookiePath == null) {
      return true;
    } else if (cookiePath.equals("/")) {
      return true;
    } else if (targetPath.regionMatches(0, cookiePath, 0, cookiePath.length())) {
      return true;
    } else {
      return false;
    }
  }

  public String toString() {
    return store.toString();
  }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy