@ -35,6 +35,7 @@ import net.yacy.search.Switchboard;
import org.bitlet.weupnp.GatewayDevice ;
import org.bitlet.weupnp.GatewayDevice ;
import org.bitlet.weupnp.GatewayDiscover ;
import org.bitlet.weupnp.GatewayDiscover ;
import org.bitlet.weupnp.PortMappingEntry ;
import org.xml.sax.SAXException ;
import org.xml.sax.SAXException ;
public class UPnP {
public class UPnP {
@ -42,7 +43,7 @@ public class UPnP {
private static final ConcurrentLog LOG = new ConcurrentLog ( "UPNP" ) ;
private static final ConcurrentLog LOG = new ConcurrentLog ( "UPNP" ) ;
private static final Switchboard SB = Switchboard . getSwitchboard ( ) ;
private static final Switchboard SB = Switchboard . getSwitchboard ( ) ;
private static Map< InetAddress , GatewayDevice > GATEWAY_DEVICES = null ;
private static GatewayDevice gatewayDevice ;
private static final Map < UPnPMappingType , UPnPMapping > MAPPINGS = new EnumMap < > (
private static final Map < UPnPMappingType , UPnPMapping > MAPPINGS = new EnumMap < > (
UPnPMappingType . class ) ;
UPnPMappingType . class ) ;
@ -53,20 +54,23 @@ public class UPnP {
"server.https" , "TCP" , "YaCy HTTPS" ) ) ;
"server.https" , "TCP" , "YaCy HTTPS" ) ) ;
}
}
private static final int MIN_CANDIDATE_PORT = 49152 ;
private static final int MAX_CANDIDATE_PORT = 65535 ;
private static boolean init ( ) {
private static boolean init ( ) {
boolean init = true ;
boolean init = true ;
try {
try {
if ( GATEWAY_DEVICES = = null ) {
if ( gatewayDevice = = null | | ! gatewayDevice . isConnected ( ) ) {
GATEWAY_DEVICES = new GatewayDiscover ( ) . discover ( ) ;
final GatewayDiscover discover = new GatewayDiscover ( ) ;
discover . discover ( ) ;
gatewayDevice = discover . getValidGateway ( ) ;
}
}
} catch ( IOException | SAXException | ParserConfigurationException e ) {
} catch ( IOException | SAXException | ParserConfigurationException e ) {
init = false ;
init = false ;
}
}
if ( GATEWAY_DEVICES ! = null ) {
if ( gatewayDevice ! = null ) {
for ( final GatewayDevice gatewayDevice : GATEWAY_DEVICES . values ( ) ) {
LOG . info ( "found device: " + gatewayDevice . getFriendlyName ( ) ) ;
LOG . info ( "found device: " + gatewayDevice . getFriendlyName ( ) ) ;
}
} else {
} else {
LOG . info ( "no device found" ) ;
LOG . info ( "no device found" ) ;
init = false ;
init = false ;
@ -91,6 +95,8 @@ public class UPnP {
addPortMapping ( entry . getKey ( ) , mapping ,
addPortMapping ( entry . getKey ( ) , mapping ,
SB . getConfigInt ( mapping . getConfigPortKey ( ) , 0 ) ) ;
SB . getConfigInt ( mapping . getConfigPortKey ( ) , 0 ) ) ;
}
}
SB . setConnectedViaUpnp ( true ) ;
}
}
/ * *
/ * *
@ -102,6 +108,8 @@ public class UPnP {
return ;
return ;
}
}
SB . setConnectedViaUpnp ( false ) ;
UPnPMapping mapping ;
UPnPMapping mapping ;
for ( final Entry < UPnPMappingType , UPnPMapping > entry : MAPPINGS
for ( final Entry < UPnPMappingType , UPnPMapping > entry : MAPPINGS
. entrySet ( ) ) {
. entrySet ( ) ) {
@ -123,7 +131,7 @@ public class UPnP {
* @param port
* @param port
* port number to map
* port number to map
* /
* /
p ublic static void addPortMapping ( final UPnPMappingType type ,
p rivate static void addPortMapping ( final UPnPMappingType type ,
final UPnPMapping mapping , final int port ) {
final UPnPMapping mapping , final int port ) {
if ( port < 1 ) {
if ( port < 1 ) {
@ -137,32 +145,49 @@ public class UPnP {
if ( ( mapping . isConfigEnabledKeyEmpty ( ) | | SB . getConfigBool (
if ( ( mapping . isConfigEnabledKeyEmpty ( ) | | SB . getConfigBool (
mapping . getConfigEnabledKey ( ) , false ) )
mapping . getConfigEnabledKey ( ) , false ) )
& & mapping . getPort ( ) = = 0
& & mapping . getPort ( ) = = 0
& & ( ( GATEWAY_DEVICES ! = null ) | | init ( ) ) ) {
& & ( ( gatewayDevice ! = null ) | | init ( ) ) ) {
String localHostIP ;
String localHostIP ;
boolean mapped ;
boolean mapped ;
String msg ;
String msg ;
for ( final GatewayDevice gatewayDevice : GATEWAY_DEVICES . values ( ) ) {
try {
try {
localHostIP = toString ( gatewayDevice . getLocalAddress ( ) ) ;
localHostIP = toString ( gatewayDevice . getLocalAddress ( ) ) ;
int portCandidate = port ;
while ( isInUse ( portCandidate ) & & portCandidate > 0 ) {
LOG . info ( portCandidate + " in use, trying different port." ) ;
portCandidate = getNewPortCandidate ( portCandidate ) ;
}
if ( portCandidate > 0 ) {
mapped = gatewayDevice . addPortMapping ( port , port ,
mapped = gatewayDevice . addPortMapping ( port Candidate , port ,
localHostIP , mapping . getProtocol ( ) ,
localHostIP , mapping . getProtocol ( ) ,
mapping . getDescription ( ) ) ;
mapping . getDescription ( ) ) ;
msg = "port " + port + " on device "
msg = "mapped port " + port + " to port " + portCandidate
+ gatewayDevice . getFriendlyName ( ) ;
+ " on device " + gatewayDevice . getFriendlyName ( )
+ ", external IP is "
if ( mapped ) {
+ gatewayDevice . getExternalIPAddress ( ) ;
LOG . info ( "mapped " + msg ) ;
mapping . setPort ( port ) ;
} else {
} else {
LOG . warn ( "could not map " + msg ) ;
mapped = false ;
}
} catch ( IOException | SAXException e ) {
msg = "no free port found" ;
LOG . severe ( "mapping error: " + e . getMessage ( ) ) ;
}
}
if ( mapped ) {
LOG . info ( "mapped " + msg ) ;
mapping . setPort ( portCandidate ) ;
SB . setUpnpPorts ( mapping . getConfigPortKey ( ) , portCandidate ) ;
} else {
LOG . warn ( "could not map " + msg ) ;
}
} catch ( IOException | SAXException e ) {
LOG . severe ( "mapping error: " + e . getMessage ( ) ) ;
}
}
}
}
}
}
@ -173,33 +198,31 @@ public class UPnP {
* @param mapping
* @param mapping
* to delete
* to delete
* /
* /
p ublic static void deletePortMapping ( final UPnPMapping mapping ) {
p rivate static void deletePortMapping ( final UPnPMapping mapping ) {
if ( mapping . getPort ( ) > 0 & & GATEWAY_DEVICES ! = null ) {
if ( mapping . getPort ( ) > 0 & & gatewayDevice ! = null ) {
boolean unmapped ;
boolean unmapped ;
String msg ;
String msg ;
for ( final GatewayDevice gatewayDevice : GATEWAY_DEVICES . values ( ) ) {
try {
unmapped = gatewayDevice . deletePortMapping (
mapping . getPort ( ) , mapping . getProtocol ( ) ) ;
msg = "port " + mapping . getPort ( ) + " on device "
try {
+ gatewayDevice . getFriendlyName ( ) ;
unmapped = gatewayDevice . deletePortMapping ( mapping . getPort ( ) ,
mapping . getProtocol ( ) ) ;
if ( unmapped ) {
msg = "port " + mapping . getPort ( ) + " on device "
LOG . info ( "unmapped " + msg ) ;
+ gatewayDevice . getFriendlyName ( ) ;
} else {
LOG . warn ( "could not unmap " + msg ) ;
}
} catch ( SAXException | IOException e ) {
if ( unmapped ) {
LOG . severe ( "unmapping error: " + e . getMessage ( ) ) ;
LOG . info ( "unmapped " + msg ) ;
} else {
LOG . warn ( "could not unmap " + msg ) ;
}
}
}
mapping . setPort ( 0 ) ; // reset mapped port
} catch ( SAXException | IOException e ) {
LOG . severe ( "unmapping error: " + e . getMessage ( ) ) ;
}
}
}
mapping . setPort ( 0 ) ; // reset mapped port
}
}
/ * *
/ * *
@ -219,6 +242,30 @@ public class UPnP {
return MAPPINGS . get ( type ) . getPort ( ) ;
return MAPPINGS . get ( type ) . getPort ( ) ;
}
}
public static int getNewPortCandidate ( final int oldCandidate ) {
int newPortCandidate = Math . min (
Math . max ( MIN_CANDIDATE_PORT , oldCandidate + 1 ) ,
MAX_CANDIDATE_PORT ) ;
if ( newPortCandidate = = MAX_CANDIDATE_PORT ) {
newPortCandidate = - 1 ;
}
return newPortCandidate ;
}
private static boolean isInUse ( final int port ) {
try {
return gatewayDevice ! = null
& & gatewayDevice . getSpecificPortMappingEntry ( port , "TCP" ,
new PortMappingEntry ( ) ) ;
} catch ( IOException | SAXException e ) {
return false ;
}
}
private static String toString ( final InetAddress inetAddress ) {
private static String toString ( final InetAddress inetAddress ) {
final String localHostIP ;
final String localHostIP ;