mirror of
https://github.com/SquidDev-CC/CC-Tweaked
synced 2025-01-07 07:50:27 +00:00
Merge pull request #313 from SquidDev-CC/feature/http-blacklist
HTTP blacklist and IP address support
This commit is contained in:
commit
123a0158af
@ -18,6 +18,7 @@ import dan200.computercraft.api.permissions.ITurtlePermissionProvider;
|
|||||||
import dan200.computercraft.api.pocket.IPocketUpgrade;
|
import dan200.computercraft.api.pocket.IPocketUpgrade;
|
||||||
import dan200.computercraft.api.redstone.IBundledRedstoneProvider;
|
import dan200.computercraft.api.redstone.IBundledRedstoneProvider;
|
||||||
import dan200.computercraft.api.turtle.ITurtleUpgrade;
|
import dan200.computercraft.api.turtle.ITurtleUpgrade;
|
||||||
|
import dan200.computercraft.core.apis.AddressPredicate;
|
||||||
import dan200.computercraft.core.filesystem.ComboMount;
|
import dan200.computercraft.core.filesystem.ComboMount;
|
||||||
import dan200.computercraft.core.filesystem.FileMount;
|
import dan200.computercraft.core.filesystem.FileMount;
|
||||||
import dan200.computercraft.core.filesystem.JarMount;
|
import dan200.computercraft.core.filesystem.JarMount;
|
||||||
@ -60,6 +61,7 @@ import net.minecraft.util.EnumHand;
|
|||||||
import net.minecraft.util.SoundEvent;
|
import net.minecraft.util.SoundEvent;
|
||||||
import net.minecraft.util.math.BlockPos;
|
import net.minecraft.util.math.BlockPos;
|
||||||
import net.minecraft.world.World;
|
import net.minecraft.world.World;
|
||||||
|
import net.minecraftforge.common.config.ConfigCategory;
|
||||||
import net.minecraftforge.common.config.Configuration;
|
import net.minecraftforge.common.config.Configuration;
|
||||||
import net.minecraftforge.common.config.Property;
|
import net.minecraftforge.common.config.Property;
|
||||||
import net.minecraftforge.fml.common.FMLCommonHandler;
|
import net.minecraftforge.fml.common.FMLCommonHandler;
|
||||||
@ -105,8 +107,18 @@ public class ComputerCraft
|
|||||||
public static final int pocketComputerGUIID = 106;
|
public static final int pocketComputerGUIID = 106;
|
||||||
|
|
||||||
// Configuration options
|
// Configuration options
|
||||||
|
private static final String[] DEFAULT_HTTP_WHITELIST = new String[] { "*" };
|
||||||
|
private static final String[] DEFAULT_HTTP_BLACKLIST = new String[] {
|
||||||
|
"127.0.0.0/8",
|
||||||
|
"10.0.0.0/8",
|
||||||
|
"172.16.0.0/12",
|
||||||
|
"192.168.0.0/16",
|
||||||
|
"fd00::/8",
|
||||||
|
};
|
||||||
|
|
||||||
public static boolean http_enable = true;
|
public static boolean http_enable = true;
|
||||||
public static String http_whitelist = "*";
|
public static AddressPredicate http_whitelist = new AddressPredicate( DEFAULT_HTTP_WHITELIST );
|
||||||
|
public static AddressPredicate http_blacklist = new AddressPredicate( DEFAULT_HTTP_BLACKLIST );
|
||||||
public static boolean disable_lua51_features = false;
|
public static boolean disable_lua51_features = false;
|
||||||
public static String default_computer_settings = "";
|
public static String default_computer_settings = "";
|
||||||
public static boolean logPeripheralErrors = false;
|
public static boolean logPeripheralErrors = false;
|
||||||
@ -185,6 +197,7 @@ public class ComputerCraft
|
|||||||
|
|
||||||
public static Property http_enable;
|
public static Property http_enable;
|
||||||
public static Property http_whitelist;
|
public static Property http_whitelist;
|
||||||
|
public static Property http_blacklist;
|
||||||
public static Property disable_lua51_features;
|
public static Property disable_lua51_features;
|
||||||
public static Property default_computer_settings;
|
public static Property default_computer_settings;
|
||||||
public static Property logPeripheralErrors;
|
public static Property logPeripheralErrors;
|
||||||
@ -252,10 +265,28 @@ public class ComputerCraft
|
|||||||
Config.config.load();
|
Config.config.load();
|
||||||
|
|
||||||
Config.http_enable = Config.config.get( Configuration.CATEGORY_GENERAL, "http_enable", http_enable );
|
Config.http_enable = Config.config.get( Configuration.CATEGORY_GENERAL, "http_enable", http_enable );
|
||||||
Config.http_enable.setComment( "Enable the \"http\" API on Computers (see \"http_whitelist\" for more fine grained control than this)" );
|
Config.http_enable.setComment( "Enable the \"http\" API on Computers (see \"http_whitelist\" and \"http_blacklist\" for more fine grained control than this)" );
|
||||||
|
|
||||||
Config.http_whitelist = Config.config.get( Configuration.CATEGORY_GENERAL, "http_whitelist", http_whitelist );
|
{
|
||||||
Config.http_whitelist.setComment( "A semicolon limited list of wildcards for domains that can be accessed through the \"http\" API on Computers. Set this to \"*\" to access to the entire internet. Example: \"*.pastebin.com;*.github.com;*.computercraft.info\" will restrict access to just those 3 domains." );
|
ConfigCategory category = Config.config.getCategory( Configuration.CATEGORY_GENERAL );
|
||||||
|
Property currentProperty = category.get( "http_whitelist" );
|
||||||
|
if( currentProperty != null && !currentProperty.isList() ) category.remove( "http_whitelist" );
|
||||||
|
|
||||||
|
Config.http_whitelist = Config.config.get( Configuration.CATEGORY_GENERAL, "http_whitelist", DEFAULT_HTTP_WHITELIST );
|
||||||
|
|
||||||
|
if( currentProperty != null && !currentProperty.isList() )
|
||||||
|
{
|
||||||
|
Config.http_whitelist.setValues( currentProperty.getString().split( ";" ) );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Config.http_whitelist.setComment( "A list of wildcards for domains or IP ranges that can be accessed through the \"http\" API on Computers.\n" +
|
||||||
|
"Set this to \"*\" to access to the entire internet. Example: \"*.pastebin.com\" will restrict access to just subdomains of pastebin.com.\n" +
|
||||||
|
"You can use domain names (\"pastebin.com\"), wilcards (\"*.pastebin.com\") or CIDR notation (\"127.0.0.0/8\")." );
|
||||||
|
|
||||||
|
Config.http_blacklist = Config.config.get( Configuration.CATEGORY_GENERAL, "http_blacklist", DEFAULT_HTTP_BLACKLIST );
|
||||||
|
Config.http_blacklist.setComment( "A list of wildcards for domains or IP ranges that cannot be accessed through the \"http\" API on Computers.\n" +
|
||||||
|
"If this is empty then all whitelisted domains will be accessible. Example: \"*.github.com\" will block access to all subdomains of github.com.\n" +
|
||||||
|
"You can use domain names (\"pastebin.com\"), wilcards (\"*.pastebin.com\") or CIDR notation (\"127.0.0.0/8\")." );
|
||||||
|
|
||||||
Config.disable_lua51_features = Config.config.get( Configuration.CATEGORY_GENERAL, "disable_lua51_features", disable_lua51_features );
|
Config.disable_lua51_features = Config.config.get( Configuration.CATEGORY_GENERAL, "disable_lua51_features", disable_lua51_features );
|
||||||
Config.disable_lua51_features.setComment( "Set this to true to disable Lua 5.1 functions that will be removed in a future update. Useful for ensuring forward compatibility of your programs now." );
|
Config.disable_lua51_features.setComment( "Set this to true to disable Lua 5.1 functions that will be removed in a future update. Useful for ensuring forward compatibility of your programs now." );
|
||||||
@ -327,7 +358,8 @@ public class ComputerCraft
|
|||||||
public static void syncConfig() {
|
public static void syncConfig() {
|
||||||
|
|
||||||
http_enable = Config.http_enable.getBoolean();
|
http_enable = Config.http_enable.getBoolean();
|
||||||
http_whitelist = Config.http_whitelist.getString();
|
http_whitelist = new AddressPredicate( Config.http_whitelist.getStringList() );
|
||||||
|
http_blacklist = new AddressPredicate( Config.http_blacklist.getStringList() );
|
||||||
disable_lua51_features = Config.disable_lua51_features.getBoolean();
|
disable_lua51_features = Config.disable_lua51_features.getBoolean();
|
||||||
default_computer_settings = Config.default_computer_settings.getString();
|
default_computer_settings = Config.default_computer_settings.getString();
|
||||||
|
|
||||||
|
@ -0,0 +1,168 @@
|
|||||||
|
package dan200.computercraft.core.apis;
|
||||||
|
|
||||||
|
import com.google.common.net.InetAddresses;
|
||||||
|
import dan200.computercraft.ComputerCraft;
|
||||||
|
|
||||||
|
import java.net.Inet6Address;
|
||||||
|
import java.net.InetAddress;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.regex.Pattern;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Used to determine whether a domain or IP address matches a series of patterns.
|
||||||
|
*/
|
||||||
|
public class AddressPredicate
|
||||||
|
{
|
||||||
|
private static class HostRange
|
||||||
|
{
|
||||||
|
private final byte[] min;
|
||||||
|
private final byte[] max;
|
||||||
|
|
||||||
|
private HostRange( byte[] min, byte[] max )
|
||||||
|
{
|
||||||
|
this.min = min;
|
||||||
|
this.max = max;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean contains( InetAddress address )
|
||||||
|
{
|
||||||
|
byte[] entry = address.getAddress();
|
||||||
|
if( entry.length != min.length ) return false;
|
||||||
|
|
||||||
|
for( int i = 0; i < entry.length; i++ )
|
||||||
|
{
|
||||||
|
int value = 0xFF & entry[ i ];
|
||||||
|
if( value < (0xFF & min[ i ]) || value > (0xFF & max[ i ]) ) return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private final List<Pattern> wildcards;
|
||||||
|
private final List<HostRange> ranges;
|
||||||
|
|
||||||
|
public AddressPredicate( String... filters )
|
||||||
|
{
|
||||||
|
List<Pattern> wildcards = this.wildcards = new ArrayList<Pattern>();
|
||||||
|
List<HostRange> ranges = this.ranges = new ArrayList<HostRange>();
|
||||||
|
|
||||||
|
for( String filter : filters )
|
||||||
|
{
|
||||||
|
int cidr = filter.indexOf( '/' );
|
||||||
|
if( cidr >= 0 )
|
||||||
|
{
|
||||||
|
String addressStr = filter.substring( 0, cidr );
|
||||||
|
String prefixSizeStr = filter.substring( cidr + 1 );
|
||||||
|
|
||||||
|
int prefixSize;
|
||||||
|
try
|
||||||
|
{
|
||||||
|
prefixSize = Integer.parseInt( prefixSizeStr );
|
||||||
|
}
|
||||||
|
catch( NumberFormatException e )
|
||||||
|
{
|
||||||
|
ComputerCraft.log.warn( "Cannot parse CIDR size from {} ({})", filter, prefixSizeStr );
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
InetAddress address;
|
||||||
|
try
|
||||||
|
{
|
||||||
|
address = InetAddresses.forString( addressStr );
|
||||||
|
}
|
||||||
|
catch( IllegalArgumentException e )
|
||||||
|
{
|
||||||
|
ComputerCraft.log.warn( "Cannot parse IP address from {} ({})", filter, addressStr );
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Mask the bytes of the IP address.
|
||||||
|
byte[] minBytes = address.getAddress(), maxBytes = address.getAddress();
|
||||||
|
int size = prefixSize;
|
||||||
|
for( int i = 0; i < minBytes.length; i++ )
|
||||||
|
{
|
||||||
|
if( size <= 0 )
|
||||||
|
{
|
||||||
|
minBytes[ i ] &= 0;
|
||||||
|
maxBytes[ i ] |= 0xFF;
|
||||||
|
}
|
||||||
|
else if( size < 8 )
|
||||||
|
{
|
||||||
|
minBytes[ i ] &= 0xFF << (8 - size);
|
||||||
|
maxBytes[ i ] |= ~(0xFF << (8 - size));
|
||||||
|
}
|
||||||
|
|
||||||
|
size -= 8;
|
||||||
|
}
|
||||||
|
|
||||||
|
ranges.add( new HostRange( minBytes, maxBytes ) );
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
wildcards.add( Pattern.compile( "^\\Q" + filter.replaceAll( "\\*", "\\\\E.*\\\\Q" ) + "\\E$" ) );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Determine whether a host name matches a series of patterns.
|
||||||
|
*
|
||||||
|
* This is intended to allow early exiting, before one has to look up the IP address. You should use
|
||||||
|
* {@link #matches(InetAddress)} instead of/in addition to this one.
|
||||||
|
*
|
||||||
|
* @param domain The domain to match.
|
||||||
|
* @return Whether the patterns were matched.
|
||||||
|
*/
|
||||||
|
public boolean matches( String domain )
|
||||||
|
{
|
||||||
|
for( Pattern domainPattern : wildcards )
|
||||||
|
{
|
||||||
|
if( domainPattern.matcher( domain ).matches() ) return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
private boolean matchesAddress( InetAddress address )
|
||||||
|
{
|
||||||
|
String addressString = address.getHostAddress();
|
||||||
|
for( Pattern domainPattern : wildcards )
|
||||||
|
{
|
||||||
|
if( domainPattern.matcher( addressString ).matches() ) return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
for( HostRange range : ranges )
|
||||||
|
{
|
||||||
|
if( range.contains( address ) ) return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Determine whether the given address matches a series of patterns
|
||||||
|
*
|
||||||
|
* @param address The address to check.
|
||||||
|
* @return Whether it matches any of these patterns.
|
||||||
|
*/
|
||||||
|
public boolean matches( InetAddress address )
|
||||||
|
{
|
||||||
|
// Match the host name
|
||||||
|
String host = address.getHostName();
|
||||||
|
if( host != null && matches( host ) ) return true;
|
||||||
|
|
||||||
|
// Match the normal address
|
||||||
|
if( matchesAddress( address ) ) return true;
|
||||||
|
|
||||||
|
// If we're an IPv4 address in disguise then let's check that.
|
||||||
|
if( address instanceof Inet6Address && InetAddresses.is6to4Address( (Inet6Address) address )
|
||||||
|
&& matchesAddress( InetAddresses.get6to4IPv4Address( (Inet6Address) address ) ) )
|
||||||
|
{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
@ -7,13 +7,13 @@
|
|||||||
package dan200.computercraft.core.apis;
|
package dan200.computercraft.core.apis;
|
||||||
|
|
||||||
import dan200.computercraft.api.lua.ILuaContext;
|
import dan200.computercraft.api.lua.ILuaContext;
|
||||||
import dan200.computercraft.api.lua.ILuaObject;
|
|
||||||
import dan200.computercraft.api.lua.LuaException;
|
import dan200.computercraft.api.lua.LuaException;
|
||||||
import dan200.computercraft.core.apis.handles.BinaryInputHandle;
|
import dan200.computercraft.core.apis.http.HTTPCheck;
|
||||||
import dan200.computercraft.core.apis.handles.EncodedInputHandle;
|
import dan200.computercraft.core.apis.http.HTTPRequest;
|
||||||
|
import dan200.computercraft.core.apis.http.HTTPTask;
|
||||||
|
|
||||||
import javax.annotation.Nonnull;
|
import javax.annotation.Nonnull;
|
||||||
import java.io.InputStream;
|
import java.net.URL;
|
||||||
import java.util.*;
|
import java.util.*;
|
||||||
|
|
||||||
import static dan200.computercraft.core.apis.ArgumentHelper.*;
|
import static dan200.computercraft.core.apis.ArgumentHelper.*;
|
||||||
@ -21,12 +21,12 @@ import static dan200.computercraft.core.apis.ArgumentHelper.*;
|
|||||||
public class HTTPAPI implements ILuaAPI
|
public class HTTPAPI implements ILuaAPI
|
||||||
{
|
{
|
||||||
private final IAPIEnvironment m_apiEnvironment;
|
private final IAPIEnvironment m_apiEnvironment;
|
||||||
private final List<HTTPRequest> m_httpRequests;
|
private final List<HTTPTask> m_httpTasks;
|
||||||
|
|
||||||
public HTTPAPI( IAPIEnvironment environment )
|
public HTTPAPI( IAPIEnvironment environment )
|
||||||
{
|
{
|
||||||
m_apiEnvironment = environment;
|
m_apiEnvironment = environment;
|
||||||
m_httpRequests = new ArrayList<HTTPRequest>();
|
m_httpTasks = new ArrayList<HTTPTask>();
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
@ -46,95 +46,31 @@ public class HTTPAPI implements ILuaAPI
|
|||||||
public void advance( double _dt )
|
public void advance( double _dt )
|
||||||
{
|
{
|
||||||
// Wait for all of our http requests
|
// Wait for all of our http requests
|
||||||
synchronized( m_httpRequests )
|
synchronized( m_httpTasks )
|
||||||
{
|
{
|
||||||
Iterator<HTTPRequest> it = m_httpRequests.iterator();
|
Iterator<HTTPTask> it = m_httpTasks.iterator();
|
||||||
while( it.hasNext() ) {
|
while( it.hasNext() )
|
||||||
final HTTPRequest h = it.next();
|
{
|
||||||
if( h.isComplete() ) {
|
final HTTPTask h = it.next();
|
||||||
final String url = h.getURL();
|
if( h.isFinished() )
|
||||||
if( h.wasSuccessful() ) {
|
{
|
||||||
// Queue the "http_success" event
|
h.whenFinished( m_apiEnvironment );
|
||||||
InputStream contents = h.getContents();
|
|
||||||
Object result = wrapStream(
|
|
||||||
h.isBinary() ? new BinaryInputHandle( contents ) : new EncodedInputHandle( contents, h.getEncoding() ),
|
|
||||||
h.getResponseCode(), h.getResponseHeaders()
|
|
||||||
);
|
|
||||||
m_apiEnvironment.queueEvent( "http_success", new Object[] { url, result } );
|
|
||||||
} else {
|
|
||||||
// Queue the "http_failure" event
|
|
||||||
InputStream contents = h.getContents();
|
|
||||||
Object result = null;
|
|
||||||
if( contents != null ) {
|
|
||||||
result = wrapStream(
|
|
||||||
h.isBinary() ? new BinaryInputHandle( contents ) : new EncodedInputHandle( contents, h.getEncoding() ),
|
|
||||||
h.getResponseCode(), h.getResponseHeaders()
|
|
||||||
);
|
|
||||||
}
|
|
||||||
m_apiEnvironment.queueEvent( "http_failure", new Object[]{ url, "Could not connect", result } );
|
|
||||||
}
|
|
||||||
it.remove();
|
it.remove();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private static ILuaObject wrapStream( final ILuaObject reader, final int responseCode, final Map<String, String> responseHeaders )
|
|
||||||
{
|
|
||||||
String[] oldMethods = reader.getMethodNames();
|
|
||||||
final int methodOffset = oldMethods.length;
|
|
||||||
|
|
||||||
final String[] newMethods = Arrays.copyOf( oldMethods, oldMethods.length + 2 );
|
|
||||||
newMethods[ methodOffset + 0 ] = "getResponseCode";
|
|
||||||
newMethods[ methodOffset + 1 ] = "getResponseHeaders";
|
|
||||||
|
|
||||||
return new ILuaObject()
|
|
||||||
{
|
|
||||||
@Nonnull
|
|
||||||
@Override
|
|
||||||
public String[] getMethodNames()
|
|
||||||
{
|
|
||||||
return newMethods;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public Object[] callMethod( @Nonnull ILuaContext context, int method, @Nonnull Object[] args ) throws LuaException, InterruptedException
|
|
||||||
{
|
|
||||||
if( method < methodOffset )
|
|
||||||
{
|
|
||||||
return reader.callMethod( context, method, args );
|
|
||||||
}
|
|
||||||
switch( method - methodOffset )
|
|
||||||
{
|
|
||||||
case 0:
|
|
||||||
{
|
|
||||||
// getResponseCode
|
|
||||||
return new Object[] { responseCode };
|
|
||||||
}
|
|
||||||
case 1:
|
|
||||||
{
|
|
||||||
// getResponseHeaders
|
|
||||||
return new Object[] { responseHeaders };
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
{
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void shutdown( )
|
public void shutdown( )
|
||||||
{
|
{
|
||||||
synchronized( m_httpRequests )
|
synchronized( m_httpTasks )
|
||||||
{
|
{
|
||||||
for( HTTPRequest r : m_httpRequests )
|
for( HTTPTask r : m_httpTasks )
|
||||||
{
|
{
|
||||||
r.cancel();
|
r.cancel();
|
||||||
}
|
}
|
||||||
m_httpRequests.clear();
|
m_httpTasks.clear();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -188,10 +124,11 @@ public class HTTPAPI implements ILuaAPI
|
|||||||
// Make the request
|
// Make the request
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
HTTPRequest request = new HTTPRequest( urlString, postString, headers, binary );
|
URL url = HTTPRequest.checkURL( urlString );
|
||||||
synchronized( m_httpRequests )
|
HTTPRequest request = new HTTPRequest( urlString, url, postString, headers, binary );
|
||||||
|
synchronized( m_httpTasks )
|
||||||
{
|
{
|
||||||
m_httpRequests.add( request );
|
m_httpTasks.add( HTTPTask.submit( request ) );
|
||||||
}
|
}
|
||||||
return new Object[] { true };
|
return new Object[] { true };
|
||||||
}
|
}
|
||||||
@ -209,7 +146,11 @@ public class HTTPAPI implements ILuaAPI
|
|||||||
// Check URL
|
// Check URL
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
HTTPRequest.checkURL( urlString );
|
URL url = HTTPRequest.checkURL( urlString );
|
||||||
|
HTTPCheck check = new HTTPCheck( urlString, url );
|
||||||
|
synchronized( m_httpTasks ) {
|
||||||
|
m_httpTasks.add( HTTPTask.submit( check ) );
|
||||||
|
}
|
||||||
return new Object[] { true };
|
return new Object[] { true };
|
||||||
}
|
}
|
||||||
catch( HTTPRequestException e )
|
catch( HTTPRequestException e )
|
||||||
|
@ -1,261 +0,0 @@
|
|||||||
/*
|
|
||||||
* This file is part of ComputerCraft - http://www.computercraft.info
|
|
||||||
* Copyright Daniel Ratcliffe, 2011-2017. Do not distribute without permission.
|
|
||||||
* Send enquiries to dratcliffe@gmail.com
|
|
||||||
*/
|
|
||||||
|
|
||||||
package dan200.computercraft.core.apis;
|
|
||||||
|
|
||||||
import com.google.common.base.Joiner;
|
|
||||||
import com.google.common.io.ByteStreams;
|
|
||||||
import dan200.computercraft.ComputerCraft;
|
|
||||||
|
|
||||||
import java.io.*;
|
|
||||||
import java.net.HttpURLConnection;
|
|
||||||
import java.net.MalformedURLException;
|
|
||||||
import java.net.URL;
|
|
||||||
import java.util.HashMap;
|
|
||||||
import java.util.List;
|
|
||||||
import java.util.Map;
|
|
||||||
import java.util.regex.Pattern;
|
|
||||||
|
|
||||||
public class HTTPRequest
|
|
||||||
{
|
|
||||||
public static URL checkURL( String urlString ) throws HTTPRequestException
|
|
||||||
{
|
|
||||||
URL url;
|
|
||||||
try
|
|
||||||
{
|
|
||||||
url = new URL( urlString );
|
|
||||||
}
|
|
||||||
catch( MalformedURLException e )
|
|
||||||
{
|
|
||||||
throw new HTTPRequestException( "URL malformed" );
|
|
||||||
}
|
|
||||||
|
|
||||||
// Validate the URL
|
|
||||||
String protocol = url.getProtocol().toLowerCase();
|
|
||||||
if( !protocol.equals("http") && !protocol.equals("https") )
|
|
||||||
{
|
|
||||||
throw new HTTPRequestException( "URL not http" );
|
|
||||||
}
|
|
||||||
|
|
||||||
// Compare the URL to the whitelist
|
|
||||||
boolean allowed = false;
|
|
||||||
String whitelistString = ComputerCraft.http_whitelist;
|
|
||||||
String[] allowedURLs = whitelistString.split( ";" );
|
|
||||||
for( String allowedURL : allowedURLs )
|
|
||||||
{
|
|
||||||
Pattern allowedURLPattern = Pattern.compile( "^\\Q" + allowedURL.replaceAll( "\\*", "\\\\E.*\\\\Q" ) + "\\E$" );
|
|
||||||
if( allowedURLPattern.matcher( url.getHost() ).matches() )
|
|
||||||
{
|
|
||||||
allowed = true;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if( !allowed )
|
|
||||||
{
|
|
||||||
throw new HTTPRequestException( "Domain not permitted" );
|
|
||||||
}
|
|
||||||
|
|
||||||
return url;
|
|
||||||
}
|
|
||||||
|
|
||||||
public HTTPRequest( String url, final String postText, final Map<String, String> headers, boolean binary ) throws HTTPRequestException
|
|
||||||
{
|
|
||||||
// Parse the URL
|
|
||||||
m_urlString = url;
|
|
||||||
m_url = checkURL( m_urlString );
|
|
||||||
m_binary = binary;
|
|
||||||
|
|
||||||
// Start the thread
|
|
||||||
m_cancelled = false;
|
|
||||||
m_complete = false;
|
|
||||||
m_success = false;
|
|
||||||
m_result = null;
|
|
||||||
m_responseCode = -1;
|
|
||||||
|
|
||||||
Thread thread = new Thread( new Runnable() {
|
|
||||||
@Override
|
|
||||||
public void run()
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
// Connect to the URL
|
|
||||||
HttpURLConnection connection = (HttpURLConnection)m_url.openConnection();
|
|
||||||
|
|
||||||
if( postText != null )
|
|
||||||
{
|
|
||||||
connection.setRequestMethod( "POST" );
|
|
||||||
connection.setDoOutput( true );
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
connection.setRequestMethod( "GET" );
|
|
||||||
}
|
|
||||||
|
|
||||||
// Set headers
|
|
||||||
connection.setRequestProperty( "accept-charset", "UTF-8" );
|
|
||||||
if( postText != null )
|
|
||||||
{
|
|
||||||
connection.setRequestProperty( "content-type", "application/x-www-form-urlencoded; charset=utf-8" );
|
|
||||||
connection.setRequestProperty( "content-encoding", "UTF-8" );
|
|
||||||
}
|
|
||||||
if( headers != null )
|
|
||||||
{
|
|
||||||
for( Map.Entry<String, String> header : headers.entrySet() )
|
|
||||||
{
|
|
||||||
connection.setRequestProperty( header.getKey(), header.getValue() );
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Send POST text
|
|
||||||
if( postText != null )
|
|
||||||
{
|
|
||||||
OutputStream os = connection.getOutputStream();
|
|
||||||
OutputStreamWriter osw;
|
|
||||||
try
|
|
||||||
{
|
|
||||||
osw = new OutputStreamWriter( os, "UTF-8" );
|
|
||||||
}
|
|
||||||
catch( UnsupportedEncodingException e )
|
|
||||||
{
|
|
||||||
osw = new OutputStreamWriter( os );
|
|
||||||
}
|
|
||||||
BufferedWriter writer = new BufferedWriter( osw );
|
|
||||||
writer.write( postText, 0, postText.length() );
|
|
||||||
writer.close();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Read response
|
|
||||||
InputStream is;
|
|
||||||
int code = connection.getResponseCode();
|
|
||||||
boolean responseSuccess;
|
|
||||||
if (code >= 200 && code < 400) {
|
|
||||||
is = connection.getInputStream();
|
|
||||||
responseSuccess = true;
|
|
||||||
} else {
|
|
||||||
is = connection.getErrorStream();
|
|
||||||
responseSuccess = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
byte[] result = ByteStreams.toByteArray( is );
|
|
||||||
is.close();
|
|
||||||
|
|
||||||
synchronized( m_lock )
|
|
||||||
{
|
|
||||||
if( m_cancelled )
|
|
||||||
{
|
|
||||||
// We cancelled
|
|
||||||
m_complete = true;
|
|
||||||
m_success = false;
|
|
||||||
m_result = null;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
// We completed
|
|
||||||
m_complete = true;
|
|
||||||
m_success = responseSuccess;
|
|
||||||
m_result = result;
|
|
||||||
m_responseCode = connection.getResponseCode();
|
|
||||||
m_encoding = connection.getContentEncoding();
|
|
||||||
|
|
||||||
Joiner joiner = Joiner.on( ',' );
|
|
||||||
Map<String, String> headers = m_responseHeaders = new HashMap<String, String>();
|
|
||||||
for (Map.Entry<String, List<String>> header : connection.getHeaderFields().entrySet()) {
|
|
||||||
headers.put(header.getKey(), joiner.join( header.getValue() ));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
connection.disconnect(); // disconnect
|
|
||||||
|
|
||||||
}
|
|
||||||
catch( IOException e )
|
|
||||||
{
|
|
||||||
synchronized( m_lock )
|
|
||||||
{
|
|
||||||
// There was an error
|
|
||||||
m_complete = true;
|
|
||||||
m_success = false;
|
|
||||||
m_result = null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} );
|
|
||||||
thread.setDaemon(true);
|
|
||||||
thread.start();
|
|
||||||
}
|
|
||||||
|
|
||||||
public String getURL() {
|
|
||||||
return m_urlString;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void cancel()
|
|
||||||
{
|
|
||||||
synchronized(m_lock) {
|
|
||||||
m_cancelled = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public boolean isComplete()
|
|
||||||
{
|
|
||||||
synchronized(m_lock) {
|
|
||||||
return m_complete;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public int getResponseCode() {
|
|
||||||
synchronized(m_lock) {
|
|
||||||
return m_responseCode;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public Map<String, String> getResponseHeaders() {
|
|
||||||
synchronized (m_lock) {
|
|
||||||
return m_responseHeaders;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public boolean wasSuccessful()
|
|
||||||
{
|
|
||||||
synchronized(m_lock) {
|
|
||||||
return m_success;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public boolean isBinary()
|
|
||||||
{
|
|
||||||
return m_binary;
|
|
||||||
}
|
|
||||||
|
|
||||||
public InputStream getContents()
|
|
||||||
{
|
|
||||||
byte[] result;
|
|
||||||
synchronized(m_lock) {
|
|
||||||
result = m_result;
|
|
||||||
}
|
|
||||||
|
|
||||||
if( result != null ) {
|
|
||||||
return new ByteArrayInputStream( result );
|
|
||||||
}
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
public String getEncoding() {
|
|
||||||
return m_encoding;
|
|
||||||
}
|
|
||||||
|
|
||||||
private final Object m_lock = new Object();
|
|
||||||
private final URL m_url;
|
|
||||||
private final String m_urlString;
|
|
||||||
|
|
||||||
private boolean m_complete;
|
|
||||||
private boolean m_cancelled;
|
|
||||||
private boolean m_success;
|
|
||||||
private String m_encoding;
|
|
||||||
private byte[] m_result;
|
|
||||||
private boolean m_binary;
|
|
||||||
private int m_responseCode;
|
|
||||||
private Map<String, String> m_responseHeaders;
|
|
||||||
}
|
|
@ -0,0 +1,45 @@
|
|||||||
|
package dan200.computercraft.core.apis.http;
|
||||||
|
|
||||||
|
import dan200.computercraft.core.apis.HTTPRequestException;
|
||||||
|
import dan200.computercraft.core.apis.IAPIEnvironment;
|
||||||
|
|
||||||
|
import java.net.URL;
|
||||||
|
|
||||||
|
public class HTTPCheck implements HTTPTask.IHTTPTask
|
||||||
|
{
|
||||||
|
private final String urlString;
|
||||||
|
private final URL url;
|
||||||
|
private String error;
|
||||||
|
|
||||||
|
public HTTPCheck( String urlString, URL url )
|
||||||
|
{
|
||||||
|
this.urlString = urlString;
|
||||||
|
this.url = url;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void run()
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
HTTPRequest.checkHost( url );
|
||||||
|
}
|
||||||
|
catch( HTTPRequestException e )
|
||||||
|
{
|
||||||
|
error = e.getMessage();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void whenFinished( IAPIEnvironment environment )
|
||||||
|
{
|
||||||
|
if( error == null )
|
||||||
|
{
|
||||||
|
environment.queueEvent( "http_check", new Object[] { urlString, true } );
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
environment.queueEvent( "http_check", new Object[] { urlString, false, error } );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,290 @@
|
|||||||
|
/*
|
||||||
|
* This file is part of ComputerCraft - http://www.computercraft.info
|
||||||
|
* Copyright Daniel Ratcliffe, 2011-2017. Do not distribute without permission.
|
||||||
|
* Send enquiries to dratcliffe@gmail.com
|
||||||
|
*/
|
||||||
|
|
||||||
|
package dan200.computercraft.core.apis.http;
|
||||||
|
|
||||||
|
import com.google.common.base.Joiner;
|
||||||
|
import com.google.common.io.ByteStreams;
|
||||||
|
import dan200.computercraft.ComputerCraft;
|
||||||
|
import dan200.computercraft.api.lua.ILuaContext;
|
||||||
|
import dan200.computercraft.api.lua.ILuaObject;
|
||||||
|
import dan200.computercraft.api.lua.LuaException;
|
||||||
|
import dan200.computercraft.core.apis.HTTPRequestException;
|
||||||
|
import dan200.computercraft.core.apis.IAPIEnvironment;
|
||||||
|
import dan200.computercraft.core.apis.handles.BinaryInputHandle;
|
||||||
|
import dan200.computercraft.core.apis.handles.EncodedInputHandle;
|
||||||
|
|
||||||
|
import javax.annotation.Nonnull;
|
||||||
|
import java.io.*;
|
||||||
|
import java.net.*;
|
||||||
|
import java.util.Arrays;
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.Map;
|
||||||
|
|
||||||
|
public class HTTPRequest implements HTTPTask.IHTTPTask
|
||||||
|
{
|
||||||
|
public static URL checkURL( String urlString ) throws HTTPRequestException
|
||||||
|
{
|
||||||
|
URL url;
|
||||||
|
try
|
||||||
|
{
|
||||||
|
url = new URL( urlString );
|
||||||
|
}
|
||||||
|
catch( MalformedURLException e )
|
||||||
|
{
|
||||||
|
throw new HTTPRequestException( "URL malformed" );
|
||||||
|
}
|
||||||
|
|
||||||
|
// Validate the URL
|
||||||
|
String protocol = url.getProtocol().toLowerCase();
|
||||||
|
if( !protocol.equals( "http" ) && !protocol.equals( "https" ) )
|
||||||
|
{
|
||||||
|
throw new HTTPRequestException( "URL not http" );
|
||||||
|
}
|
||||||
|
|
||||||
|
// Compare the URL to the whitelist
|
||||||
|
if( !ComputerCraft.http_whitelist.matches( url.getHost() ) || ComputerCraft.http_blacklist.matches( url.getHost() ) )
|
||||||
|
{
|
||||||
|
throw new HTTPRequestException( "Domain not permitted" );
|
||||||
|
}
|
||||||
|
|
||||||
|
return url;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static InetAddress checkHost( URL url ) throws HTTPRequestException
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
InetAddress resolved = InetAddress.getByName( url.getHost() );
|
||||||
|
if( !ComputerCraft.http_whitelist.matches( resolved ) || ComputerCraft.http_blacklist.matches( resolved ) )
|
||||||
|
{
|
||||||
|
throw new HTTPRequestException( "Domain not permitted" );
|
||||||
|
}
|
||||||
|
|
||||||
|
return resolved;
|
||||||
|
}
|
||||||
|
catch( UnknownHostException e )
|
||||||
|
{
|
||||||
|
throw new HTTPRequestException( "Unknown host" );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private final URL m_url;
|
||||||
|
private final String m_urlString;
|
||||||
|
private final String m_postText;
|
||||||
|
private final Map<String, String> m_headers;
|
||||||
|
|
||||||
|
private boolean m_success = false;
|
||||||
|
private String m_encoding;
|
||||||
|
private byte[] m_result;
|
||||||
|
private boolean m_binary;
|
||||||
|
private int m_responseCode = -1;
|
||||||
|
private Map<String, String> m_responseHeaders;
|
||||||
|
private String m_errorMessage;
|
||||||
|
|
||||||
|
public HTTPRequest( String urlString, URL url, final String postText, final Map<String, String> headers, boolean binary ) throws HTTPRequestException
|
||||||
|
{
|
||||||
|
// Parse the URL
|
||||||
|
m_urlString = urlString;
|
||||||
|
m_url = url;
|
||||||
|
m_binary = binary;
|
||||||
|
m_postText = postText;
|
||||||
|
m_headers = headers;
|
||||||
|
}
|
||||||
|
|
||||||
|
public InputStream getContents()
|
||||||
|
{
|
||||||
|
byte[] result = m_result;
|
||||||
|
if( result != null )
|
||||||
|
{
|
||||||
|
return new ByteArrayInputStream( result );
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void run()
|
||||||
|
{
|
||||||
|
// First verify the address is allowed.
|
||||||
|
try
|
||||||
|
{
|
||||||
|
checkHost( m_url );
|
||||||
|
}
|
||||||
|
catch( HTTPRequestException e )
|
||||||
|
{
|
||||||
|
m_success = false;
|
||||||
|
m_errorMessage = e.getMessage();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
// Connect to the URL
|
||||||
|
HttpURLConnection connection = (HttpURLConnection) m_url.openConnection();
|
||||||
|
|
||||||
|
if( m_postText != null )
|
||||||
|
{
|
||||||
|
connection.setRequestMethod( "POST" );
|
||||||
|
connection.setDoOutput( true );
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
connection.setRequestMethod( "GET" );
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set headers
|
||||||
|
connection.setRequestProperty( "accept-charset", "UTF-8" );
|
||||||
|
if( m_postText != null )
|
||||||
|
{
|
||||||
|
connection.setRequestProperty( "content-type", "application/x-www-form-urlencoded; charset=utf-8" );
|
||||||
|
connection.setRequestProperty( "content-encoding", "UTF-8" );
|
||||||
|
}
|
||||||
|
if( m_postText != null )
|
||||||
|
{
|
||||||
|
for( Map.Entry<String, String> header : m_headers.entrySet() )
|
||||||
|
{
|
||||||
|
connection.setRequestProperty( header.getKey(), header.getValue() );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Send POST text
|
||||||
|
if( m_postText != null )
|
||||||
|
{
|
||||||
|
OutputStream os = connection.getOutputStream();
|
||||||
|
OutputStreamWriter osw;
|
||||||
|
try
|
||||||
|
{
|
||||||
|
osw = new OutputStreamWriter( os, "UTF-8" );
|
||||||
|
}
|
||||||
|
catch( UnsupportedEncodingException e )
|
||||||
|
{
|
||||||
|
osw = new OutputStreamWriter( os );
|
||||||
|
}
|
||||||
|
BufferedWriter writer = new BufferedWriter( osw );
|
||||||
|
writer.write( m_postText, 0, m_postText.length() );
|
||||||
|
writer.close();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Read response
|
||||||
|
InputStream is;
|
||||||
|
int code = connection.getResponseCode();
|
||||||
|
boolean responseSuccess;
|
||||||
|
if( code >= 200 && code < 400 )
|
||||||
|
{
|
||||||
|
is = connection.getInputStream();
|
||||||
|
responseSuccess = true;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
is = connection.getErrorStream();
|
||||||
|
responseSuccess = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
byte[] result = ByteStreams.toByteArray( is );
|
||||||
|
is.close();
|
||||||
|
|
||||||
|
// We completed
|
||||||
|
m_success = responseSuccess;
|
||||||
|
m_result = result;
|
||||||
|
m_responseCode = connection.getResponseCode();
|
||||||
|
m_encoding = connection.getContentEncoding();
|
||||||
|
|
||||||
|
Joiner joiner = Joiner.on( ',' );
|
||||||
|
Map<String, String> headers = m_responseHeaders = new HashMap<String, String>();
|
||||||
|
for( Map.Entry<String, List<String>> header : connection.getHeaderFields().entrySet() )
|
||||||
|
{
|
||||||
|
headers.put( header.getKey(), joiner.join( header.getValue() ) );
|
||||||
|
}
|
||||||
|
|
||||||
|
connection.disconnect(); // disconnect
|
||||||
|
}
|
||||||
|
catch( IOException e )
|
||||||
|
{
|
||||||
|
// There was an error
|
||||||
|
m_success = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void whenFinished( IAPIEnvironment environment )
|
||||||
|
{
|
||||||
|
final String url = m_urlString;
|
||||||
|
if( m_success )
|
||||||
|
{
|
||||||
|
// Queue the "http_success" event
|
||||||
|
InputStream contents = getContents();
|
||||||
|
Object result = wrapStream(
|
||||||
|
m_binary ? new BinaryInputHandle( contents ) : new EncodedInputHandle( contents, m_encoding ),
|
||||||
|
m_responseCode, m_responseHeaders
|
||||||
|
);
|
||||||
|
environment.queueEvent( "http_success", new Object[] { url, result } );
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// Queue the "http_failure" event
|
||||||
|
String error = "Could not connect";
|
||||||
|
if( m_errorMessage != null ) error = m_errorMessage;
|
||||||
|
|
||||||
|
InputStream contents = getContents();
|
||||||
|
Object result = null;
|
||||||
|
if( contents != null )
|
||||||
|
{
|
||||||
|
result = wrapStream(
|
||||||
|
m_binary ? new BinaryInputHandle( contents ) : new EncodedInputHandle( contents, m_encoding ),
|
||||||
|
m_responseCode, m_responseHeaders
|
||||||
|
);
|
||||||
|
}
|
||||||
|
environment.queueEvent( "http_failure", new Object[] { url, error, result } );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static ILuaObject wrapStream( final ILuaObject reader, final int responseCode, final Map<String, String> responseHeaders )
|
||||||
|
{
|
||||||
|
String[] oldMethods = reader.getMethodNames();
|
||||||
|
final int methodOffset = oldMethods.length;
|
||||||
|
|
||||||
|
final String[] newMethods = Arrays.copyOf( oldMethods, oldMethods.length + 2 );
|
||||||
|
newMethods[ methodOffset + 0 ] = "getResponseCode";
|
||||||
|
newMethods[ methodOffset + 1 ] = "getResponseHeaders";
|
||||||
|
|
||||||
|
return new ILuaObject()
|
||||||
|
{
|
||||||
|
@Nonnull
|
||||||
|
@Override
|
||||||
|
public String[] getMethodNames()
|
||||||
|
{
|
||||||
|
return newMethods;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Object[] callMethod( @Nonnull ILuaContext context, int method, @Nonnull Object[] args ) throws LuaException, InterruptedException
|
||||||
|
{
|
||||||
|
if( method < methodOffset )
|
||||||
|
{
|
||||||
|
return reader.callMethod( context, method, args );
|
||||||
|
}
|
||||||
|
switch( method - methodOffset )
|
||||||
|
{
|
||||||
|
case 0:
|
||||||
|
{
|
||||||
|
// getResponseCode
|
||||||
|
return new Object[] { responseCode };
|
||||||
|
}
|
||||||
|
case 1:
|
||||||
|
{
|
||||||
|
// getResponseHeaders
|
||||||
|
return new Object[] { responseHeaders };
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
{
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,61 @@
|
|||||||
|
package dan200.computercraft.core.apis.http;
|
||||||
|
|
||||||
|
import com.google.common.util.concurrent.ThreadFactoryBuilder;
|
||||||
|
import dan200.computercraft.core.apis.IAPIEnvironment;
|
||||||
|
|
||||||
|
import java.util.concurrent.*;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* A task which executes asynchronously on a new thread.
|
||||||
|
*
|
||||||
|
* This functions very similarly to a {@link Future}, but with an additional
|
||||||
|
* method which is called on the main thread when the task is completed.
|
||||||
|
*/
|
||||||
|
public class HTTPTask
|
||||||
|
{
|
||||||
|
public interface IHTTPTask extends Runnable
|
||||||
|
{
|
||||||
|
void whenFinished( IAPIEnvironment environment );
|
||||||
|
}
|
||||||
|
|
||||||
|
private static final ExecutorService httpThreads = new ThreadPoolExecutor(
|
||||||
|
4, Integer.MAX_VALUE,
|
||||||
|
60L, TimeUnit.SECONDS,
|
||||||
|
new SynchronousQueue<Runnable>(),
|
||||||
|
new ThreadFactoryBuilder()
|
||||||
|
.setDaemon( true )
|
||||||
|
.setPriority( Thread.MIN_PRIORITY + (Thread.NORM_PRIORITY - Thread.MIN_PRIORITY) / 2 )
|
||||||
|
.setNameFormat( "ComputerCraft-HTTP-%d" )
|
||||||
|
.build()
|
||||||
|
);
|
||||||
|
|
||||||
|
private final Future<?> future;
|
||||||
|
private final IHTTPTask task;
|
||||||
|
|
||||||
|
private HTTPTask( Future<?> future, IHTTPTask task )
|
||||||
|
{
|
||||||
|
this.future = future;
|
||||||
|
this.task = task;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static HTTPTask submit( IHTTPTask task )
|
||||||
|
{
|
||||||
|
Future<?> future = httpThreads.submit( task );
|
||||||
|
return new HTTPTask( future, task );
|
||||||
|
}
|
||||||
|
|
||||||
|
public void cancel()
|
||||||
|
{
|
||||||
|
future.cancel( false );
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isFinished()
|
||||||
|
{
|
||||||
|
return future.isDone();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void whenFinished( IAPIEnvironment environment )
|
||||||
|
{
|
||||||
|
task.whenFinished( environment );
|
||||||
|
}
|
||||||
|
}
|
@ -44,6 +44,7 @@ gui.computercraft:wired_modem.peripheral_disconnected=Peripheral "%s" disconnect
|
|||||||
|
|
||||||
gui.computercraft:config.http_enable=Enable HTTP API
|
gui.computercraft:config.http_enable=Enable HTTP API
|
||||||
gui.computercraft:config.http_whitelist=HTTP whitelist
|
gui.computercraft:config.http_whitelist=HTTP whitelist
|
||||||
|
gui.computercraft:config.http_blacklist=HTTP blacklist
|
||||||
gui.computercraft:config.disable_lua51_features=Disable Lua 5.1 features
|
gui.computercraft:config.disable_lua51_features=Disable Lua 5.1 features
|
||||||
gui.computercraft:config.default_computer_settings=Default Computer settings
|
gui.computercraft:config.default_computer_settings=Default Computer settings
|
||||||
gui.computercraft:config.log_peripheral_errors=Log peripheral errors
|
gui.computercraft:config.log_peripheral_errors=Log peripheral errors
|
||||||
|
@ -741,6 +741,18 @@ if http then
|
|||||||
end
|
end
|
||||||
return ok, err
|
return ok, err
|
||||||
end
|
end
|
||||||
|
|
||||||
|
local nativeCheckURL = http.checkURL
|
||||||
|
http.checkURLAsync = nativeCheckURL
|
||||||
|
http.checkURL = function( _url )
|
||||||
|
local ok, err = nativeCheckURL( _url )
|
||||||
|
if not ok then return ok, err end
|
||||||
|
|
||||||
|
while true do
|
||||||
|
local event, url, ok, err = os.pullEvent( "http_check" )
|
||||||
|
if url == _url then return ok, err end
|
||||||
|
end
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
-- Install the lua part of the FS api
|
-- Install the lua part of the FS api
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
Functions in the HTTP API:
|
Functions in the HTTP API:
|
||||||
http.checkURL( url )
|
http.checkURL( url )
|
||||||
|
http.checkURLAsync( url )
|
||||||
http.request( url, [postData], [headers] )
|
http.request( url, [postData], [headers] )
|
||||||
http.get( url, [headers] )
|
http.get( url, [headers] )
|
||||||
http.post( url, postData, [headers] )
|
http.post( url, postData, [headers] )
|
||||||
|
Loading…
Reference in New Issue
Block a user