io.soluble.pjb.bridge.http.SocketContextServer Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of php-java-bridge Show documentation
Show all versions of php-java-bridge Show documentation
PHPJavaBridge server (soluble fork)
The newest version!
/*-*- mode: Java; tab-width:8 -*-*/
package io.soluble.pjb.bridge.http;
/*
* Copyright (C) 2003-2007 Jost Boekemeier
*
* Permission is hereby granted, free of charge, to any person obtaining a
* copy of this software and associated documentation files (the "Software"),
* to deal in the Software without restriction, including without limitation
* the rights to use, copy, modify, merge, publish, distribute, sublicense,
* and/or sell copies of the Software, and to permit persons to whom the
* Software is furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
* THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR
* OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
* ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
* OTHER DEALINGS IN THE SOFTWARE.
*/
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.Socket;
import java.util.ArrayList;
import java.util.Collections;
import java.util.Iterator;
import java.util.List;
import io.soluble.pjb.bridge.AppThreadPool;
import io.soluble.pjb.bridge.ILogger;
import io.soluble.pjb.bridge.ISocketFactory;
import io.soluble.pjb.bridge.JavaBridge;
import io.soluble.pjb.bridge.Util;
/**
* This class manages the fallback physical connection for the
* operating system which doesn't support named pipes, "Windows", or when the
* System property io.soluble.pjb.bridge.promiscuous is set to true.
*
* When isAvailable() returns true, a server socket bound to the local
* interface (127.0.0.1) has been created on a port in the range
* [9267,...,[9367 and will be used for further communication, see
* response header X_JAVABRIDGE_REDIRECT. If this communication
* channel is not available either, the PHP clients must continue to
* send all statements via PUT requests.
*
* It is possible to switch
* off this server by setting the VM property
* io.soluble.pjb.bridge.no_socket_server to true, e.g.:
* -Dio.soluble.pjb.bridge.no_socket_server=true.
*
* @see io.soluble.pjb.bridge.http.ContextServer
*/
public final class SocketContextServer implements Runnable, IContextServer {
private AppThreadPool threadPool;
private ISocketFactory serverSocket = null;
protected List sockets = Collections.synchronizedList(new ArrayList());
private ILogger logger;
private String contextName;
protected class Channel extends AbstractChannel {
protected Socket sock;
protected InputStream in;
protected OutputStream out;
protected String name;
public Channel(String name, InputStream in, OutputStream out, Socket sock) {
this.name = name;
this.in = in;
this.out = out;
this.sock = sock;
sockets.add(sock);
}
public String getName() {
return name;
}
public InputStream getInputStream() {
return in;
}
public OutputStream getOuptutStream() {
return out;
}
public Socket getSocket() {
return sock;
}
public void shutdown() {
if (in != null) try {
in.close();
} catch (IOException e) {/*ignore*/}
if (out != null) try {
out.close();
} catch (IOException e) {/*ignore*/}
if (sockets.remove(sock)) try {
sock.close();
} catch (IOException e) {/*ignore*/}
}
}
/**
* Create a new ContextServer using the ThreadPool.
*
* @param threadPool Obtain runnables from this pool. If null, new threads will be created.
*/
public SocketContextServer(AppThreadPool threadPool, boolean promiscuous, String contextName) {
this.threadPool = threadPool;
this.contextName = contextName;
try {
serverSocket = JavaBridge.bind(promiscuous ? "INET:0" : "INET_LOCAL:0");
SecurityManager sec = System.getSecurityManager();
if (sec != null) sec.checkAccept("127.0.0.1", Integer.parseInt(serverSocket.getSocketName()));
Thread t = new Util.Thread(this, "JavaBridgeSocketContextServer(" + serverSocket.getSocketName() + ")");
t.start();
} catch (Throwable t) {
Util.warn("Local communication channel not available.");
Util.printStackTrace(t);
if (serverSocket != null) try {
serverSocket.close();
} catch (IOException e) {
}
serverSocket = null;
}
}
private boolean accept() {
InputStream in = null;
OutputStream out = null;
Socket socket = null;
Channel channel = null;
try {
try {
socket = this.serverSocket.accept();
} catch (IOException ex) {
return false;
} // socket closed
in = socket.getInputStream();
out = socket.getOutputStream();
ContextRunner runner = new ContextRunner(channel = new Channel(getChannelName(), in, out, socket), logger);
if (threadPool != null) {
threadPool.start(runner);
} else {
Thread t = new Util.Thread(runner, "JavaBridgeContextRunner(" + contextName + ")");
t.start();
}
} catch (SecurityException t) {
if (channel != null) channel.shutdown();
ContextFactory.destroyAll();
Util.printStackTrace(t);
return false;
} catch (Throwable t) {
if (channel != null) channel.shutdown();
Util.printStackTrace(t);
}
return true;
}
/**
* {@inheritDoc}
*/
public void run() {
while (serverSocket != null) {
if (!accept()) destroy();
}
if (Util.logLevel > 4)
System.err.println("SocketContextServer stopped, the local channel is not available anymore.");
}
private void closeAllSockets() {
synchronized (sockets) {
for (Iterator ii = sockets.iterator(); ii.hasNext(); ) {
Socket sock = (Socket) ii.next();
ii.remove();
try {
sock.close();
} catch (IOException e) {
}
}
}
}
/**
* Destroy the server
*/
public void destroy() {
closeAllSockets();
if (serverSocket != null) {
try {
serverSocket.close();
} catch (IOException e) {
Util.printStackTrace(e);
}
serverSocket = null;
}
}
private static boolean checkTestTunnel(String property) {
try {
return !"true".equals(System.getProperty(property));
} catch (SecurityException e) {
return false;
} catch (Throwable t) {
return true;
}
}
public static final boolean SOCKET_SERVER_AVAIL = checkTestTunnel("io.soluble.pjb.bridge.no_socket_server");
/**
* Check if the ContextServer is ready, i.e. it has created a server socket.
*
* @return true if there's a server socket listening, false otherwise.
*/
public boolean isAvailable() {
// The standalone runner sets an empty context name, otherwise the promiscuous option means that the servlet engine should use chunked encoding
return (SOCKET_SERVER_AVAIL && serverSocket != null);
}
/**
* Returns the server port.
*
* @return The server port.
*/
public String getChannelName() {
return serverSocket.getSocketName();
}
/**
* {@inheritDoc}
*/
public boolean start(AbstractChannelName channelName, ILogger logger) {
this.logger = logger;
return isAvailable();
}
}