git-svn-id: https://svn.berlios.de/svnroot/repos/yacy/trunk@2495 6c8d7289-2bf4-0310-a012-ef5d649a1542
parent
4825bfaaf3
commit
dae763d8e3
@ -0,0 +1,136 @@
|
|||||||
|
// IResourceInfo.java
|
||||||
|
// -------------------------------------
|
||||||
|
// part of YACY
|
||||||
|
// (C) by Michael Peter Christen; mc@anomic.de
|
||||||
|
// first published on http://www.anomic.de
|
||||||
|
// Frankfurt, Germany, 2006
|
||||||
|
//
|
||||||
|
// This file ist contributed by Martin Thelian
|
||||||
|
//
|
||||||
|
// $LastChangedDate: 2006-02-20 23:57:42 +0100 (Mo, 20 Feb 2006) $
|
||||||
|
// $LastChangedRevision: 1715 $
|
||||||
|
// $LastChangedBy: theli $
|
||||||
|
//
|
||||||
|
// This program is free software; you can redistribute it and/or modify
|
||||||
|
// it under the terms of the GNU General Public License as published by
|
||||||
|
// the Free Software Foundation; either version 2 of the License, or
|
||||||
|
// (at your option) any later version.
|
||||||
|
//
|
||||||
|
// This program 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 General Public License for more details.
|
||||||
|
//
|
||||||
|
// You should have received a copy of the GNU General Public License
|
||||||
|
// along with this program; if not, write to the Free Software
|
||||||
|
// Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
//
|
||||||
|
// Using this software in any meaning (reading, learning, copying, compiling,
|
||||||
|
// running) means that you agree that the Author(s) is (are) not responsible
|
||||||
|
// for cost, loss of data or any harm that may be caused directly or indirectly
|
||||||
|
// by usage of this softare or this documentation. The usage of this software
|
||||||
|
// is on your own risk. The installation and usage (starting/running) of this
|
||||||
|
// software may allow other people or application to access your computer and
|
||||||
|
// any attached devices and is highly dependent on the configuration of the
|
||||||
|
// software which must be done by the user of the software; the author(s) is
|
||||||
|
// (are) also not responsible for proper configuration and usage of the
|
||||||
|
// software, even if provoked by documentation provided together with
|
||||||
|
// the software.
|
||||||
|
//
|
||||||
|
// Any changes to this file according to the GPL as documented in the file
|
||||||
|
// gpl.txt aside this file in the shipment you received can be done to the
|
||||||
|
// lines that follows this copyright notice here, but changes must not be
|
||||||
|
// done inside the copyright notive above. A re-distribution must contain
|
||||||
|
// the intact and unchanged copyright notice.
|
||||||
|
// Contributions and changes to the program code must be marked as such.
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
package de.anomic.plasma.cache;
|
||||||
|
|
||||||
|
import java.util.Date;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
import de.anomic.net.URL;
|
||||||
|
|
||||||
|
public interface IResourceInfo {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return the resource information as map
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
public Map getMap();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the URL of this content
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
public URL getUrl();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the referer URL of this URL
|
||||||
|
* @return referer URL
|
||||||
|
*/
|
||||||
|
public URL getRefererUrl();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the mimetype of the cached object
|
||||||
|
* @return mimetype
|
||||||
|
*/
|
||||||
|
public String getMimeType();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the modification date of the cached object
|
||||||
|
* @return the modifiaction date
|
||||||
|
*/
|
||||||
|
public Date getModificationDate();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the url hash of the content URL
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
public String getUrlHash();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Specifies if the resource was requested with a
|
||||||
|
* if modified since date
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
public Date ifModifiedSince();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Specifies if the resource was requested with
|
||||||
|
* client specific information (e.g. cookies for http)
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
public boolean requestWithCookie();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Specifies if the request prohibits indexing
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
public boolean requestProhibitsIndexing();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Determines if a resource that was downloaded by the crawler
|
||||||
|
* is allowed to be indexed.
|
||||||
|
*
|
||||||
|
* @return an error string describing the reason why the
|
||||||
|
* resourse should not be indexed or null if indexing is allowed
|
||||||
|
*/
|
||||||
|
public String shallIndexCacheForCrawler();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Determines if a resource that was downloaded by the proxy
|
||||||
|
* is allowed to be indexed.
|
||||||
|
*
|
||||||
|
* @return an error string describing the reason why the
|
||||||
|
* resourse should not be indexed or null if indexing is allowed
|
||||||
|
*/
|
||||||
|
public String shallIndexCacheForProxy();
|
||||||
|
|
||||||
|
public String shallStoreCacheForProxy();
|
||||||
|
public boolean shallUseCacheForProxy();
|
||||||
|
|
||||||
|
public boolean validResponseStatus(String responseStatus);
|
||||||
|
}
|
@ -0,0 +1,86 @@
|
|||||||
|
// RespourceInfoFactory.java
|
||||||
|
// -------------------------------------
|
||||||
|
// part of YACY
|
||||||
|
// (C) by Michael Peter Christen; mc@anomic.de
|
||||||
|
// first published on http://www.anomic.de
|
||||||
|
// Frankfurt, Germany, 2006
|
||||||
|
//
|
||||||
|
// This file ist contributed by Martin Thelian
|
||||||
|
//
|
||||||
|
// $LastChangedDate: 2006-02-20 23:57:42 +0100 (Mo, 20 Feb 2006) $
|
||||||
|
// $LastChangedRevision: 1715 $
|
||||||
|
// $LastChangedBy: theli $
|
||||||
|
//
|
||||||
|
// This program is free software; you can redistribute it and/or modify
|
||||||
|
// it under the terms of the GNU General Public License as published by
|
||||||
|
// the Free Software Foundation; either version 2 of the License, or
|
||||||
|
// (at your option) any later version.
|
||||||
|
//
|
||||||
|
// This program 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 General Public License for more details.
|
||||||
|
//
|
||||||
|
// You should have received a copy of the GNU General Public License
|
||||||
|
// along with this program; if not, write to the Free Software
|
||||||
|
// Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
//
|
||||||
|
// Using this software in any meaning (reading, learning, copying, compiling,
|
||||||
|
// running) means that you agree that the Author(s) is (are) not responsible
|
||||||
|
// for cost, loss of data or any harm that may be caused directly or indirectly
|
||||||
|
// by usage of this softare or this documentation. The usage of this software
|
||||||
|
// is on your own risk. The installation and usage (starting/running) of this
|
||||||
|
// software may allow other people or application to access your computer and
|
||||||
|
// any attached devices and is highly dependent on the configuration of the
|
||||||
|
// software which must be done by the user of the software; the author(s) is
|
||||||
|
// (are) also not responsible for proper configuration and usage of the
|
||||||
|
// software, even if provoked by documentation provided together with
|
||||||
|
// the software.
|
||||||
|
//
|
||||||
|
// Any changes to this file according to the GPL as documented in the file
|
||||||
|
// gpl.txt aside this file in the shipment you received can be done to the
|
||||||
|
// lines that follows this copyright notice here, but changes must not be
|
||||||
|
// done inside the copyright notive above. A re-distribution must contain
|
||||||
|
// the intact and unchanged copyright notice.
|
||||||
|
// Contributions and changes to the program code must be marked as such.
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
package de.anomic.plasma.cache;
|
||||||
|
|
||||||
|
import java.lang.reflect.Constructor;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
import de.anomic.net.URL;
|
||||||
|
|
||||||
|
public class ResourceInfoFactory {
|
||||||
|
public IResourceInfo buildResourceInfoObj(
|
||||||
|
URL resourceURL,
|
||||||
|
Map resourceMetadata
|
||||||
|
) throws Exception {
|
||||||
|
|
||||||
|
String protocString = resourceURL.getProtocol();
|
||||||
|
|
||||||
|
// the full qualified class name
|
||||||
|
String className = this.getClass().getPackage().getName() + "." + protocString + ".ResourceInfo";
|
||||||
|
|
||||||
|
// loading class by name
|
||||||
|
Class moduleClass = Class.forName(className);
|
||||||
|
|
||||||
|
// getting the constructor
|
||||||
|
Constructor classConstructor = moduleClass.getConstructor( new Class[] {
|
||||||
|
URL.class,
|
||||||
|
Map.class
|
||||||
|
} );
|
||||||
|
|
||||||
|
// instantiating class
|
||||||
|
IResourceInfo infoObject = (IResourceInfo) classConstructor.newInstance(new Object[] {
|
||||||
|
resourceURL,
|
||||||
|
resourceMetadata
|
||||||
|
});
|
||||||
|
|
||||||
|
// return the newly created object
|
||||||
|
return infoObject;
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,467 @@
|
|||||||
|
// ResourceInfo.java
|
||||||
|
// -------------------------------------
|
||||||
|
// part of YACY
|
||||||
|
// (C) by Michael Peter Christen; mc@anomic.de
|
||||||
|
// first published on http://www.anomic.de
|
||||||
|
// Frankfurt, Germany, 2006
|
||||||
|
//
|
||||||
|
// This file ist contributed by Martin Thelian
|
||||||
|
//
|
||||||
|
// $LastChangedDate: 2006-02-20 23:57:42 +0100 (Mo, 20 Feb 2006) $
|
||||||
|
// $LastChangedRevision: 1715 $
|
||||||
|
// $LastChangedBy: theli $
|
||||||
|
//
|
||||||
|
// This program is free software; you can redistribute it and/or modify
|
||||||
|
// it under the terms of the GNU General Public License as published by
|
||||||
|
// the Free Software Foundation; either version 2 of the License, or
|
||||||
|
// (at your option) any later version.
|
||||||
|
//
|
||||||
|
// This program 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 General Public License for more details.
|
||||||
|
//
|
||||||
|
// You should have received a copy of the GNU General Public License
|
||||||
|
// along with this program; if not, write to the Free Software
|
||||||
|
// Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
||||||
|
//
|
||||||
|
// Using this software in any meaning (reading, learning, copying, compiling,
|
||||||
|
// running) means that you agree that the Author(s) is (are) not responsible
|
||||||
|
// for cost, loss of data or any harm that may be caused directly or indirectly
|
||||||
|
// by usage of this softare or this documentation. The usage of this software
|
||||||
|
// is on your own risk. The installation and usage (starting/running) of this
|
||||||
|
// software may allow other people or application to access your computer and
|
||||||
|
// any attached devices and is highly dependent on the configuration of the
|
||||||
|
// software which must be done by the user of the software; the author(s) is
|
||||||
|
// (are) also not responsible for proper configuration and usage of the
|
||||||
|
// software, even if provoked by documentation provided together with
|
||||||
|
// the software.
|
||||||
|
//
|
||||||
|
// Any changes to this file according to the GPL as documented in the file
|
||||||
|
// gpl.txt aside this file in the shipment you received can be done to the
|
||||||
|
// lines that follows this copyright notice here, but changes must not be
|
||||||
|
// done inside the copyright notive above. A re-distribution must contain
|
||||||
|
// the intact and unchanged copyright notice.
|
||||||
|
// Contributions and changes to the program code must be marked as such.
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
package de.anomic.plasma.cache.http;
|
||||||
|
|
||||||
|
import java.util.Date;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
import de.anomic.http.httpHeader;
|
||||||
|
import de.anomic.index.indexURL;
|
||||||
|
import de.anomic.net.URL;
|
||||||
|
import de.anomic.plasma.plasmaHTCache;
|
||||||
|
import de.anomic.plasma.cache.ResourceInfoFactory;
|
||||||
|
import de.anomic.plasma.cache.IResourceInfo;
|
||||||
|
import de.anomic.server.serverDate;
|
||||||
|
|
||||||
|
public class ResourceInfo implements IResourceInfo {
|
||||||
|
private URL url;
|
||||||
|
private String urlHash;
|
||||||
|
private httpHeader responseHeader;
|
||||||
|
private httpHeader requestHeader;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Constructor used by the {@link ResourceInfoFactory}
|
||||||
|
* @param objectURL
|
||||||
|
* @param objectInfo
|
||||||
|
*/
|
||||||
|
public ResourceInfo(URL objectURL, Map objectInfo) {
|
||||||
|
if (objectURL == null) throw new NullPointerException();
|
||||||
|
if (objectInfo == null) throw new NullPointerException();
|
||||||
|
|
||||||
|
// generating the url hash
|
||||||
|
this.url = objectURL;
|
||||||
|
this.urlHash = indexURL.urlHash(this.url.toNormalform());
|
||||||
|
|
||||||
|
// create the http header object
|
||||||
|
this.responseHeader = new httpHeader(null, objectInfo);
|
||||||
|
}
|
||||||
|
|
||||||
|
public ResourceInfo(URL objectURL, httpHeader requestHeaders, httpHeader responseHeaders) {
|
||||||
|
if (objectURL == null) throw new NullPointerException();
|
||||||
|
if (responseHeaders == null) throw new NullPointerException();
|
||||||
|
|
||||||
|
// generating the url hash
|
||||||
|
this.url = objectURL;
|
||||||
|
this.urlHash = indexURL.urlHash(this.url.toNormalform());
|
||||||
|
|
||||||
|
this.requestHeader = requestHeaders;
|
||||||
|
this.responseHeader = responseHeaders;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Map getMap() {
|
||||||
|
return this.responseHeader;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see de.anomic.plasma.cache.IResourceInfo#getMimeType()
|
||||||
|
*/
|
||||||
|
public String getMimeType() {
|
||||||
|
if (this.responseHeader == null) return null;
|
||||||
|
|
||||||
|
String mimeType = this.responseHeader.mime();
|
||||||
|
mimeType = mimeType.trim().toLowerCase();
|
||||||
|
|
||||||
|
int pos = mimeType.indexOf(';');
|
||||||
|
return ((pos < 0) ? mimeType : mimeType.substring(0, pos));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see de.anomic.plasma.cache.IResourceInfo#getModificationDate()
|
||||||
|
*/
|
||||||
|
public Date getModificationDate() {
|
||||||
|
Date docDate = null;
|
||||||
|
|
||||||
|
if (this.responseHeader != null) {
|
||||||
|
docDate = this.responseHeader.lastModified();
|
||||||
|
if (docDate == null) docDate = this.responseHeader.date();
|
||||||
|
}
|
||||||
|
if (docDate == null) docDate = new Date(serverDate.correctedUTCTime());
|
||||||
|
|
||||||
|
return docDate;
|
||||||
|
}
|
||||||
|
|
||||||
|
public URL getRefererUrl() {
|
||||||
|
if (this.requestHeader == null) return null;
|
||||||
|
try {
|
||||||
|
return new URL((String) this.requestHeader.get(httpHeader.REFERER, ""));
|
||||||
|
} catch (Exception e) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see de.anomic.plasma.cache.IResourceInfo#getUrl()
|
||||||
|
*/
|
||||||
|
public URL getUrl() {
|
||||||
|
return this.url;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see de.anomic.plasma.cache.IResourceInfo#getUrlHash()
|
||||||
|
*/
|
||||||
|
public String getUrlHash() {
|
||||||
|
return this.urlHash;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see de.anomic.plasma.cache.IResourceInfo#shallIndexCacheForCrawler()
|
||||||
|
*/
|
||||||
|
public String shallIndexCacheForCrawler() {
|
||||||
|
String mimeType = this.getMimeType();
|
||||||
|
if (plasmaHTCache.isPicture(mimeType)) { return "Media_Content_(Picture)"; }
|
||||||
|
if (!plasmaHTCache.isText(mimeType)) { return "Media_Content_(not_text)"; }
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @see de.anomic.plasma.cache.IResourceInfo#shallIndexCacheForProxy()
|
||||||
|
*/
|
||||||
|
public String shallIndexCacheForProxy() {
|
||||||
|
// -set-cookie in response
|
||||||
|
// the set-cookie from the server does not indicate that the content is special
|
||||||
|
// thus we do not care about it here for indexing
|
||||||
|
|
||||||
|
// a picture cannot be indexed
|
||||||
|
String mimeType = this.getMimeType();
|
||||||
|
if (plasmaHTCache.isPicture(mimeType)) {
|
||||||
|
return "Media_Content_(Picture)";
|
||||||
|
}
|
||||||
|
if (!plasmaHTCache.isText(mimeType)) {
|
||||||
|
return "Media_Content_(not_text)";
|
||||||
|
}
|
||||||
|
|
||||||
|
// -if-modified-since in request
|
||||||
|
// if the page is fresh at the very moment we can index it
|
||||||
|
Date ifModifiedSince = getModificationDate();
|
||||||
|
if ((ifModifiedSince != null) && (this.responseHeader.containsKey(httpHeader.LAST_MODIFIED))) {
|
||||||
|
// parse date
|
||||||
|
Date d = this.responseHeader.lastModified();
|
||||||
|
if (d == null) {
|
||||||
|
d = new Date(serverDate.correctedUTCTime());
|
||||||
|
}
|
||||||
|
// finally, we shall treat the cache as stale if the modification time is after the if-.. time
|
||||||
|
if (d.after(ifModifiedSince)) {
|
||||||
|
//System.out.println("***not indexed because if-modified-since");
|
||||||
|
return "Stale_(Last-Modified>Modified-Since)";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// -pragma in cached response
|
||||||
|
if (this.responseHeader.containsKey(httpHeader.PRAGMA) &&
|
||||||
|
((String) this.responseHeader.get(httpHeader.PRAGMA)).toUpperCase().equals("NO-CACHE")) {
|
||||||
|
return "Denied_(pragma_no_cache)";
|
||||||
|
}
|
||||||
|
|
||||||
|
// see for documentation also:
|
||||||
|
// http://www.web-caching.com/cacheability.html
|
||||||
|
|
||||||
|
// look for freshnes information
|
||||||
|
|
||||||
|
// -expires in cached response
|
||||||
|
// the expires value gives us a very easy hint when the cache is stale
|
||||||
|
// sometimes, the expires date is set to the past to prevent that a page is cached
|
||||||
|
// we use that information to see if we should index it
|
||||||
|
final Date expires = this.responseHeader.expires();
|
||||||
|
if (expires != null && expires.before(new Date(serverDate.correctedUTCTime()))) {
|
||||||
|
return "Stale_(Expired)";
|
||||||
|
}
|
||||||
|
|
||||||
|
// -lastModified in cached response
|
||||||
|
// this information is too weak to use it to prevent indexing
|
||||||
|
// even if we can apply a TTL heuristic for cache usage
|
||||||
|
|
||||||
|
// -cache-control in cached response
|
||||||
|
// the cache-control has many value options.
|
||||||
|
String cacheControl = (String) this.responseHeader.get(httpHeader.CACHE_CONTROL);
|
||||||
|
if (cacheControl != null) {
|
||||||
|
cacheControl = cacheControl.trim().toUpperCase();
|
||||||
|
/* we have the following cases for cache-control:
|
||||||
|
"public" -- can be indexed
|
||||||
|
"private", "no-cache", "no-store" -- cannot be indexed
|
||||||
|
"max-age=<delta-seconds>" -- stale/fresh dependent on date
|
||||||
|
*/
|
||||||
|
if (cacheControl.startsWith("PRIVATE") ||
|
||||||
|
cacheControl.startsWith("NO-CACHE") ||
|
||||||
|
cacheControl.startsWith("NO-STORE")) {
|
||||||
|
// easy case
|
||||||
|
return "Stale_(denied_by_cache-control=" + cacheControl + ")";
|
||||||
|
// } else if (cacheControl.startsWith("PUBLIC")) {
|
||||||
|
// // ok, do nothing
|
||||||
|
} else if (cacheControl.startsWith("MAX-AGE=")) {
|
||||||
|
// we need also the load date
|
||||||
|
final Date date = this.responseHeader.date();
|
||||||
|
if (date == null) {
|
||||||
|
return "Stale_(no_date_given_in_response)";
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
final long ttl = 1000 * Long.parseLong(cacheControl.substring(8)); // milliseconds to live
|
||||||
|
if (serverDate.correctedUTCTime() - date.getTime() > ttl) {
|
||||||
|
//System.out.println("***not indexed because cache-control");
|
||||||
|
return "Stale_(expired_by_cache-control)";
|
||||||
|
}
|
||||||
|
} catch (Exception e) {
|
||||||
|
return "Error_(" + e.getMessage() + ")";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public String shallStoreCacheForProxy() {
|
||||||
|
if (this.requestHeader != null) {
|
||||||
|
// -authorization cases in request
|
||||||
|
// authorization makes pages very individual, and therefore we cannot use the
|
||||||
|
// content in the cache
|
||||||
|
if (this.requestHeader.containsKey(httpHeader.AUTHORIZATION)) { return "personalized"; }
|
||||||
|
// -ranges in request and response
|
||||||
|
// we do not cache partial content
|
||||||
|
if (this.requestHeader.containsKey(httpHeader.RANGE)) { return "partial"; }
|
||||||
|
}
|
||||||
|
|
||||||
|
if (this.responseHeader != null) {
|
||||||
|
// -ranges in request and response
|
||||||
|
// we do not cache partial content
|
||||||
|
if (this.responseHeader.containsKey(httpHeader.CONTENT_RANGE)) { return "partial"; }
|
||||||
|
|
||||||
|
// -if-modified-since in request
|
||||||
|
// we do not care about if-modified-since, because this case only occurres if the
|
||||||
|
// cache file does not exist, and we need as much info as possible for the indexing
|
||||||
|
|
||||||
|
// -cookies in request
|
||||||
|
// we do not care about cookies, because that would prevent loading more pages
|
||||||
|
// from one domain once a request resulted in a client-side stored cookie
|
||||||
|
|
||||||
|
// -set-cookie in response
|
||||||
|
// we do not care about cookies in responses, because that info comes along
|
||||||
|
// any/many pages from a server and does not express the validity of the page
|
||||||
|
// in modes of life-time/expiration or individuality
|
||||||
|
|
||||||
|
// -pragma in response
|
||||||
|
// if we have a pragma non-cache, we don't cache. usually if this is wanted from
|
||||||
|
// the server, it makes sense
|
||||||
|
String cacheControl = (String) this.responseHeader.get(httpHeader.PRAGMA);
|
||||||
|
if (cacheControl != null && cacheControl.trim().toUpperCase().equals("NO-CACHE")) { return "controlled_no_cache"; }
|
||||||
|
|
||||||
|
// -expires in response
|
||||||
|
// we do not care about expires, because at the time this is called the data is
|
||||||
|
// obvious valid and that header info is used in the indexing later on
|
||||||
|
|
||||||
|
// -cache-control in response
|
||||||
|
// the cache-control has many value options.
|
||||||
|
cacheControl = (String) this.responseHeader.get(httpHeader.CACHE_CONTROL);
|
||||||
|
if (cacheControl != null) {
|
||||||
|
cacheControl = cacheControl.trim().toUpperCase();
|
||||||
|
if (cacheControl.startsWith("MAX-AGE=")) {
|
||||||
|
// we need also the load date
|
||||||
|
Date date = this.responseHeader.date();
|
||||||
|
if (date == null) return "stale_no_date_given_in_response";
|
||||||
|
try {
|
||||||
|
long ttl = 1000 * Long.parseLong(cacheControl.substring(8)); // milliseconds to live
|
||||||
|
if (serverDate.correctedUTCTime() - date.getTime() > ttl) {
|
||||||
|
//System.out.println("***not indexed because cache-control");
|
||||||
|
return "stale_expired";
|
||||||
|
}
|
||||||
|
} catch (Exception e) {
|
||||||
|
return "stale_error_" + e.getMessage() + ")";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean shallUseCacheForProxy() {
|
||||||
|
|
||||||
|
String cacheControl;
|
||||||
|
if (this.requestHeader != null) {
|
||||||
|
// -authorization cases in request
|
||||||
|
if (this.requestHeader.containsKey(httpHeader.AUTHORIZATION)) { return false; }
|
||||||
|
|
||||||
|
// -ranges in request
|
||||||
|
// we do not cache partial content
|
||||||
|
if (this.requestHeader.containsKey(httpHeader.RANGE)) { return false; }
|
||||||
|
|
||||||
|
// if the client requests a un-cached copy of the resource ...
|
||||||
|
cacheControl = (String) this.requestHeader.get(httpHeader.PRAGMA);
|
||||||
|
if (cacheControl != null && cacheControl.trim().toUpperCase().equals("NO-CACHE")) { return false; }
|
||||||
|
|
||||||
|
cacheControl = (String) this.requestHeader.get(httpHeader.CACHE_CONTROL);
|
||||||
|
if (cacheControl != null) {
|
||||||
|
cacheControl = cacheControl.trim().toUpperCase();
|
||||||
|
if (cacheControl.startsWith("NO-CACHE") || cacheControl.startsWith("MAX-AGE=0")) { return false; }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// -if-modified-since in request
|
||||||
|
// The entity has to be transferred only if it has
|
||||||
|
// been modified since the date given by the If-Modified-Since header.
|
||||||
|
if (this.requestHeader.containsKey(httpHeader.IF_MODIFIED_SINCE)) {
|
||||||
|
// checking this makes only sense if the cached response contains
|
||||||
|
// a Last-Modified field. If the field does not exist, we go the safe way
|
||||||
|
if (!this.responseHeader.containsKey(httpHeader.LAST_MODIFIED)) { return false; }
|
||||||
|
// parse date
|
||||||
|
Date d1, d2;
|
||||||
|
d2 = this.responseHeader.lastModified(); if (d2 == null) { d2 = new Date(serverDate.correctedUTCTime()); }
|
||||||
|
d1 = this.requestHeader.ifModifiedSince(); if (d1 == null) { d1 = new Date(serverDate.correctedUTCTime()); }
|
||||||
|
// finally, we shall treat the cache as stale if the modification time is after the if-.. time
|
||||||
|
if (d2.after(d1)) { return false; }
|
||||||
|
}
|
||||||
|
|
||||||
|
String mimeType = this.getMimeType();
|
||||||
|
if (!plasmaHTCache.isPicture(mimeType)) {
|
||||||
|
// -cookies in request
|
||||||
|
// unfortunately, we should reload in case of a cookie
|
||||||
|
// but we think that pictures can still be considered as fresh
|
||||||
|
// -set-cookie in cached response
|
||||||
|
// this is a similar case as for COOKIE.
|
||||||
|
if (this.requestHeader.containsKey(httpHeader.COOKIE) ||
|
||||||
|
this.responseHeader.containsKey(httpHeader.SET_COOKIE) ||
|
||||||
|
this.responseHeader.containsKey(httpHeader.SET_COOKIE2)) {
|
||||||
|
return false; // too strong
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// -pragma in cached response
|
||||||
|
// logically, we would not need to care about no-cache pragmas in cached response headers,
|
||||||
|
// because they cannot exist since they are not written to the cache.
|
||||||
|
// So this IF should always fail..
|
||||||
|
cacheControl = (String) this.responseHeader.get(httpHeader.PRAGMA);
|
||||||
|
if (cacheControl != null && cacheControl.trim().toUpperCase().equals("NO-CACHE")) { return false; }
|
||||||
|
|
||||||
|
// see for documentation also:
|
||||||
|
// http://www.web-caching.com/cacheability.html
|
||||||
|
// http://vancouver-webpages.com/CacheNow/
|
||||||
|
|
||||||
|
// look for freshnes information
|
||||||
|
// if we don't have any freshnes indication, we treat the file as stale.
|
||||||
|
// no handle for freshness control:
|
||||||
|
|
||||||
|
// -expires in cached response
|
||||||
|
// the expires value gives us a very easy hint when the cache is stale
|
||||||
|
Date expires = this.responseHeader.expires();
|
||||||
|
if (expires != null) {
|
||||||
|
// System.out.println("EXPIRES-TEST: expires=" + expires + ", NOW=" + serverDate.correctedGMTDate() + ", url=" + url);
|
||||||
|
if (expires.before(new Date(serverDate.correctedUTCTime()))) { return false; }
|
||||||
|
}
|
||||||
|
Date lastModified = this.responseHeader.lastModified();
|
||||||
|
cacheControl = (String) this.responseHeader.get(httpHeader.CACHE_CONTROL);
|
||||||
|
if (cacheControl == null && lastModified == null && expires == null) { return false; }
|
||||||
|
|
||||||
|
// -lastModified in cached response
|
||||||
|
// we can apply a TTL (Time To Live) heuristic here. We call the time delta between the last read
|
||||||
|
// of the file and the last modified date as the age of the file. If we consider the file as
|
||||||
|
// middel-aged then, the maximum TTL would be cache-creation plus age.
|
||||||
|
// This would be a TTL factor of 100% we want no more than 10% TTL, so that a 10 month old cache
|
||||||
|
// file may only be treated as fresh for one more month, not more.
|
||||||
|
Date date = this.responseHeader.date();
|
||||||
|
if (lastModified != null) {
|
||||||
|
if (date == null) { date = new Date(serverDate.correctedUTCTime()); }
|
||||||
|
long age = date.getTime() - lastModified.getTime();
|
||||||
|
if (age < 0) { return false; }
|
||||||
|
// TTL (Time-To-Live) is age/10 = (d2.getTime() - d1.getTime()) / 10
|
||||||
|
// the actual living-time is serverDate.correctedGMTDate().getTime() - d2.getTime()
|
||||||
|
// therefore the cache is stale, if serverDate.correctedGMTDate().getTime() - d2.getTime() > age/10
|
||||||
|
if (serverDate.correctedUTCTime() - date.getTime() > age / 10) { return false; }
|
||||||
|
}
|
||||||
|
|
||||||
|
// -cache-control in cached response
|
||||||
|
// the cache-control has many value options.
|
||||||
|
if (cacheControl != null) {
|
||||||
|
cacheControl = cacheControl.trim().toUpperCase();
|
||||||
|
if (cacheControl.startsWith("PRIVATE") ||
|
||||||
|
cacheControl.startsWith("NO-CACHE") ||
|
||||||
|
cacheControl.startsWith("NO-STORE")) {
|
||||||
|
// easy case
|
||||||
|
return false;
|
||||||
|
// } else if (cacheControl.startsWith("PUBLIC")) {
|
||||||
|
// // ok, do nothing
|
||||||
|
} else if (cacheControl.startsWith("MAX-AGE=")) {
|
||||||
|
// we need also the load date
|
||||||
|
if (date == null) { return false; }
|
||||||
|
try {
|
||||||
|
final long ttl = 1000 * Long.parseLong(cacheControl.substring(8)); // milliseconds to live
|
||||||
|
if (serverDate.correctedUTCTime() - date.getTime() > ttl) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
} catch (Exception e) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean validResponseStatus(String responseStatus) {
|
||||||
|
return responseStatus.startsWith("200") ||
|
||||||
|
responseStatus.startsWith("203");
|
||||||
|
}
|
||||||
|
|
||||||
|
public Date ifModifiedSince() {
|
||||||
|
return (this.requestHeader == null) ? null : this.requestHeader.ifModifiedSince();
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean requestWithCookie() {
|
||||||
|
return (this.requestHeader == null) ? false : this.requestHeader.containsKey(httpHeader.COOKIE);
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean requestProhibitsIndexing() {
|
||||||
|
return (this.requestHeader == null)
|
||||||
|
? false
|
||||||
|
: this.requestHeader.containsKey(httpHeader.X_YACY_INDEX_CONTROL) &&
|
||||||
|
((String)this.requestHeader.get(httpHeader.X_YACY_INDEX_CONTROL)).toUpperCase().equals("NO-INDEX");
|
||||||
|
}
|
||||||
|
|
||||||
|
public httpHeader getRequestHeader() {
|
||||||
|
return this.requestHeader;
|
||||||
|
}
|
||||||
|
|
||||||
|
public httpHeader getResponseHeader() {
|
||||||
|
return this.responseHeader;
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in new issue