Java tutorial
/****************************************************************************** * Copyright 2011-2012 Tavendo GmbH * <p/> * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * <p/> * http://www.apache.org/licenses/LICENSE-2.0 * <p/> * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. ******************************************************************************/ package com.oddrun.befrest.connectivity; import java.io.IOException; import java.lang.ref.WeakReference; import java.net.InetSocketAddress; import java.net.URI; import java.net.URISyntaxException; import java.nio.channels.SocketChannel; import java.util.List; import android.os.Handler; import android.os.HandlerThread; import android.os.Looper; import android.os.Message; import android.util.Log; import org.apache.http.message.BasicNameValuePair; public class WebSocketConnection implements WebSocket { private static final boolean DEBUG = true; private static final String TAG = WebSocketConnection.class.getName(); protected Handler mMasterHandler; protected WebSocketReader mReader; protected WebSocketWriter mWriter; protected HandlerThread mWriterThread; protected SocketChannel mTransportChannel; private URI mWsUri; private String mWsScheme; private String mWsHost; private int mWsPort; private String mWsPath; private String mWsQuery; private String[] mWsSubprotocols; private List<BasicNameValuePair> mWsHeaders; private WebSocket.ConnectionHandler mWsHandler; protected WebSocketOptions mOptions; private boolean mActive; private boolean mPrevConnected; /** * Asynchronous socket connector. */ private class WebSocketConnector extends Thread { public void run() { Thread.currentThread().setName("WebSocketConnector"); /* * connect TCP socket */ try { mTransportChannel = SocketChannel.open(); // the following will block until connection was established or // an error occurred! mTransportChannel.socket().connect(new InetSocketAddress(mWsHost, mWsPort), mOptions.getSocketConnectTimeout()); // before doing any data transfer on the socket, set socket // options mTransportChannel.socket().setSoTimeout(mOptions.getSocketReceiveTimeout()); mTransportChannel.socket().setTcpNoDelay(mOptions.getTcpNoDelay()); } catch (IOException e) { onClose(WebSocketConnectionHandler.CLOSE_CANNOT_CONNECT, e.getMessage()); return; } if (mTransportChannel.isConnected()) { try { // create & start WebSocket reader createReader(); // create & start WebSocket writer createWriter(); // start WebSockets handshake WebSocketMessage.ClientHandshake hs = new WebSocketMessage.ClientHandshake( mWsHost + ":" + mWsPort); hs.mPath = mWsPath; hs.mQuery = mWsQuery; hs.mSubprotocols = mWsSubprotocols; hs.mHeaderList = mWsHeaders; mWriter.forward(hs); mPrevConnected = true; } catch (Exception e) { onClose(WebSocketConnectionHandler.CLOSE_INTERNAL_ERROR, e.getMessage()); return; } } else { onClose(WebSocketConnectionHandler.CLOSE_CANNOT_CONNECT, "Could not connect to WebSocket server"); return; } } } public WebSocketConnection() { if (DEBUG) Log.d(TAG, "created"); // create WebSocket master handler // createHandler(); mMasterHandler = new MasterHandler(new WeakReference<WebSocketConnection>(this)); // set initial values mActive = false; mPrevConnected = false; } public void sendTextMessage(String payload) { mWriter.forward(new WebSocketMessage.TextMessage(payload)); } public void sendRawTextMessage(byte[] payload) { mWriter.forward(new WebSocketMessage.RawTextMessage(payload)); } public void sendBinaryMessage(byte[] payload) { mWriter.forward(new WebSocketMessage.BinaryMessage(payload)); } public boolean isConnected() { return mTransportChannel != null && mTransportChannel.isConnected(); } private void failConnection(int code, String reason) { if (DEBUG) Log.d(TAG, "fail connection [code = " + code + ", reason = " + reason); if (mReader != null) { mReader.quit(); try { mReader.join(); } catch (InterruptedException e) { if (DEBUG) e.printStackTrace(); } mReader = null; } else { if (DEBUG) Log.d(TAG, "mReader already NULL"); } if (mWriter != null) { mWriter.forward(new WebSocketMessage.Quit()); try { mWriterThread.join(); } catch (InterruptedException e) { if (DEBUG) e.printStackTrace(); } mWriter = null; mWriterThread = null; } else { if (DEBUG) Log.d(TAG, "mWriter already NULL"); } if (mTransportChannel != null) { try { mTransportChannel.close(); } catch (IOException e) { if (DEBUG) e.printStackTrace(); } mTransportChannel = null; } else { if (DEBUG) Log.d(TAG, "mTransportChannel already NULL"); } onClose(code, reason); if (DEBUG) Log.d(TAG, "worker threads stopped"); } public void connect(String wsUri, WebSocket.ConnectionHandler wsHandler) throws WebSocketException { connect(wsUri, null, wsHandler, new WebSocketOptions(), null); } public void connect(String wsUri, WebSocket.ConnectionHandler wsHandler, WebSocketOptions options) throws WebSocketException { connect(wsUri, null, wsHandler, options, null); } public void connect(String wsUri, String[] wsSubprotocols, WebSocket.ConnectionHandler wsHandler, WebSocketOptions options, List<BasicNameValuePair> headers) throws WebSocketException { // don't connect if already connected .. user needs to disconnect first // if (mTransportChannel != null && mTransportChannel.isConnected()) { throw new WebSocketException("already connected"); } // parse WebSockets URI // try { mWsUri = new URI(wsUri); if (!mWsUri.getScheme().equals("ws") && !mWsUri.getScheme().equals("wss")) { throw new WebSocketException("unsupported scheme for WebSockets URI"); } if (mWsUri.getScheme().equals("wss")) { throw new WebSocketException("secure WebSockets not implemented"); } mWsScheme = mWsUri.getScheme(); if (mWsUri.getPort() == -1) { if (mWsScheme.equals("ws")) { mWsPort = 80; } else { mWsPort = 443; } } else { mWsPort = mWsUri.getPort(); } if (mWsUri.getHost() == null) { throw new WebSocketException("no host specified in WebSockets URI"); } else { mWsHost = mWsUri.getHost(); } if (mWsUri.getRawPath() == null || mWsUri.getRawPath().equals("")) { mWsPath = "/"; } else { mWsPath = mWsUri.getRawPath(); } if (mWsUri.getRawQuery() == null || mWsUri.getRawQuery().equals("")) { mWsQuery = null; } else { mWsQuery = mWsUri.getRawQuery(); } } catch (URISyntaxException e) { throw new WebSocketException("invalid WebSockets URI"); } mWsSubprotocols = wsSubprotocols; mWsHeaders = headers; mWsHandler = wsHandler; // make copy of options! mOptions = new WebSocketOptions(options); // set connection active mActive = true; // use asynch connector on short-lived background thread new WebSocketConnector().start(); } public void disconnect() { mActive = false; mPrevConnected = false; if (mReader != null) { mReader.quit(); try { mReader.join(); } catch (InterruptedException e) { if (DEBUG) e.printStackTrace(); } mReader = null; } else { if (DEBUG) Log.d(TAG, "mReader already NULL"); } if (mWriter != null) { mWriter.forward(new WebSocketMessage.Quit()); try { mWriterThread.join(); } catch (InterruptedException e) { if (DEBUG) e.printStackTrace(); } mWriter = null; mWriterThread = null; } else { if (DEBUG) Log.d(TAG, "mWriter already NULL"); } if (mTransportChannel != null) { try { mTransportChannel.close(); } catch (IOException e) { if (DEBUG) e.printStackTrace(); } mTransportChannel = null; } else { if (DEBUG) Log.d(TAG, "mTransportChannel already NULL"); } if (DEBUG) Log.d(TAG, "disconnected"); } /** * Reconnect to the server with the latest options * * @return true if reconnection performed */ public boolean reconnect() { if (!isConnected() && (mWsUri != null)) { new WebSocketConnector().start(); return true; } return false; } /** * Perform reconnection * * @return true if reconnection was scheduled */ protected boolean scheduleReconnect() { /** * Reconnect only if: * - connection active (connected but not disconnected) * - has previous success connections * - reconnect interval is set */ int interval = mOptions.getReconnectInterval(); boolean need = mActive && mPrevConnected && (interval > 0); if (need) { if (DEBUG) Log.d(TAG, "Reconnection scheduled"); mMasterHandler.postDelayed(new Runnable() { public void run() { if (DEBUG) Log.d(TAG, "Reconnecting..."); reconnect(); } }, interval); } return need; } /** * Common close handler * * @param code Close code. * @param reason Close reason (human-readable). */ private void onClose(int code, String reason) { boolean reconnecting = false; if ((code == ConnectionHandler.CLOSE_CANNOT_CONNECT) || (code == ConnectionHandler.CLOSE_CONNECTION_LOST)) { // reconnecting = scheduleReconnect(); } if (mWsHandler != null) { try { if (reconnecting) { mWsHandler.onClose(ConnectionHandler.CLOSE_RECONNECT, reason); } else { mWsHandler.onClose(code, reason); } } catch (Exception e) { if (DEBUG) e.printStackTrace(); } //mWsHandler = null; } else { if (DEBUG) Log.d(TAG, "mWsHandler already NULL"); } } /** * Create master message handler. */ /** * protected void createHandler() { * <p/> * mMasterHandler = new Handler(Looper.getMainLooper()) { * <p/> * public void handleMessage(Message msg) { * <p/> * if (msg.obj instanceof WebSocketMessage.TextMessage) { * <p/> * WebSocketMessage.TextMessage textMessage = (WebSocketMessage.TextMessage) msg.obj; * <p/> * <p/> * if (mWsHandler != null) { * mWsHandler.onTextMessage(textMessage.mPayload); * } else { * if (DEBUG) * Log.d(TAG, "could not call onTextMessage() .. handler already NULL"); * } * <p/> * } else if (msg.obj instanceof WebSocketMessage.RawTextMessage) { * <p/> * WebSocketMessage.RawTextMessage rawTextMessage = (WebSocketMessage.RawTextMessage) msg.obj; * <p/> * if (mWsHandler != null) { * mWsHandler.onRawTextMessage(rawTextMessage.mPayload); * } else { * if (DEBUG) * Log.d(TAG, "could not call onRawTextMessage() .. handler already NULL"); * } * <p/> * } else if (msg.obj instanceof WebSocketMessage.BinaryMessage) { * <p/> * WebSocketMessage.BinaryMessage binaryMessage = (WebSocketMessage.BinaryMessage) msg.obj; * <p/> * if (mWsHandler != null) { * mWsHandler.onBinaryMessage(binaryMessage.mPayload); * } else { * if (DEBUG) * Log.d(TAG, "could not call onBinaryMessage() .. handler already NULL"); * } * <p/> * } else if (msg.obj instanceof WebSocketMessage.Ping) { * <p/> * WebSocketMessage.Ping ping = (WebSocketMessage.Ping) msg.obj; * if (DEBUG) Log.d(TAG, "WebSockets Ping received"); * <p/> * // reply with Pong * WebSocketMessage.Pong pong = new WebSocketMessage.Pong(); * pong.mPayload = ping.mPayload; * mWriter.forward(pong); * <p/> * } else if (msg.obj instanceof WebSocketMessage.Pong) { * * @SuppressWarnings("unused") WebSocketMessage.Pong pong = (WebSocketMessage.Pong) msg.obj; * <p/> * if (DEBUG) Log.d(TAG, "WebSockets Pong received"); * <p/> * } else if (msg.obj instanceof WebSocketMessage.Close) { * <p/> * WebSocketMessage.Close close = (WebSocketMessage.Close) msg.obj; * <p/> * if (DEBUG) * Log.d(TAG, "WebSockets Close received (" + close.mCode + " - " + close.mReason + ")"); * <p/> * final int tavendoCloseCode = (close.mCode == 1000) ? ConnectionHandler.CLOSE_NORMAL : ConnectionHandler.CLOSE_CONNECTION_LOST; * <p/> * if (mActive) { * mWriter.forward(new WebSocketMessage.Close(1000)); * } else { * // we've initiated disconnect, so ready to close the channel * try { * mTransportChannel.close(); * } catch (IOException e) { * if (DEBUG) e.printStackTrace(); * } * } * <p/> * onClose(tavendoCloseCode, close.mReason); * <p/> * } else if (msg.obj instanceof WebSocketMessage.ServerHandshake) { * <p/> * WebSocketMessage.ServerHandshake serverHandshake = (WebSocketMessage.ServerHandshake) msg.obj; * <p/> * if (DEBUG) Log.d(TAG, "opening handshake received"); * <p/> * if (serverHandshake.mSuccess) { * if (mWsHandler != null) { * mWsHandler.onOpen(); * } else { * if (DEBUG) * Log.d(TAG, "could not call onOpen() .. handler already NULL"); * } * } * <p/> * } else if (msg.obj instanceof WebSocketMessage.ConnectionLost) { * @SuppressWarnings("unused") WebSocketMessage.ConnectionLost connnectionLost = (WebSocketMessage.ConnectionLost) msg.obj; * failConnection(WebSocketConnectionHandler.CLOSE_CONNECTION_LOST, "WebSockets connection lost"); * <p/> * } else if (msg.obj instanceof WebSocketMessage.ProtocolViolation) { * @SuppressWarnings("unused") WebSocketMessage.ProtocolViolation protocolViolation = (WebSocketMessage.ProtocolViolation) msg.obj; * failConnection(WebSocketConnectionHandler.CLOSE_PROTOCOL_ERROR, "WebSockets protocol violation"); * <p/> * } else if (msg.obj instanceof WebSocketMessage.Error) { * <p/> * WebSocketMessage.Error error = (WebSocketMessage.Error) msg.obj; * failConnection(WebSocketConnectionHandler.CLOSE_INTERNAL_ERROR, "WebSockets internal error (" + error.mException.toString() + ")"); * <p/> * } else if (msg.obj instanceof WebSocketMessage.ServerError) { * <p/> * WebSocketMessage.ServerError error = (WebSocketMessage.ServerError) msg.obj; * int errCode = WebSocketConnectionHandler.CLOSE_SERVER_ERROR; * if (error.mStatusCode == 401) * errCode = WebSocketConnectionHandler.CLOSE_UNAUTHORIZED; //hojjat * failConnection(errCode, "Server error " + error.mStatusCode + " (" + error.mStatusMessage + ")"); * <p/> * } else { * <p/> * processAppMessage(msg.obj); * <p/> * } * } * }; * } */ protected void processAppMessage(Object message) { } /** * Create WebSockets background writer. */ protected void createWriter() { mWriterThread = new HandlerThread("WebSocketWriter"); mWriterThread.start(); mWriter = new WebSocketWriter(mWriterThread.getLooper(), mMasterHandler, mTransportChannel, mOptions); if (DEBUG) Log.d(TAG, "WS writer created and started"); } /** * Create WebSockets background reader. */ protected void createReader() { mReader = new WebSocketReader(mMasterHandler, mTransportChannel, mOptions, "WebSocketReader"); mReader.start(); if (DEBUG) Log.d(TAG, "WS reader created and started"); } private static class MasterHandler extends Handler { WebSocketConnection wsConnection; public MasterHandler(WeakReference<WebSocketConnection> weakReference) { super(Looper.getMainLooper()); wsConnection = weakReference.get(); } @Override public void handleMessage(Message msg) { if (msg.obj instanceof WebSocketMessage.TextMessage) { WebSocketMessage.TextMessage textMessage = (WebSocketMessage.TextMessage) msg.obj; if (wsConnection.mWsHandler != null) { wsConnection.mWsHandler.onTextMessage(textMessage.mPayload); } else { if (DEBUG) Log.d(TAG, "could not call onTextMessage() .. handler already NULL"); } } else if (msg.obj instanceof WebSocketMessage.RawTextMessage) { WebSocketMessage.RawTextMessage rawTextMessage = (WebSocketMessage.RawTextMessage) msg.obj; if (wsConnection.mWsHandler != null) { wsConnection.mWsHandler.onRawTextMessage(rawTextMessage.mPayload); } else { if (DEBUG) Log.d(TAG, "could not call onRawTextMessage() .. handler already NULL"); } } else if (msg.obj instanceof WebSocketMessage.BinaryMessage) { WebSocketMessage.BinaryMessage binaryMessage = (WebSocketMessage.BinaryMessage) msg.obj; if (wsConnection.mWsHandler != null) { wsConnection.mWsHandler.onBinaryMessage(binaryMessage.mPayload); } else { if (DEBUG) Log.d(TAG, "could not call onBinaryMessage() .. handler already NULL"); } } else if (msg.obj instanceof WebSocketMessage.Ping) { WebSocketMessage.Ping ping = (WebSocketMessage.Ping) msg.obj; if (DEBUG) Log.d(TAG, "WebSockets Ping received"); // reply with Pong WebSocketMessage.Pong pong = new WebSocketMessage.Pong(); pong.mPayload = ping.mPayload; wsConnection.mWriter.forward(pong); } else if (msg.obj instanceof WebSocketMessage.Pong) { @SuppressWarnings("unused") WebSocketMessage.Pong pong = (WebSocketMessage.Pong) msg.obj; if (DEBUG) Log.d(TAG, "WebSockets Pong received"); } else if (msg.obj instanceof WebSocketMessage.Close) { WebSocketMessage.Close close = (WebSocketMessage.Close) msg.obj; if (DEBUG) Log.d(TAG, "WebSockets Close received (" + close.mCode + " - " + close.mReason + ")"); final int tavendoCloseCode = (close.mCode == 1000) ? ConnectionHandler.CLOSE_NORMAL : ConnectionHandler.CLOSE_CONNECTION_LOST; if (wsConnection.mActive) { // wsConnection.mWriter.forward(new WebSocketMessage.Close(1000)); wsConnection.disconnect(); } else { // we've initiated disconnect, so ready to close the channel // try { // wsConnection.mTransportChannel.close(); // } catch (IOException e) { // if (DEBUG) e.printStackTrace(); // } } wsConnection.onClose(tavendoCloseCode, close.mReason); } else if (msg.obj instanceof WebSocketMessage.ServerHandshake) { WebSocketMessage.ServerHandshake serverHandshake = (WebSocketMessage.ServerHandshake) msg.obj; if (DEBUG) Log.d(TAG, "opening handshake received"); if (serverHandshake.mSuccess) { if (wsConnection.mWsHandler != null) { wsConnection.mWsHandler.onOpen(); } else { if (DEBUG) Log.d(TAG, "could not call onOpen() .. handler already NULL"); } } } else if (msg.obj instanceof WebSocketMessage.ConnectionLost) { @SuppressWarnings("unused") WebSocketMessage.ConnectionLost connnectionLost = (WebSocketMessage.ConnectionLost) msg.obj; wsConnection.failConnection(WebSocketConnectionHandler.CLOSE_CONNECTION_LOST, "WebSockets connection lost"); } else if (msg.obj instanceof WebSocketMessage.ProtocolViolation) { @SuppressWarnings("unused") WebSocketMessage.ProtocolViolation protocolViolation = (WebSocketMessage.ProtocolViolation) msg.obj; wsConnection.failConnection(WebSocketConnectionHandler.CLOSE_PROTOCOL_ERROR, "WebSockets protocol violation"); } else if (msg.obj instanceof WebSocketMessage.Error) { WebSocketMessage.Error error = (WebSocketMessage.Error) msg.obj; wsConnection.failConnection(WebSocketConnectionHandler.CLOSE_INTERNAL_ERROR, "WebSockets internal error (" + error.mException.toString() + ")"); } else if (msg.obj instanceof WebSocketMessage.ServerError) { WebSocketMessage.ServerError error = (WebSocketMessage.ServerError) msg.obj; int errCode = WebSocketConnectionHandler.CLOSE_SERVER_ERROR; if (error.mStatusCode == 401) errCode = WebSocketConnectionHandler.CLOSE_UNAUTHORIZED; //hojjat wsConnection.failConnection(errCode, "Server error " + error.mStatusCode + " (" + error.mStatusMessage + ")"); } else { wsConnection.processAppMessage(msg.obj); } } } }