mirror of
https://github.com/SquidDev-CC/CC-Tweaked
synced 2025-01-15 11:45:42 +00:00
Remove HTTPTask, queueing the event when it has finished executing
This means we don't have to have lots of shared state between the run and whenFinished method, and allows for easier chaining of futures later on.
This commit is contained in:
parent
1fdfcdb5f2
commit
4bd5b0d236
@ -10,18 +10,19 @@ import dan200.computercraft.api.lua.ILuaContext;
|
|||||||
import dan200.computercraft.api.lua.LuaException;
|
import dan200.computercraft.api.lua.LuaException;
|
||||||
import dan200.computercraft.core.apis.http.HTTPCheck;
|
import dan200.computercraft.core.apis.http.HTTPCheck;
|
||||||
import dan200.computercraft.core.apis.http.HTTPRequest;
|
import dan200.computercraft.core.apis.http.HTTPRequest;
|
||||||
import dan200.computercraft.core.apis.http.HTTPTask;
|
import dan200.computercraft.core.apis.http.HTTPExecutor;
|
||||||
|
|
||||||
import javax.annotation.Nonnull;
|
import javax.annotation.Nonnull;
|
||||||
import java.net.URL;
|
import java.net.URL;
|
||||||
import java.util.*;
|
import java.util.*;
|
||||||
|
import java.util.concurrent.Future;
|
||||||
|
|
||||||
import static dan200.computercraft.core.apis.ArgumentHelper.*;
|
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<HTTPTask> m_httpTasks;
|
private final List<Future<?>> m_httpTasks;
|
||||||
|
|
||||||
public HTTPAPI( IAPIEnvironment environment )
|
public HTTPAPI( IAPIEnvironment environment )
|
||||||
{
|
{
|
||||||
@ -48,15 +49,11 @@ public class HTTPAPI implements ILuaAPI
|
|||||||
// Wait for all of our http requests
|
// Wait for all of our http requests
|
||||||
synchronized( m_httpTasks )
|
synchronized( m_httpTasks )
|
||||||
{
|
{
|
||||||
Iterator<HTTPTask> it = m_httpTasks.iterator();
|
Iterator<Future<?>> it = m_httpTasks.iterator();
|
||||||
while( it.hasNext() )
|
while( it.hasNext() )
|
||||||
{
|
{
|
||||||
final HTTPTask h = it.next();
|
final Future<?> h = it.next();
|
||||||
if( h.isFinished() )
|
if( h.isDone() ) it.remove();
|
||||||
{
|
|
||||||
h.whenFinished( m_apiEnvironment );
|
|
||||||
it.remove();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -66,9 +63,9 @@ public class HTTPAPI implements ILuaAPI
|
|||||||
{
|
{
|
||||||
synchronized( m_httpTasks )
|
synchronized( m_httpTasks )
|
||||||
{
|
{
|
||||||
for( HTTPTask r : m_httpTasks )
|
for( Future<?> r : m_httpTasks )
|
||||||
{
|
{
|
||||||
r.cancel();
|
r.cancel( false );
|
||||||
}
|
}
|
||||||
m_httpTasks.clear();
|
m_httpTasks.clear();
|
||||||
}
|
}
|
||||||
@ -125,10 +122,10 @@ public class HTTPAPI implements ILuaAPI
|
|||||||
try
|
try
|
||||||
{
|
{
|
||||||
URL url = HTTPRequest.checkURL( urlString );
|
URL url = HTTPRequest.checkURL( urlString );
|
||||||
HTTPRequest request = new HTTPRequest( urlString, url, postString, headers, binary );
|
HTTPRequest request = new HTTPRequest( m_apiEnvironment, urlString, url, postString, headers, binary );
|
||||||
synchronized( m_httpTasks )
|
synchronized( m_httpTasks )
|
||||||
{
|
{
|
||||||
m_httpTasks.add( HTTPTask.submit( request ) );
|
m_httpTasks.add( HTTPExecutor.EXECUTOR.submit( request ) );
|
||||||
}
|
}
|
||||||
return new Object[] { true };
|
return new Object[] { true };
|
||||||
}
|
}
|
||||||
@ -147,9 +144,10 @@ public class HTTPAPI implements ILuaAPI
|
|||||||
try
|
try
|
||||||
{
|
{
|
||||||
URL url = HTTPRequest.checkURL( urlString );
|
URL url = HTTPRequest.checkURL( urlString );
|
||||||
HTTPCheck check = new HTTPCheck( urlString, url );
|
HTTPCheck check = new HTTPCheck( m_apiEnvironment, urlString, url );
|
||||||
synchronized( m_httpTasks ) {
|
synchronized( m_httpTasks )
|
||||||
m_httpTasks.add( HTTPTask.submit( check ) );
|
{
|
||||||
|
m_httpTasks.add( HTTPExecutor.EXECUTOR.submit( check ) );
|
||||||
}
|
}
|
||||||
return new Object[] { true };
|
return new Object[] { true };
|
||||||
}
|
}
|
||||||
|
@ -5,14 +5,15 @@ import dan200.computercraft.core.apis.IAPIEnvironment;
|
|||||||
|
|
||||||
import java.net.URL;
|
import java.net.URL;
|
||||||
|
|
||||||
public class HTTPCheck implements HTTPTask.IHTTPTask
|
public class HTTPCheck implements Runnable
|
||||||
{
|
{
|
||||||
|
private final IAPIEnvironment environment;
|
||||||
private final String urlString;
|
private final String urlString;
|
||||||
private final URL url;
|
private final URL url;
|
||||||
private String error;
|
|
||||||
|
|
||||||
public HTTPCheck( String urlString, URL url )
|
public HTTPCheck( IAPIEnvironment environment, String urlString, URL url )
|
||||||
{
|
{
|
||||||
|
this.environment = environment;
|
||||||
this.urlString = urlString;
|
this.urlString = urlString;
|
||||||
this.url = url;
|
this.url = url;
|
||||||
}
|
}
|
||||||
@ -22,24 +23,12 @@ public class HTTPCheck implements HTTPTask.IHTTPTask
|
|||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
HTTPRequest.checkHost( url );
|
HTTPRequest.checkHost( url.getHost() );
|
||||||
|
environment.queueEvent( "http_check", new Object[] { urlString, true } );
|
||||||
}
|
}
|
||||||
catch( HTTPRequestException e )
|
catch( HTTPRequestException e )
|
||||||
{
|
{
|
||||||
error = e.getMessage();
|
environment.queueEvent( "http_check", new Object[] { urlString, false, 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,36 @@
|
|||||||
|
/*
|
||||||
|
* 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.util.concurrent.ListeningExecutorService;
|
||||||
|
import com.google.common.util.concurrent.MoreExecutors;
|
||||||
|
import com.google.common.util.concurrent.ThreadFactoryBuilder;
|
||||||
|
|
||||||
|
import java.util.concurrent.SynchronousQueue;
|
||||||
|
import java.util.concurrent.ThreadPoolExecutor;
|
||||||
|
import java.util.concurrent.TimeUnit;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Just a shared object for executing simple HTTP related tasks.
|
||||||
|
*/
|
||||||
|
public final class HTTPExecutor
|
||||||
|
{
|
||||||
|
public static final ListeningExecutorService EXECUTOR = MoreExecutors.listeningDecorator( 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 HTTPExecutor()
|
||||||
|
{
|
||||||
|
}
|
||||||
|
}
|
@ -25,7 +25,7 @@ import java.util.HashMap;
|
|||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
|
|
||||||
public class HTTPRequest implements HTTPTask.IHTTPTask
|
public class HTTPRequest implements Runnable
|
||||||
{
|
{
|
||||||
public static URL checkURL( String urlString ) throws HTTPRequestException
|
public static URL checkURL( String urlString ) throws HTTPRequestException
|
||||||
{
|
{
|
||||||
@ -55,11 +55,11 @@ public class HTTPRequest implements HTTPTask.IHTTPTask
|
|||||||
return url;
|
return url;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static InetAddress checkHost( URL url ) throws HTTPRequestException
|
public static InetAddress checkHost( String host ) throws HTTPRequestException
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
InetAddress resolved = InetAddress.getByName( url.getHost() );
|
InetAddress resolved = InetAddress.getByName( host );
|
||||||
if( !ComputerCraft.http_whitelist.matches( resolved ) || ComputerCraft.http_blacklist.matches( resolved ) )
|
if( !ComputerCraft.http_whitelist.matches( resolved ) || ComputerCraft.http_blacklist.matches( resolved ) )
|
||||||
{
|
{
|
||||||
throw new HTTPRequestException( "Domain not permitted" );
|
throw new HTTPRequestException( "Domain not permitted" );
|
||||||
@ -73,22 +73,16 @@ public class HTTPRequest implements HTTPTask.IHTTPTask
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private final IAPIEnvironment m_environment;
|
||||||
private final URL m_url;
|
private final URL m_url;
|
||||||
private final String m_urlString;
|
private final String m_urlString;
|
||||||
private final String m_postText;
|
private final String m_postText;
|
||||||
private final Map<String, String> m_headers;
|
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 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
|
public HTTPRequest( IAPIEnvironment environment, String urlString, URL url, final String postText, final Map<String, String> headers, boolean binary ) throws HTTPRequestException
|
||||||
{
|
{
|
||||||
// Parse the URL
|
m_environment = environment;
|
||||||
m_urlString = urlString;
|
m_urlString = urlString;
|
||||||
m_url = url;
|
m_url = url;
|
||||||
m_binary = binary;
|
m_binary = binary;
|
||||||
@ -96,28 +90,19 @@ public class HTTPRequest implements HTTPTask.IHTTPTask
|
|||||||
m_headers = headers;
|
m_headers = headers;
|
||||||
}
|
}
|
||||||
|
|
||||||
public InputStream getContents()
|
|
||||||
{
|
|
||||||
byte[] result = m_result;
|
|
||||||
if( result != null )
|
|
||||||
{
|
|
||||||
return new ByteArrayInputStream( result );
|
|
||||||
}
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void run()
|
public void run()
|
||||||
{
|
{
|
||||||
// First verify the address is allowed.
|
// First verify the address is allowed.
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
checkHost( m_url );
|
checkHost( m_url.getHost() );
|
||||||
}
|
}
|
||||||
catch( HTTPRequestException e )
|
catch( HTTPRequestException e )
|
||||||
{
|
{
|
||||||
m_success = false;
|
// Queue the failure event if not.
|
||||||
m_errorMessage = e.getMessage();
|
String error = e.getMessage();
|
||||||
|
m_environment.queueEvent( "http_failure", new Object[] { m_urlString, error == null ? "Could not connect" : error, null } );
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -186,58 +171,36 @@ public class HTTPRequest implements HTTPTask.IHTTPTask
|
|||||||
byte[] result = ByteStreams.toByteArray( is );
|
byte[] result = ByteStreams.toByteArray( is );
|
||||||
is.close();
|
is.close();
|
||||||
|
|
||||||
// We completed
|
// We've got some sort of response, so let's build a resulting object.
|
||||||
m_success = responseSuccess;
|
|
||||||
m_result = result;
|
|
||||||
m_responseCode = connection.getResponseCode();
|
|
||||||
m_encoding = connection.getContentEncoding();
|
|
||||||
|
|
||||||
Joiner joiner = Joiner.on( ',' );
|
Joiner joiner = Joiner.on( ',' );
|
||||||
Map<String, String> headers = m_responseHeaders = new HashMap<String, String>();
|
Map<String, String> headers = new HashMap<String, String>();
|
||||||
for( Map.Entry<String, List<String>> header : connection.getHeaderFields().entrySet() )
|
for( Map.Entry<String, List<String>> header : connection.getHeaderFields().entrySet() )
|
||||||
{
|
{
|
||||||
headers.put( header.getKey(), joiner.join( header.getValue() ) );
|
headers.put( header.getKey(), joiner.join( header.getValue() ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
InputStream contents = new ByteArrayInputStream( result );
|
||||||
|
ILuaObject stream = wrapStream(
|
||||||
|
m_binary ? new BinaryInputHandle( contents ) : new EncodedInputHandle( contents, connection.getContentEncoding() ),
|
||||||
|
connection.getResponseCode(), headers
|
||||||
|
);
|
||||||
|
|
||||||
connection.disconnect(); // disconnect
|
connection.disconnect(); // disconnect
|
||||||
|
|
||||||
|
// Queue the appropriate event.
|
||||||
|
if( responseSuccess )
|
||||||
|
{
|
||||||
|
m_environment.queueEvent( "http_success", new Object[] { m_urlString, stream } );
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
m_environment.queueEvent( "http_failure", new Object[] { m_urlString, "Could not connect", stream } );
|
||||||
|
}
|
||||||
}
|
}
|
||||||
catch( IOException e )
|
catch( IOException e )
|
||||||
{
|
{
|
||||||
// There was an error
|
// There was an error
|
||||||
m_success = false;
|
m_environment.queueEvent( "http_failure", new Object[] { m_urlString, "Could not connect", null } );
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@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 } );
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,61 +0,0 @@
|
|||||||
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 );
|
|
||||||
}
|
|
||||||
}
|
|
Loading…
Reference in New Issue
Block a user