com.openxc.interfaces.network.NetworkVehicleInterface Maven / Gradle / Ivy
package com.openxc.interfaces.network;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.InetSocketAddress;
import java.net.Socket;
import java.net.URI;
import android.content.Context;
import android.util.Log;
import com.google.common.base.Objects;
import com.openxc.interfaces.UriBasedVehicleInterfaceMixin;
import com.openxc.interfaces.VehicleInterface;
import com.openxc.remote.RawMeasurement;
import com.openxc.sources.BytestreamDataSource;
import com.openxc.sources.DataSourceException;
import com.openxc.sources.DataSourceResourceException;
import com.openxc.sources.SourceCallback;
/**
* A vehicle data source reading measurements from an OpenXC network device.
*
* This class looks for a network device and expects to read OpenXC-compatible,
* newline separated JSON messages.
*/
public class NetworkVehicleInterface extends BytestreamDataSource
implements VehicleInterface {
private static final String TAG = "NetworkVehicleInterface";
private static final int SOCKET_TIMEOUT = 10000;
private static final String SCHEMA_SPECIFIC_PREFIX = "//";
private Socket mSocket;
private InputStream mInStream;
private OutputStream mOutStream;
private URI mUri;
/**
* Construct an instance of NetworkVehicleInterface with a receiver
* callback and custom device URI.
*
* If the device cannot be found at initialization, the object will block
* waiting for a signal to check again.
*
*
* @param context
* The Activity or Service context, used to get access to the
* Android NetworkManager.
* @param callback
* An object implementing the SourceCallback that should receive
* data as it is received and parsed.
* @param uri
* The network host's address.
* @throws DataSourceException
* If no connection could be established
*/
public NetworkVehicleInterface(SourceCallback callback, Context context,
URI uri) throws DataSourceException {
super(callback, context);
setUri(uri);
start();
}
public NetworkVehicleInterface(Context context, URI uri)
throws DataSourceException {
this(null, context, uri);
}
public NetworkVehicleInterface(Context context, String uriString)
throws DataSourceException {
this(context, UriBasedVehicleInterfaceMixin.createUri(
massageUri(uriString)));
}
public boolean setResource(String otherResource) throws DataSourceException {
if(!UriBasedVehicleInterfaceMixin.sameResource(mUri,
massageUri(otherResource))) {
setUri(otherResource);
try {
if(mSocket != null) {
mSocket.close();
}
} catch(IOException e) {
}
return true;
}
return false;
}
/**
* Return true if the address and port are valid.
*
* @return true if the address and port are valid.
*/
public static boolean validateResource(String uriString) {
return UriBasedVehicleInterfaceMixin.validateResource(
massageUri(uriString));
}
@Override
public String toString() {
return Objects.toStringHelper(this)
.add("uri", mUri)
.toString();
}
public boolean receive(RawMeasurement command) {
String message = command.serialize() + "\u0000";
byte[] bytes = message.getBytes();
return write(bytes);
}
@Override
public boolean isConnected() {
mConnectionLock.readLock().lock();
boolean connected = mSocket != null && mSocket.isConnected() && super.isConnected();
mConnectionLock.readLock().unlock();
return connected;
}
protected int read(byte[] bytes) throws IOException {
mConnectionLock.readLock().lock();
int bytesRead = -1;
try {
if(isConnected()) {
bytesRead = mInStream.read(bytes, 0, bytes.length);
}
} finally {
mConnectionLock.readLock().unlock();
}
return bytesRead;
}
protected void connect() throws NetworkSourceException {
if(!isRunning()) {
return;
}
mConnectionLock.writeLock().lock();
try {
mSocket = new Socket();
mSocket.connect(new InetSocketAddress(mUri.getHost(),
mUri.getPort()), SOCKET_TIMEOUT);
if(!isConnected()) {
Log.d(TAG, "Could not connect to server");
disconnected();
} else {
connected();
connectStreams();
}
} catch(IOException e) {
String message = "Error opening streams";
Log.e(TAG, message, e);
disconnect();
throw new NetworkSourceException(message, e);
} catch(AssertionError e) {
String message = "Error opening streams";
Log.e(TAG, message, e);
disconnect();
throw new NetworkSourceException(message, e);
} finally {
mConnectionLock.writeLock().unlock();
}
}
protected void disconnect() {
if(!isConnected()) {
return;
}
mConnectionLock.writeLock().lock();
try {
Log.d(TAG, "Disconnecting from the socket " + mSocket);
try {
if(mInStream != null) {
mInStream.close();
mInStream = null;
}
} catch(IOException e) {
Log.w(TAG, "Unable to close the input stream", e);
}
try {
if(mOutStream != null) {
mOutStream.close();
mOutStream = null;
}
} catch(IOException e) {
Log.w(TAG, "Unable to close the output stream", e);
}
try {
if(mSocket != null) {
mSocket.close();
mSocket = null;
}
} catch(IOException e) {
Log.w(TAG, "Unable to close the socket", e);
}
disconnected();
} finally {
mConnectionLock.writeLock().unlock();
}
Log.d(TAG, "Disconnected from the socket");
}
/**
* Writes given data to the socket.
*
* @param bytes data to write to the socket.
* @return true if the data was written successfully.
*/
private synchronized boolean write(byte[] bytes) {
mConnectionLock.readLock().lock();
boolean success = true;
try {
if(isConnected()) {
Log.d(TAG, "Writing bytes to socket: " + bytes);
mOutStream.write(bytes);
} else {
Log.w(TAG, "No connection established, could not send anything.");
success = false;
}
} catch(IOException e) {
Log.w(TAG, "Unable to write CAN message to Network. Error: " + e.toString());
success = false;
} finally {
mConnectionLock.readLock().unlock();
}
return success;
}
protected String getTag() {
return TAG;
}
private void connectStreams() throws NetworkSourceException {
mConnectionLock.writeLock().lock();
try {
try {
mInStream = mSocket.getInputStream();
mOutStream = mSocket.getOutputStream();
Log.i(TAG, "Socket created, streams assigned");
} catch(IOException e) {
String message = "Error opening Network socket streams";
Log.e(TAG, message, e);
disconnected();
throw new NetworkSourceException(message);
}
} finally {
mConnectionLock.writeLock().unlock();
}
}
/**
* Add the prefix reuqired to parse with URI if it's not already there.
*/
private static String massageUri(String uriString) {
if(!uriString.startsWith(SCHEMA_SPECIFIC_PREFIX)) {
uriString = SCHEMA_SPECIFIC_PREFIX + uriString;
}
return uriString;
}
private void setUri(String uri) throws DataSourceException {
setUri(UriBasedVehicleInterfaceMixin.createUri(massageUri(uri)));
}
private void setUri(URI uri) throws DataSourceResourceException {
if(uri == null || !UriBasedVehicleInterfaceMixin.validateResource(uri)) {
throw new DataSourceResourceException("URI is not valid");
}
mUri = uri;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy