You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
yacy_search_server/source/de/anomic/http/httpRemoteProxyConfig.java

318 lines
10 KiB

//httpRemoteProxyConfig.java
//-----------------------
//part of the AnomicHTTPD caching proxy
//(C) by Michael Peter Christen; mc@yacy.net
//first published on http://www.anomic.de
//Frankfurt, Germany, 2004
//
//this file was contributed by Martin Thelian
//$LastChangedDate$
//$LastChangedBy$
//$LastChangedRevision$
//
//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
//You must compile this file with
//javac -classpath .:../Classes Settings_p.java
//if the shell's current path is HTROOT
package de.anomic.http;
import java.util.Arrays;
import java.util.HashSet;
import java.util.Set;
import de.anomic.plasma.plasmaSwitchboard;
public final class httpRemoteProxyConfig {
/*
* Remote Proxy configuration
*/
private boolean remoteProxyUse;
private boolean remoteProxyUse4Yacy;
private boolean remoteProxyUse4SSL;
private String remoteProxyHost;
private int remoteProxyPort;
private String remoteProxyUser;
private String remoteProxyPwd;
private String[] remoteProxyNoProxyPatterns = null;
private final Set<String> remoteProxyAllowProxySet = new HashSet<String>();
private final Set<String> remoteProxyDisallowProxySet = new HashSet<String>();
/**
* *The* remote Proxy configuration
*/
private static httpRemoteProxyConfig remoteProxyConfig = null;
/**
* @param remoteProxyConfig the remoteProxyConfig to set
*/
public static synchronized void setRemoteProxyConfig(final httpRemoteProxyConfig remoteProxyConfig) {
httpRemoteProxyConfig.remoteProxyConfig = remoteProxyConfig;
}
/**
* @return the remoteProxyConfig
*/
public static synchronized httpRemoteProxyConfig getRemoteProxyConfig() {
return remoteProxyConfig;
}
/**
* creates a new remoteProxyConfig for given proxy
*
* @param proxyHostName
* @param proxyHostPort
* @return
*/
public static httpRemoteProxyConfig createRemoteProxyConfig(
final String proxyHostName,
final int proxyHostPort
) {
final httpRemoteProxyConfig newConfig = new httpRemoteProxyConfig();
newConfig.remoteProxyUse = true;
newConfig.remoteProxyUse4SSL = true;
newConfig.remoteProxyUse4Yacy = true;
newConfig.remoteProxyPort = proxyHostPort;
newConfig.remoteProxyHost = proxyHostName;
if ((newConfig.remoteProxyHost == null) || (newConfig.remoteProxyHost.length() == 0)) {
newConfig.remoteProxyUse = false;
}
return newConfig;
}
/**
* initializes the global remoteProxyConfig with values from plasmaSwitchboard (config)
*
* @param sb
* @return
*/
public static void init(final plasmaSwitchboard sb) {
// reading the proxy host name
final String proxyHostName = sb.getConfig("remoteProxyHost", "").trim();
// reading the proxy host port
int proxyHostPort;
try {
proxyHostPort = Integer.parseInt(sb.getConfig("remoteProxyPort", "3128"));
} catch (final NumberFormatException e) {
proxyHostPort = 3128;
}
// create new config
final httpRemoteProxyConfig newConfig = createRemoteProxyConfig(proxyHostName, proxyHostPort);
// determining if remote proxy usage is enabled
newConfig.remoteProxyUse = newConfig.remoteProxyUse && sb.getConfigBool("remoteProxyUse", false);
// determining if remote proxy should be used for yacy -> yacy communication
newConfig.remoteProxyUse4Yacy = sb.getConfig("remoteProxyUse4Yacy", "true").equalsIgnoreCase("true");
// determining if remote proxy should be used for ssl connections
newConfig.remoteProxyUse4SSL = sb.getConfig("remoteProxyUse4SSL", "true").equalsIgnoreCase("true");
newConfig.remoteProxyUser = sb.getConfig("remoteProxyUser", "").trim();
newConfig.remoteProxyPwd = sb.getConfig("remoteProxyPwd", "").trim();
// determining addresses for which the remote proxy should not be used
final String remoteProxyNoProxy = sb.getConfig("remoteProxyNoProxy","").trim();
newConfig.remoteProxyNoProxyPatterns = remoteProxyNoProxy.split(",");
// trim split entries
int i = 0;
for(final String pattern: newConfig.remoteProxyNoProxyPatterns) {
newConfig.remoteProxyNoProxyPatterns[i] = pattern.trim();
i++;
}
setRemoteProxyConfig(newConfig);
sb.getLog().logConfig("Remote proxy configuration:\n" + getRemoteProxyConfig().toString());
}
public boolean useProxy() {
return this.remoteProxyUse;
}
public boolean useProxy4Yacy() {
return this.remoteProxyUse4Yacy;
}
public boolean useProxy4SSL() {
return this.remoteProxyUse4SSL;
}
public String getProxyHost() {
return this.remoteProxyHost;
}
public int getProxyPort() {
return this.remoteProxyPort;
}
public String getProxyUser() {
return this.remoteProxyUser;
}
public String getProxyPwd() {
return this.remoteProxyPwd;
}
/**
* checks if server matches the noProxyPattern and caches result as (Dis-)AllowedHost
*
* @param server
* @return
*/
public boolean matchesProxyNoProxyPatterns(final String server) {
boolean noProxy = false;
for (final String pattern :remoteProxyNoProxyPatterns) {
if (server.matches(pattern)) {
noProxy = true;
break;
}
}
// set either remoteProxyAllowProxySet or remoteProxyDisallowProxySet accordingly
if(noProxy) {
addDisallowedHost(server);
} else {
addAllowedHost(server);
}
return noProxy;
}
public String toString() {
final StringBuilder toStrBuf = new StringBuilder(50);
toStrBuf
.append("Status: ").append(this.remoteProxyUse?"ON":"OFF").append(" | ")
.append("Host: ");
if ((this.remoteProxyUser != null) && (this.remoteProxyUser.length() > 0)) {
toStrBuf.append(this.remoteProxyUser)
.append("@");
}
toStrBuf
.append((this.remoteProxyHost==null)?"unknown":this.remoteProxyHost).append(":").append(this.remoteProxyPort).append(" | ")
.append("Usage: HTTP");
if (this.remoteProxyUse4Yacy) toStrBuf.append(" YACY");
if (this.remoteProxyUse4SSL) toStrBuf.append(" SSL");
toStrBuf.append(" | ")
.append("No Proxy for: ")
.append(Arrays.toString(this.remoteProxyNoProxyPatterns));
return toStrBuf.toString();
}
/**
* @param host
* @return
*/
public boolean isHostAllowedToUseProxy(final String host) {
return remoteProxyAllowProxySet.contains(host);
}
/**
* @param host
* @return
*/
public boolean isHostNotAllowedToUseProxy(final String host) {
return remoteProxyDisallowProxySet.contains(host);
}
/**
* @param host
*/
public void addAllowedHost(final String host) {
remoteProxyAllowProxySet.add(host);
}
/**
* @param host
*/
public void addDisallowedHost(final String host) {
remoteProxyDisallowProxySet.add(host);
}
/**
* checks if proxy-config is allowed and not denied
*
* @param server
* @param remProxyConfig
* @return proxy which should be used or null if no proxy should be used for server
*/
public boolean useForHost(final String server) {
// possible remote proxy
// check no-proxy rule
boolean useProxy;
if (useProxy()) {
useProxy = true;
if (!isHostAllowedToUseProxy(server)) {
if (isHostNotAllowedToUseProxy(server)) {
useProxy = false;
} else {
// analyse remoteProxyNoProxy;
if(matchesProxyNoProxyPatterns(server)) {
useProxy = false;
}
}
}
} else {
// not enabled
useProxy = false;
}
return useProxy;
}
/**
* checks if proxy-config exists for server (is allowed and not denied)
*
* @param server
* @return proxy which should be used or null if no proxy should be used
*/
public static httpRemoteProxyConfig getProxyConfigForHost(final String server) {
final httpRemoteProxyConfig proxyConfig = getRemoteProxyConfig();
if(proxyConfig == null || !proxyConfig.useForHost(server)) {
return null;
}
return proxyConfig;
}
/**
* gets proxy config for full adress (without protocol) ie. "localhost:8080/path/file.html"
*
* @param address full address with host and optionally port or path <host>:<port>/<path>
* @return null if no proxy should be used
*/
public static httpRemoteProxyConfig getProxyConfigForURI(final String address) {
// host goes to port
int p = address.indexOf(":");
if (p < 0) {
// no port, so host goes to path
p = address.indexOf("/");
}
// host should have minimum 1 character ;)
if(p > 0) {
final String server = address.substring(0, p);
return getProxyConfigForHost(server);
}
// check if full address is in allow/deny-list
return getProxyConfigForHost(address);
}
}