| /* |
| * libjingle |
| * Copyright 2014, Google Inc. |
| * |
| * Redistribution and use in source and binary forms, with or without |
| * modification, are permitted provided that the following conditions are met: |
| * |
| * 1. Redistributions of source code must retain the above copyright notice, |
| * this list of conditions and the following disclaimer. |
| * 2. Redistributions in binary form must reproduce the above copyright notice, |
| * this list of conditions and the following disclaimer in the documentation |
| * and/or other materials provided with the distribution. |
| * 3. The name of the author may not be used to endorse or promote products |
| * derived from this software without specific prior written permission. |
| * |
| * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED |
| * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF |
| * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO |
| * EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, |
| * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, |
| * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; |
| * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, |
| * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR |
| * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF |
| * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| */ |
| package org.appspot.apprtc; |
| |
| import android.util.Log; |
| |
| import de.tavendo.autobahn.WebSocket.WebSocketConnectionObserver; |
| import de.tavendo.autobahn.WebSocketConnection; |
| import de.tavendo.autobahn.WebSocketException; |
| |
| import org.json.JSONException; |
| import org.json.JSONObject; |
| |
| import java.net.URI; |
| import java.net.URISyntaxException; |
| import java.util.LinkedList; |
| |
| import org.appspot.apprtc.util.AsyncHttpURLConnection; |
| import org.appspot.apprtc.util.AsyncHttpURLConnection.AsyncHttpEvents; |
| import org.appspot.apprtc.util.LooperExecutor; |
| |
| /** |
| * WebSocket client implementation. |
| * All public methods should be called from a looper executor thread |
| * passed in constructor. |
| * All events are issued on the same thread. |
| */ |
| |
| public class WebSocketChannelClient { |
| private static final String TAG = "WSChannelRTCClient"; |
| private static final int CLOSE_TIMEOUT = 1000; |
| private final WebSocketChannelEvents events; |
| private final LooperExecutor executor; |
| private WebSocketConnection ws; |
| private WebSocketObserver wsObserver; |
| private String wsServerUrl; |
| private String postServerUrl; |
| private String roomID; |
| private String clientID; |
| private WebSocketConnectionState state; |
| private final Object closeEventLock = new Object(); |
| private boolean closeEvent; |
| // WebSocket send queue. Messages are added to the queue when WebSocket |
| // client is not registered and are consumed in register() call. |
| private LinkedList<String> wsSendQueue; |
| |
| /** |
| * WebSocketConnectionState is the names of possible WS connection states. |
| */ |
| public enum WebSocketConnectionState { |
| NEW, CONNECTED, REGISTERED, CLOSED, ERROR |
| }; |
| |
| /** |
| * Callback interface for messages delivered on WebSocket. |
| * All events are invoked from UI thread. |
| */ |
| public interface WebSocketChannelEvents { |
| public void onWebSocketOpen(); |
| public void onWebSocketMessage(final String message); |
| public void onWebSocketClose(); |
| public void onWebSocketError(final String description); |
| } |
| |
| public WebSocketChannelClient(LooperExecutor executor, |
| WebSocketChannelEvents events) { |
| this.executor = executor; |
| this.events = events; |
| roomID = null; |
| clientID = null; |
| wsSendQueue = new LinkedList<String>(); |
| state = WebSocketConnectionState.NEW; |
| } |
| |
| public WebSocketConnectionState getState() { |
| return state; |
| } |
| |
| public void connect(final String wsUrl, final String postUrl, |
| final String roomID, final String clientID) { |
| if (state != WebSocketConnectionState.NEW) { |
| Log.e(TAG, "WebSocket is already connected."); |
| return; |
| } |
| wsServerUrl = wsUrl; |
| postServerUrl = postUrl; |
| this.roomID = roomID; |
| this.clientID = clientID; |
| closeEvent = false; |
| |
| Log.d(TAG, "Connecting WebSocket to: " + wsUrl + ". Post URL: " + postUrl); |
| ws = new WebSocketConnection(); |
| wsObserver = new WebSocketObserver(); |
| try { |
| ws.connect(new URI(wsServerUrl), wsObserver); |
| } catch (URISyntaxException e) { |
| reportError("URI error: " + e.getMessage()); |
| } catch (WebSocketException e) { |
| reportError("WebSocket connection error: " + e.getMessage()); |
| } |
| } |
| |
| public void register() { |
| if (state != WebSocketConnectionState.CONNECTED) { |
| Log.w(TAG, "WebSocket register() in state " + state); |
| return; |
| } |
| JSONObject json = new JSONObject(); |
| try { |
| json.put("cmd", "register"); |
| json.put("roomid", roomID); |
| json.put("clientid", clientID); |
| Log.d(TAG, "C->WSS: " + json.toString()); |
| ws.sendTextMessage(json.toString()); |
| state = WebSocketConnectionState.REGISTERED; |
| // Send any previously accumulated messages. |
| synchronized (wsSendQueue) { |
| for (String sendMessage : wsSendQueue) { |
| send(sendMessage); |
| } |
| wsSendQueue.clear(); |
| } |
| } catch (JSONException e) { |
| reportError("WebSocket register JSON error: " + e.getMessage()); |
| } |
| } |
| |
| public void send(String message) { |
| switch (state) { |
| case NEW: |
| case CONNECTED: |
| // Store outgoing messages and send them after websocket client |
| // is registered. |
| Log.d(TAG, "WS ACC: " + message); |
| synchronized (wsSendQueue) { |
| wsSendQueue.add(message); |
| return; |
| } |
| case ERROR: |
| case CLOSED: |
| Log.e(TAG, "WebSocket send() in error or closed state : " + message); |
| return; |
| case REGISTERED: |
| JSONObject json = new JSONObject(); |
| try { |
| json.put("cmd", "send"); |
| json.put("msg", message); |
| message = json.toString(); |
| Log.d(TAG, "C->WSS: " + message); |
| ws.sendTextMessage(message); |
| } catch (JSONException e) { |
| reportError("WebSocket send JSON error: " + e.getMessage()); |
| } |
| break; |
| } |
| return; |
| } |
| |
| // This call can be used to send WebSocket messages before WebSocket |
| // connection is opened. However for now this way of sending messages |
| // is not used until possible race condition of arriving ice candidates |
| // send through websocket before SDP answer sent through http post will be |
| // resolved. |
| public void post(String message) { |
| sendWSSMessage("POST", message); |
| } |
| |
| public void disconnect(boolean waitForComplete) { |
| Log.d(TAG, "Disonnect WebSocket. State: " + state); |
| if (state == WebSocketConnectionState.REGISTERED) { |
| send("{\"type\": \"bye\"}"); |
| state = WebSocketConnectionState.CONNECTED; |
| } |
| // Close WebSocket in CONNECTED or ERROR states only. |
| if (state == WebSocketConnectionState.CONNECTED |
| || state == WebSocketConnectionState.ERROR) { |
| ws.disconnect(); |
| |
| // Send DELETE to http WebSocket server. |
| sendWSSMessage("DELETE", ""); |
| |
| state = WebSocketConnectionState.CLOSED; |
| |
| // Wait for websocket close event to prevent websocket library from |
| // sending any pending messages to deleted looper thread. |
| if (waitForComplete) { |
| synchronized (closeEventLock) { |
| if (!closeEvent) { |
| try { |
| closeEventLock.wait(CLOSE_TIMEOUT); |
| } catch (InterruptedException e) { |
| Log.e(TAG, "Wait error: " + e.toString()); |
| } |
| } |
| } |
| } |
| } |
| Log.d(TAG, "Disonnecting WebSocket done."); |
| } |
| |
| private void reportError(final String errorMessage) { |
| Log.e(TAG, errorMessage); |
| executor.execute(new Runnable() { |
| @Override |
| public void run() { |
| if (state != WebSocketConnectionState.ERROR) { |
| state = WebSocketConnectionState.ERROR; |
| events.onWebSocketError(errorMessage); |
| } |
| } |
| }); |
| } |
| |
| // Asynchronously send POST/DELETE to WebSocket server. |
| private void sendWSSMessage(final String method, final String message) { |
| String postUrl = postServerUrl + "/" + roomID + "/" + clientID; |
| Log.d(TAG, "WS " + method + " : " + postUrl + " : " + message); |
| AsyncHttpURLConnection httpConnection = new AsyncHttpURLConnection( |
| method, postUrl, message, new AsyncHttpEvents() { |
| @Override |
| public void OnHttpError(String errorMessage) { |
| reportError("WS " + method + " error: " + errorMessage); |
| } |
| |
| @Override |
| public void OnHttpComplete(String response) { |
| } |
| }); |
| httpConnection.send(); |
| } |
| |
| private class WebSocketObserver implements WebSocketConnectionObserver { |
| @Override |
| public void onOpen() { |
| Log.d(TAG, "WebSocket connection opened to: " + wsServerUrl); |
| executor.execute(new Runnable() { |
| @Override |
| public void run() { |
| state = WebSocketConnectionState.CONNECTED; |
| events.onWebSocketOpen(); |
| } |
| }); |
| } |
| |
| @Override |
| public void onClose(WebSocketCloseNotification code, String reason) { |
| Log.d(TAG, "WebSocket connection closed. Code: " + code |
| + ". Reason: " + reason + ". State: " + state); |
| synchronized (closeEventLock) { |
| closeEvent = true; |
| closeEventLock.notify(); |
| } |
| executor.execute(new Runnable() { |
| @Override |
| public void run() { |
| if (state != WebSocketConnectionState.CLOSED) { |
| state = WebSocketConnectionState.CLOSED; |
| events.onWebSocketClose(); |
| } |
| } |
| }); |
| } |
| |
| @Override |
| public void onTextMessage(String payload) { |
| Log.d(TAG, "WSS->C: " + payload); |
| final String message = payload; |
| executor.execute(new Runnable() { |
| @Override |
| public void run() { |
| if (state == WebSocketConnectionState.CONNECTED |
| || state == WebSocketConnectionState.REGISTERED) { |
| events.onWebSocketMessage(message); |
| } |
| } |
| }); |
| } |
| |
| @Override |
| public void onRawTextMessage(byte[] payload) { |
| } |
| |
| @Override |
| public void onBinaryMessage(byte[] payload) { |
| } |
| } |
| |
| } |