diff --git a/src/main/java/org/myrobotlab/codec/CodecUtils.java b/src/main/java/org/myrobotlab/codec/CodecUtils.java index cb7698d26a..457f3a8fe5 100644 --- a/src/main/java/org/myrobotlab/codec/CodecUtils.java +++ b/src/main/java/org/myrobotlab/codec/CodecUtils.java @@ -38,6 +38,7 @@ import org.myrobotlab.framework.MethodCache; import org.myrobotlab.framework.Platform; import org.myrobotlab.framework.StaticType; +import org.myrobotlab.framework.interfaces.ServiceInterface; import org.myrobotlab.logging.Level; import org.myrobotlab.logging.LoggerFactory; import org.myrobotlab.logging.LoggingFactory; @@ -488,7 +489,6 @@ public static String getFullName(String name) { if (name == null) { return null; } - if (getId(name) == null) { return name + '@' + Platform.getLocalInstance().getId(); } else { @@ -499,7 +499,7 @@ public static String getFullName(String name) { /** * Checks whether two service names are equal by first normalizing each. If a * name does not have a runtime ID, it is assumed to be a local service. - * + * * @param name1 * The first service name * @param name2 @@ -510,6 +510,7 @@ public static boolean checkServiceNameEquality(String name1, String name2) { return Objects.equals(getFullName(name1), getFullName(name2)); } + /** * Converts a topic method name to the name of the method that is used for * callbacks. Usually this involves prepending the string "on", removing any @@ -674,6 +675,14 @@ static public String getServiceType(String inType) { // falls back to virt class field boolean useVirtClassField = clazz == null; if (!useVirtClassField) { + // For blocking send return type checking + ServiceInterface si = Runtime.getService(serviceName); + String blockingKey = String.format("%s.%s", msg.getFullName(), msg.getMethod()); + if (si != null && Message.MSG_TYPE_RETURN.equals(msg.msgType) && si.getInbox().blockingList.containsKey(blockingKey)) { + msg.data[0] = fromJson((String) msg.data[0], si.getInbox().blockingList.get(blockingKey).second); + msg.encoding = null; + return msg; + } try { Object[] params = MethodCache.getInstance().getDecodedJsonParameters(clazz, msg.method, msg.data); if (params == null) diff --git a/src/main/java/org/myrobotlab/framework/Inbox.java b/src/main/java/org/myrobotlab/framework/Inbox.java index 1bf23c387d..40f5bf271f 100644 --- a/src/main/java/org/myrobotlab/framework/Inbox.java +++ b/src/main/java/org/myrobotlab/framework/Inbox.java @@ -30,9 +30,11 @@ import java.util.HashMap; import java.util.LinkedList; import java.util.List; +import java.util.Map; import org.myrobotlab.framework.interfaces.MessageListener; import org.myrobotlab.logging.LoggerFactory; +import org.myrobotlab.utils.ObjectTypePair; import org.slf4j.Logger; public class Inbox implements Serializable { @@ -50,9 +52,16 @@ public class Inbox implements Serializable { // value // support remote blocking... in-process blocking uses invoke - public HashMap blockingList = new HashMap<>(); + /** + * Maps blocking keys ("fullName.method") to an + * ObjectTypePair that is used to store the return + * type and return value. DO NOT SET THE VALUE + * TO A NEW OBJECT, we are synchronizing on the values, + * doing so will result in the threads never waking up. + */ + public Map> blockingList = new HashMap<>(); - List listeners = new ArrayList(); + List listeners = new ArrayList<>(); public Inbox() { this("Inbox"); @@ -62,6 +71,7 @@ public Inbox(String name) { this.name = name; } + @SuppressWarnings("unchecked") public void add(Message msg) { /** *
@@ -74,16 +84,17 @@ public void add(Message msg) {
      */
 
     // --- sendBlocking support begin --------------------
-    // TODO - possible safety check msg.status == Message.RETURN
-    // &&
     String blockingKey = String.format("%s.%s", msg.getFullName(), msg.getMethod());
     if (blockingList.containsKey(blockingKey)) {
-      Object[] returnContainer = blockingList.get(blockingKey);
+      // Generates an unchecked warning because of generic invariance, but we don't care
+      ObjectTypePair returnContainer = (ObjectTypePair) blockingList.get(blockingKey);
       if (msg.data == null) {
-        returnContainer[0] = null;
+        returnContainer.first = null;
       } else {
         // transferring data
-        returnContainer[0] = msg.data[0];
+        // Would not be able to do this if returnContainer had wildcard generic type,
+        // which is why it's Object
+        returnContainer.first = msg.data[0];
       }
 
       synchronized (returnContainer) {
diff --git a/src/main/java/org/myrobotlab/framework/Message.java b/src/main/java/org/myrobotlab/framework/Message.java
index 5a64f5a0e1..450c48dc73 100644
--- a/src/main/java/org/myrobotlab/framework/Message.java
+++ b/src/main/java/org/myrobotlab/framework/Message.java
@@ -47,6 +47,23 @@ public class Message implements Serializable {
 
   private static final long serialVersionUID = 1L;
 
+  /**
+   * A message has this type if it is sent with the expectation that a return
+   * message will be received. Services should always respond
+   * with a return message after invoking the blocking message.
+   *
+   * @see #MSG_TYPE_RETURN
+   */
+  public static final String MSG_TYPE_BLOCKING = "BLOCKING";
+
+  /**
+   * A message has this type when it contains a single data
+   * element meant to represent a service method return value.
+   * The type of the return value is not explicitly carried with the message,
+   * since it can cross language boundaries.
+   */
+  public static final String MSG_TYPE_RETURN = "RETURN";
+
   // FIXME msgId should be a String encoded value of src and an atomic increment
   // ROS comes with a seq Id, a timestamp, and a frame Id
   /**
@@ -74,8 +91,7 @@ public class Message implements Serializable {
    * history of the message, its routing stops and Services it passed through.
    * This is important to prevent endless looping of messages. Turns out
    * ArrayList is quicker than HashSet on small sets
-   * http://www.javacodegeeks.com
-   * /2010/08/java-best-practices-vector-arraylist.html
+   * Java Best Practices
    */
   protected List historyList;
 
@@ -88,11 +104,12 @@ public class Message implements Serializable {
 
   /**
    * status is currently used for BLOCKING message calls the current valid state
-   * it can be in is null | BLOCKING | RETURN FIXME - this should be msgType not
-   * status
+   * it can be in is null | BLOCKING | RETURN
+   *
+   * @see #MSG_TYPE_BLOCKING
+   * @see #MSG_TYPE_RETURN
    */
-
-  public String status;
+  public String msgType;
 
   public String encoding; // null == none |json|cli|xml|stream ...
 
@@ -151,7 +168,7 @@ final public void set(final Message other) {
     historyList = new ArrayList();
     historyList.addAll(other.historyList);
 
-    status = other.status;
+    msgType = other.msgType;
     encoding = other.encoding;
     method = other.method;
     // you know the dangers of reference copy
@@ -299,7 +316,7 @@ public boolean equals(Object o) {
             && Objects.equals(sendingMethod, message.sendingMethod)
             && Objects.equals(historyList, message.historyList)
             && Objects.equals(properties, message.properties)
-            && Objects.equals(status, message.status)
+            && Objects.equals(msgType, message.msgType)
             && Objects.equals(encoding, message.encoding)
             && Objects.equals(method, message.method)
             && Arrays.deepEquals(data, message.data);
@@ -310,7 +327,7 @@ public int hashCode() {
     int result = Objects.hash(
                     msgId, name, sender,
                     sendingMethod, historyList,
-                    properties, status, encoding,
+                    properties, msgType, encoding,
                     method
     );
     result = 31 * result + Arrays.hashCode(data);
diff --git a/src/main/java/org/myrobotlab/framework/Outbox.java b/src/main/java/org/myrobotlab/framework/Outbox.java
index 01921add9f..ac1e50b781 100644
--- a/src/main/java/org/myrobotlab/framework/Outbox.java
+++ b/src/main/java/org/myrobotlab/framework/Outbox.java
@@ -212,8 +212,7 @@ public void run() {
           continue;
         }
 
-        for (int i = 0; i < subList.size(); ++i) {
-          MRLListener listener = subList.get(i);
+        for (MRLListener listener : subList) {
           msg.setName(listener.callbackName);
           msg.method = listener.callbackMethod;
           
diff --git a/src/main/java/org/myrobotlab/framework/Service.java b/src/main/java/org/myrobotlab/framework/Service.java
index 08534f2081..41e7cdbe85 100644
--- a/src/main/java/org/myrobotlab/framework/Service.java
+++ b/src/main/java/org/myrobotlab/framework/Service.java
@@ -73,6 +73,7 @@
 import org.myrobotlab.service.interfaces.AuthorizationProvider;
 import org.myrobotlab.service.interfaces.QueueReporter;
 import org.myrobotlab.service.meta.abstracts.MetaData;
+import org.myrobotlab.utils.ObjectTypePair;
 import org.myrobotlab.string.StringUtil;
 import org.slf4j.Logger;
 
@@ -750,8 +751,7 @@ public void addListener(String topicMethod, String callbackName, String callback
       // iterate through all looking for duplicate
       boolean found = false;
       List nes = outbox.notifyList.get(listener.topicMethod);
-      for (int i = 0; i < nes.size(); ++i) {
-        MRLListener entry = nes.get(i);
+      for (MRLListener entry : nes) {
         if (entry.equals(listener)) {
           log.debug("attempting to add duplicate MRLListener {}", listener);
           found = true;
@@ -1073,7 +1073,7 @@ public List getNotifyList(String key) {
       // use the runtime to send a message
       // FIXME - parameters !
       try {
-        return (ArrayList) Runtime.getInstance().sendBlocking(getName(), "getNotifyList", new Object[] { key });
+        return Runtime.getInstance().sendBlocking(getName(), "getNotifyList", new StaticType<>(){}, new Object[] { key });
       } catch (Exception e) {
         log.error("remote getNotifyList threw", e);
         return null;
@@ -1093,7 +1093,8 @@ public ArrayList getNotifyListKeySet() {
       // use the runtime to send a message
 
       try {
-        return (ArrayList) Runtime.getInstance().sendBlocking(getFullName(), "getNotifyListKeySet");
+        return Runtime.getInstance().sendBlocking(getName(), "getNotifyListKeySet", new StaticType<>(){});
+
       } catch (Exception e) {
         log.error("remote getNotifyList threw", e);
         return null;
@@ -1194,9 +1195,10 @@ public void in(Message msg) {
   /**
    * This is where all messages are routed to and processed
    */
+  @SuppressWarnings("unchecked")
   @Override
-  final public Object invoke(Message msg) {
-    Object retobj = null;
+  final public  R invoke(Message msg, StaticType returnType) {
+    R retobj;
 
     if (log.isDebugEnabled()) {
       log.debug("--invoking {}.{}({}) {} --", name, msg.method, CodecUtils.getParameterSignature(msg.data), msg.msgId);
@@ -1211,17 +1213,17 @@ final public Object invoke(Message msg) {
     // happen in other situations...
     if (Runtime.getInstance().isLocal(msg) && !name.equals(msg.getName())) {
       // wrong Service - get the correct one
-      return Runtime.getService(msg.getName()).invoke(msg);
+      return Runtime.getService(msg.getName()).invoke(msg, returnType);
     }
 
     String blockingKey = String.format("%s.%s", msg.getFullName(), msg.getMethod());
     if (inbox.blockingList.containsKey(blockingKey)) {
-      Object[] returnContainer = inbox.blockingList.get(blockingKey);
+      ObjectTypePair returnContainer = (ObjectTypePair) inbox.blockingList.get(blockingKey);
       if (msg.getData() == null) {
-        returnContainer[0] = null;
+        returnContainer.first = null;
       } else {
         // transferring data
-        returnContainer[0] = msg.getData()[0];
+        returnContainer.first = (R) msg.getData()[0];
       }
 
       synchronized (returnContainer) {
@@ -1232,19 +1234,19 @@ final public Object invoke(Message msg) {
       return null;
     }
 
-    retobj = invokeOn(false, this, msg.method, msg.data);
+    retobj = invokeOn(false, this, msg.method, returnType, msg.data);
 
     return retobj;
   }
 
   @Override
-  final public Object invoke(String method) {
-    return invokeOn(false, this, method, (Object[]) null);
+  final public  R invoke(String method, StaticType returnType) {
+    return invokeOn(false, this, method, returnType, (Object[]) null);
   }
 
   @Override
-  final public Object invoke(String method, Object... params) {
-    return invokeOn(false, this, method, params);
+  final public  R invoke(String method, StaticType returnType, Object... params) {
+    return invokeOn(false, this, method, returnType, params);
   }
 
   /**
@@ -1281,8 +1283,8 @@ final public Object broadcast(String method, Object... params) {
    * @return the returned value from invoking
    * 
    */
-  final public Object invokeOn(String serviceName, String methodName, Object... params) {
-    return invokeOn(false, Runtime.getService(serviceName), methodName, params);
+  final public  R invokeOn(String serviceName, String methodName, StaticType returnType, Object... params) {
+    return invokeOn(false, Runtime.getService(serviceName), methodName, returnType, params);
   }
 
   /**
@@ -1297,8 +1299,8 @@ final public Object invokeOn(String serviceName, String methodName, Object... pa
    * @return return object
    */
   @Override
-  final public Object invokeOn(boolean blockLocally, Object obj, String methodName, Object... params) {
-    Object retobj = null;
+  final public  R invokeOn(boolean blockLocally, Object obj, String methodName, StaticType returnType, Object... params) {
+    R retobj = null;
     try {
       MethodCache cache = MethodCache.getInstance();
       if (obj == null) {
@@ -1310,7 +1312,7 @@ final public Object invokeOn(boolean blockLocally, Object obj, String methodName
         error("could not find method %s.%s(%s)", obj.getClass().getSimpleName(), methodName, MethodCache.formatParams(params));
         return null; // should this be allowed to throw to a higher level ?
       }
-      retobj = method.invoke(obj, params);
+      retobj = (R) method.invoke(obj, params);
       if (blockLocally) {
         Outbox outbox = null;
         if (obj instanceof ServiceInterface) {
@@ -1326,6 +1328,7 @@ final public Object invokeOn(boolean blockLocally, Object obj, String methodName
         if (subList != null) {
           for (MRLListener listener : subList) {
             Message msg = Message.createMessage(getFullName(), listener.callbackName, listener.callbackMethod, retobj);
+            msg.msgType = Message.MSG_TYPE_RETURN;
             msg.sendingMethod = methodName;
             if (runtime.isLocal(msg)) {
               ServiceInterface si = Runtime.getService(listener.callbackName);
@@ -1739,11 +1742,12 @@ public void sendAsync(String name, String method, Object... data) {
   }
 
   @Override
-  public Object sendBlocking(String name, Integer timeout, String method, Object... data) throws InterruptedException, TimeoutException {
-    Message msg = Message.createMessage(getFullName(), name, method, data);
+  public  R sendBlocking(String name, Integer timeout, String method, StaticType returnType, Object... data) throws InterruptedException, TimeoutException {
+    Message msg = Message.createMessage(getName(), name, method, data);
+    msg.sender = this.getFullName();
     msg.msgId = Runtime.getUniqueID();
 
-    return sendBlocking(msg, timeout);
+    return sendBlocking(msg, timeout, returnType);
   }
 
   /**
@@ -1760,11 +1764,12 @@ public Object sendBlocking(String name, Integer timeout, String method, Object..
    * 
    */
   @Override
-  public Object sendBlocking(Message msg, Integer timeout) throws InterruptedException, TimeoutException {
+  public  R sendBlocking(Message msg, Integer timeout, StaticType returnType) throws InterruptedException, TimeoutException {
+    msg.msgType = Message.MSG_TYPE_BLOCKING;
     if (Runtime.getInstance().isLocal(msg)) {
-      return invoke(msg);
+      return invoke(msg, returnType);
     } else {
-      return waitOn(msg.getFullName(), msg.getMethod(), timeout, msg);
+      return waitOn(msg.getFullName(), msg.getMethod(), timeout, msg, returnType);
     }
   }
 
@@ -1791,8 +1796,8 @@ public Object sendBlocking(Message msg, Integer timeout) throws InterruptedExcep
    * @throws TimeoutException
    *           boom
    */
-  protected Object waitOn(String fullName, String method, Integer timeout, Message sendMsg) throws InterruptedException, TimeoutException {
-
+  @SuppressWarnings("unchecked")
+  protected  R waitOn(String fullName, String method, Integer timeout, Message sendMsg, StaticType returnType) throws InterruptedException, TimeoutException {
     String subscriber = null;
     if (sendMsg != null) {
       // InProcCli proxies - so the subscription needs to be from the sender NOT
@@ -1805,14 +1810,14 @@ protected Object waitOn(String fullName, String method, Integer timeout, Message
     // put in-process lock in map
     String callbackMethod = CodecUtils.getCallbackTopicName(method);
     String blockingKey = String.format("%s.%s", subscriber, callbackMethod);
-    Object[] blockingLockContainer = null;
+    ObjectTypePair blockingLockContainer;
     if (!inbox.blockingList.containsKey(blockingKey)) {
-      blockingLockContainer = new Object[1];
+      blockingLockContainer = new ObjectTypePair<>(null, returnType);
       inbox.blockingList.put(blockingKey, blockingLockContainer);
     } else {
       // if it already exists - other threads are already waiting for the
       // same callback ...
-      blockingLockContainer = inbox.blockingList.get(blockingKey);
+      blockingLockContainer = (ObjectTypePair) inbox.blockingList.get(blockingKey);
     }
 
     // send subscription
@@ -1845,26 +1850,32 @@ public void run() {
     // cleanup
     unsubscribe(fullName, method, subscriber, CodecUtils.getCallbackTopicName(method));
 
-    return blockingLockContainer[0];
+    if (returnType.equals(blockingLockContainer.second)) {
+      // Unchecked cast is fine because the deserializer ensures second is assignable to R
+      return blockingLockContainer.first;
+    } else {
+      error("Return type was changed during blocking call. Was {}, now {}", returnType, blockingLockContainer.second);
+      return null;
+    }
 
   }
 
   // equivalent to sendBlocking without the sending a message
   @Override
-  public Object waitFor(String fullName, String method, Integer timeout) throws InterruptedException, TimeoutException {
-    return waitOn(fullName, method, timeout, null);
+  public  R waitFor(String fullName, String method, Integer timeout, StaticType returnType) throws InterruptedException, TimeoutException {
+    return waitOn(fullName, method, timeout, null, returnType);
   }
 
   // BOXING - End --------------------------------------
   @Override
-  public Object sendBlocking(String name, String method) throws InterruptedException, TimeoutException {
-    return sendBlocking(name, method, (Object[]) null);
+  public  R sendBlocking(String name, String method, StaticType returnType) throws InterruptedException, TimeoutException {
+    return sendBlocking(name, method, returnType, (Object[]) null);
   }
 
   @Override
-  public Object sendBlocking(String name, String method, Object... data) throws InterruptedException, TimeoutException {
+  public  R sendBlocking(String name, String method, StaticType returnType, Object... data) throws InterruptedException, TimeoutException {
     // default 1 second timeout - FIXME CONFIGURABLE
-    return sendBlocking(name, 1000, method, data);
+    return sendBlocking(name, 1000, method, returnType, data);
   }
 
   @Override
diff --git a/src/main/java/org/myrobotlab/framework/StaticType.java b/src/main/java/org/myrobotlab/framework/StaticType.java
index fddc2912f9..b1969db08d 100644
--- a/src/main/java/org/myrobotlab/framework/StaticType.java
+++ b/src/main/java/org/myrobotlab/framework/StaticType.java
@@ -70,6 +70,11 @@ protected StaticType() {
 
     }
 
+    private StaticType(Type storedType) {
+        this.storedType = storedType;
+        validateType(storedType);
+    }
+
     /**
      * Gets the stored {@link Type}
      * instance. This type should contain the type of
@@ -111,4 +116,8 @@ private static void validateType(Type type) {
         }
 
     }
+
+    public static  StaticType fromJavaType(Type type) {
+        return new StaticType<>(type){};
+    }
 }
diff --git a/src/main/java/org/myrobotlab/framework/interfaces/Invoker.java b/src/main/java/org/myrobotlab/framework/interfaces/Invoker.java
index f3524ac6e4..e17d71d7f1 100644
--- a/src/main/java/org/myrobotlab/framework/interfaces/Invoker.java
+++ b/src/main/java/org/myrobotlab/framework/interfaces/Invoker.java
@@ -1,9 +1,21 @@
 package org.myrobotlab.framework.interfaces;
 
+import org.myrobotlab.framework.StaticType;
+
 public interface Invoker {
 
-  public Object invoke(String method);
+  default Object invoke(String method) {
+    return invoke(method, new StaticType<>(){});
+  }
+
+  default  R invoke(String method, StaticType returnType) {
+    return invoke(method, returnType, new Object[0]);
+  }
+
 
-  public Object invoke(String method, Object... params);
+  default Object invoke(String method, Object... params) {
+    return invoke(method, new StaticType<>() {}, params);
+  }
+   R invoke(String method, StaticType returnType, Object... params);
 
 }
diff --git a/src/main/java/org/myrobotlab/framework/interfaces/MessageInvoker.java b/src/main/java/org/myrobotlab/framework/interfaces/MessageInvoker.java
index 9c80862978..f5f80bd392 100644
--- a/src/main/java/org/myrobotlab/framework/interfaces/MessageInvoker.java
+++ b/src/main/java/org/myrobotlab/framework/interfaces/MessageInvoker.java
@@ -1,6 +1,7 @@
 package org.myrobotlab.framework.interfaces;
 
 import org.myrobotlab.framework.Message;
+import org.myrobotlab.framework.StaticType;
 
 public interface MessageInvoker {
   /**
@@ -10,7 +11,11 @@ public interface MessageInvoker {
    * @param msg
    * @return
    */
-  public Object invoke(Message msg);
+  default Object invoke(Message msg) {
+    return invoke(msg, new StaticType<>(){});
+  }
+
+   R invoke(Message msg, StaticType returnType);
 
   /**
    * Invoke a method on a service with params.
@@ -21,6 +26,10 @@ public interface MessageInvoker {
    * @param params - the parameters to pass to the method
    * @return - the return value of the method
    */
-  public Object invokeOn(boolean blockLocally, Object obj, String method, Object... params);
+  default Object invokeOn(boolean blockLocally, Object obj, String method, Object... params) {
+    return invokeOn(blockLocally, obj, method, new StaticType<>(){}, params);
+  }
+
+   R invokeOn(boolean blockLocally, Object obj, String method, StaticType returnType, Object... params);
 
 }
diff --git a/src/main/java/org/myrobotlab/framework/interfaces/MessageSender.java b/src/main/java/org/myrobotlab/framework/interfaces/MessageSender.java
index ba0f5f1f19..4da259fe24 100644
--- a/src/main/java/org/myrobotlab/framework/interfaces/MessageSender.java
+++ b/src/main/java/org/myrobotlab/framework/interfaces/MessageSender.java
@@ -1,6 +1,7 @@
 package org.myrobotlab.framework.interfaces;
 
 import org.myrobotlab.framework.Message;
+import org.myrobotlab.framework.StaticType;
 import org.myrobotlab.framework.TimeoutException;
 
 public interface MessageSender extends NameProvider {
@@ -14,7 +15,7 @@ public interface MessageSender extends NameProvider {
    * @param method
    *          - method of destination service
    */
-  public void send(String name, String method);
+  void send(String name, String method);
 
   /**
    * Send invoking messages to remote location to invoke {name} instance's
@@ -27,7 +28,7 @@ public interface MessageSender extends NameProvider {
    * @param data
    *          - parameter data
    */
-  public void send(String name, String method, Object... data);
+  void send(String name, String method, Object... data);
 
   /**
    * Base method for sending messages.
@@ -35,16 +36,36 @@ public interface MessageSender extends NameProvider {
    * @param msg
    *          - message to be sent
    */
-  public void send(Message msg);
+  void send(Message msg);
 
-  public Object sendBlocking(String name, String method) throws InterruptedException, TimeoutException;
+  default Object sendBlocking(String name, String method) throws InterruptedException, TimeoutException {
+    return sendBlocking(name, method, new StaticType<>() {});
+  }
 
-  public Object sendBlocking(String name, String method, Object... data) throws InterruptedException, TimeoutException;
+   R sendBlocking(String name, String method, StaticType returnType) throws InterruptedException, TimeoutException;
 
-  public Object sendBlocking(String name, Integer timeout, String method, Object... data) throws InterruptedException, TimeoutException;
+  default Object sendBlocking(String name, String method, Object... data) throws InterruptedException, TimeoutException {
+    return sendBlocking(name, method, new StaticType<>(){}, data);
+  }
 
-  public Object sendBlocking(Message msg, Integer timeout) throws InterruptedException, TimeoutException;
+   R sendBlocking(String name, String method, StaticType returnType, Object... data) throws InterruptedException, TimeoutException;
 
-  public Object waitFor(String fullName, String method, Integer timeout) throws InterruptedException, TimeoutException;
+  default Object sendBlocking(String name, Integer timeout, String method, Object... data) throws InterruptedException, TimeoutException {
+    return sendBlocking(name, timeout, method, new StaticType<>(){}, data);
+  }
+
+   R sendBlocking(String name, Integer timeout, String method, StaticType returnType, Object... data) throws InterruptedException, TimeoutException;
+
+  default Object sendBlocking(Message msg, Integer timeout) throws InterruptedException, TimeoutException {
+    return sendBlocking(msg, timeout, new StaticType<>(){});
+  }
+
+   R sendBlocking(Message msg, Integer timeout, StaticType returnType) throws InterruptedException, TimeoutException;
+
+  default Object waitFor(String fullName, String method, Integer timeout) throws InterruptedException, TimeoutException {
+    return waitFor(fullName, method, timeout, new StaticType<>() {});
+  }
+
+   R waitFor(String fullName, String method, Integer timeout, StaticType returnType) throws InterruptedException, TimeoutException;
 
 }
diff --git a/src/main/java/org/myrobotlab/framework/interfaces/ServiceQueue.java b/src/main/java/org/myrobotlab/framework/interfaces/ServiceQueue.java
index b2c85cc97b..e83abc1354 100644
--- a/src/main/java/org/myrobotlab/framework/interfaces/ServiceQueue.java
+++ b/src/main/java/org/myrobotlab/framework/interfaces/ServiceQueue.java
@@ -19,11 +19,6 @@ public interface ServiceQueue {
 
   public void out(Message msg);
 
-  // TODO - put in seperate Invoking interface
-  public Object invoke(String method);
-
-  public Object invoke(String method, Object... params);
-
   // public boolean isLocal();
 
 }
diff --git a/src/main/java/org/myrobotlab/service/Solr.java b/src/main/java/org/myrobotlab/service/Solr.java
index 33a81a21ee..a388b034a8 100644
--- a/src/main/java/org/myrobotlab/service/Solr.java
+++ b/src/main/java/org/myrobotlab/service/Solr.java
@@ -1008,7 +1008,7 @@ public void onMessage(Message message) {
     doc.setField("message_dataEncoding", message.encoding);
     doc.setField("message_name", message.getName());
     doc.setField("sender_method", message.sendingMethod);
-    doc.setField("message_status", message.status);
+    doc.setField("message_status", message.msgType);
     /*
      * This makes no sense.. if (message.getHops() != null) { for (String
      * history : message.getHops()) { doc.addField("history",
diff --git a/src/main/java/org/myrobotlab/service/interfaces/LogPublisher.java b/src/main/java/org/myrobotlab/service/interfaces/LogPublisher.java
index bba0b85e10..45ef8e3c98 100644
--- a/src/main/java/org/myrobotlab/service/interfaces/LogPublisher.java
+++ b/src/main/java/org/myrobotlab/service/interfaces/LogPublisher.java
@@ -1,5 +1,6 @@
 package org.myrobotlab.service.interfaces;
 
+import org.myrobotlab.framework.interfaces.Invoker;
 import org.myrobotlab.framework.interfaces.NameProvider;
 
 /**
@@ -9,7 +10,7 @@
  *         A LogPublisher can publish its own logging messages
  * 
  */
-public interface LogPublisher extends NameProvider {
+public interface LogPublisher extends NameProvider, Invoker {
 
   /**
    * A String is currently used as the log entry - but it could be an object in
@@ -21,17 +22,5 @@ public interface LogPublisher extends NameProvider {
    *          msg to publish
    * @return string
    */
-  public String publishLog(String msg);
-
-  /**
-   * a way to publish the log messages and log entries
-   * 
-   * @param method
-   *          method
-   * @param params
-   *          params
-   * @return returned object
-   * 
-   */
-  public Object invoke(String method, Object... params);
+  String publishLog(String msg);
 }
diff --git a/src/main/java/org/myrobotlab/service/interfaces/MrlCommPublisher.java b/src/main/java/org/myrobotlab/service/interfaces/MrlCommPublisher.java
index 563f1e007a..3769229ed7 100755
--- a/src/main/java/org/myrobotlab/service/interfaces/MrlCommPublisher.java
+++ b/src/main/java/org/myrobotlab/service/interfaces/MrlCommPublisher.java
@@ -1,6 +1,7 @@
 package org.myrobotlab.service.interfaces;
 
 import org.myrobotlab.arduino.BoardInfo;
+import org.myrobotlab.framework.interfaces.Invoker;
 import org.myrobotlab.sensor.EncoderData;
 import org.myrobotlab.service.data.PinData;
 import org.myrobotlab.service.data.SerialRelayData;
@@ -12,7 +13,7 @@
  * @author kwatters
  *
  */
-public interface MrlCommPublisher {
+public interface MrlCommPublisher extends Invoker {
 
   public void onBytes(byte[] data);
 
@@ -48,8 +49,5 @@ public BoardInfo publishBoardInfo(Integer version/* byte */,
 
   public void ackTimeout();
 
-  // Necessary evil so Msg.java can invoke the publish methods on the publisher
-  // service.
-  public Object invoke(String method, Object... params);
 
 }
diff --git a/src/main/java/org/myrobotlab/utils/ObjectTypePair.java b/src/main/java/org/myrobotlab/utils/ObjectTypePair.java
new file mode 100644
index 0000000000..640940f967
--- /dev/null
+++ b/src/main/java/org/myrobotlab/utils/ObjectTypePair.java
@@ -0,0 +1,17 @@
+package org.myrobotlab.utils;
+
+import org.myrobotlab.framework.StaticType;
+
+/**
+ * A container class that holds an object with its
+ * associated type information. This makes
+ * constraining objects and types while inside of collections
+ * or other containers easier.
+ * @param  The type of the object contained
+ * @author AutonomicPerfectionist
+ */
+public class ObjectTypePair extends Pair> {
+    public ObjectTypePair(T first, StaticType second) {
+        super(first, second);
+    }
+}
diff --git a/src/main/java/org/myrobotlab/utils/Pair.java b/src/main/java/org/myrobotlab/utils/Pair.java
new file mode 100644
index 0000000000..9aeca2e80c
--- /dev/null
+++ b/src/main/java/org/myrobotlab/utils/Pair.java
@@ -0,0 +1,23 @@
+package org.myrobotlab.utils;
+
+
+/**
+ * A simple container for two objects
+ * of potentially different types. Allows
+ * generics to be leveraged for type constraining
+ * or more compiler-friendly collections operations.
+ *
+ * @param  The type of {@link #first}
+ * @param  The type of {@link #second}
+ * @author AutonomicPerfectionist
+ */
+public class Pair {
+    public A first;
+    public B second;
+
+    public Pair(A first, B second) {
+        this.first = first;
+        this.second = second;
+    }
+
+}
diff --git a/src/test/java/org/myrobotlab/arduino/MrlCommDirectTest.java b/src/test/java/org/myrobotlab/arduino/MrlCommDirectTest.java
index 7bb84fdc91..6443b08d75 100755
--- a/src/test/java/org/myrobotlab/arduino/MrlCommDirectTest.java
+++ b/src/test/java/org/myrobotlab/arduino/MrlCommDirectTest.java
@@ -7,6 +7,7 @@
 
 import org.junit.Test;
 import org.myrobotlab.framework.QueueStats;
+import org.myrobotlab.framework.StaticType;
 import org.myrobotlab.logging.LoggerFactory;
 import org.myrobotlab.sensor.EncoderData;
 import org.myrobotlab.serial.PortJSSC;
@@ -361,7 +362,7 @@ public void ackTimeout() {
   }
 
   @Override
-  public Object invoke(String method, Object... params) {
+  public  R invoke(String method, StaticType returnType, Object... params) {
     log.warn("Dont invoke in a unit test!!!!!!!!!!!!!!!!!!!!!!");
     return null;
   }
diff --git a/src/test/java/org/myrobotlab/codec/CodecUtilsTest.java b/src/test/java/org/myrobotlab/codec/CodecUtilsTest.java
index 7ffd26fcd4..064c8533fc 100644
--- a/src/test/java/org/myrobotlab/codec/CodecUtilsTest.java
+++ b/src/test/java/org/myrobotlab/codec/CodecUtilsTest.java
@@ -1,21 +1,27 @@
 package org.myrobotlab.codec;
 
-import static org.junit.Assert.assertEquals;
-import static org.junit.Assert.assertFalse;
-import static org.junit.Assert.assertTrue;
-
-import java.util.ArrayList;
-import java.util.LinkedHashMap;
-import java.util.List;
-import java.util.Map;
-
 import org.bouncycastle.util.Strings;
 import org.junit.Ignore;
 import org.junit.Test;
+import org.myrobotlab.codec.json.JsonDeserializationException;
+import org.myrobotlab.framework.Message;
 import org.myrobotlab.framework.Platform;
+import org.myrobotlab.framework.StaticType;
+import org.myrobotlab.service.Runtime;
 import org.myrobotlab.service.data.Locale;
 import org.myrobotlab.service.data.Orientation;
 import org.myrobotlab.test.AbstractTest;
+import org.myrobotlab.utils.ObjectTypePair;
+
+import java.util.ArrayList;
+import java.util.LinkedHashMap;
+import java.util.List;
+import java.util.Map;
+
+import static org.junit.Assert.assertEquals;
+import static org.junit.Assert.assertFalse;
+import static org.junit.Assert.assertThrows;
+import static org.junit.Assert.assertTrue;
 
 public class CodecUtilsTest extends AbstractTest {
 
@@ -187,6 +193,47 @@ public void testDefaultSerialization() {
   }
 
   @Test
+  public void returnMessageTestSimpleTypes() {
+    String retVal = "retVal";
+    // Put directly in blocking list because sendBlocking() won't use it for local services
+    Runtime.getInstance().getInbox().blockingList.put(
+            "runtime.onBlocking",
+            new ObjectTypePair<>(null, new StaticType(){})
+    );
+
+    Message returnMsg = Message.createMessage("test", "runtime", "onBlocking", new Object[]{CodecUtils.toJson(retVal)});
+    returnMsg.msgType = Message.MSG_TYPE_RETURN;
+    returnMsg.encoding = "json";
+    Message deserMessage = CodecUtils.decodeMessageParams(new Message(returnMsg));
+
+    assertEquals(retVal, deserMessage.data[0]);
+
+    // This should now throw, notice the Integer type instead of String
+    Runtime.getInstance().getInbox().blockingList.put(
+            "runtime.onBlocking",
+            new ObjectTypePair<>(null, new StaticType(){})
+    );
+    returnMsg = Message.createMessage("test", "runtime", "onBlocking", new Object[]{CodecUtils.toJson(retVal)});
+    returnMsg.msgType = Message.MSG_TYPE_RETURN;
+    returnMsg.encoding = "json";
+    Message finalReturnMsg = returnMsg;
+    assertThrows(JsonDeserializationException.class, () -> CodecUtils.decodeMessageParams(finalReturnMsg));
+
+    Runtime.getInstance().getInbox().blockingList.put(
+            "runtime.onBlocking",
+            new ObjectTypePair<>(null, new StaticType(){})
+    );
+    returnMsg = Message.createMessage("test", "runtime", "onBlocking", new Object[]{CodecUtils.toJson(retVal)});
+    // Setting msgType to null makes CodecUtils ignore blockingList
+    returnMsg.msgType = null;
+    returnMsg.encoding = "json";
+    deserMessage = CodecUtils.decodeMessageParams(new Message(returnMsg));
+
+    assertEquals(retVal, deserMessage.data[0]);
+
+
+  }
+
   public void testNormalizeServiceName() {
     Platform.getLocalInstance().setId("test-id");
     assertEquals("runtime@test-id", CodecUtils.getFullName("runtime"));
diff --git a/src/test/java/org/myrobotlab/service/PythonTest.java b/src/test/java/org/myrobotlab/service/PythonTest.java
index 97a3f2fe0c..2b9d3dadf6 100644
--- a/src/test/java/org/myrobotlab/service/PythonTest.java
+++ b/src/test/java/org/myrobotlab/service/PythonTest.java
@@ -5,6 +5,7 @@
 
 import java.util.Map;
 
+import org.junit.Test;
 import org.myrobotlab.framework.Service;
 import org.python.core.PyInteger;
 
diff --git a/src/test/java/org/myrobotlab/service/VirtualArduinoTest.java b/src/test/java/org/myrobotlab/service/VirtualArduinoTest.java
index bac8388134..aa47a198e0 100755
--- a/src/test/java/org/myrobotlab/service/VirtualArduinoTest.java
+++ b/src/test/java/org/myrobotlab/service/VirtualArduinoTest.java
@@ -11,6 +11,7 @@
 import org.myrobotlab.arduino.virtual.MrlServo;
 import org.myrobotlab.framework.QueueStats;
 import org.myrobotlab.framework.Service;
+import org.myrobotlab.framework.StaticType;
 import org.myrobotlab.sensor.EncoderData;
 import org.myrobotlab.service.data.PinData;
 import org.myrobotlab.service.data.SerialRelayData;
@@ -218,7 +219,7 @@ public void ackTimeout() {
   }
 
   @Override
-  public Object invoke(String method, Object... params) {
+  public  R invoke(String method, StaticType returnType, Object... params) {
     log.warn("Don't invoke in a unit test!!!!!!");
     return null;
   }
diff --git a/src/test/java/org/myrobotlab/service/WebGuiTest.java b/src/test/java/org/myrobotlab/service/WebGuiTest.java
index 643485fc0c..53a76f85cc 100644
--- a/src/test/java/org/myrobotlab/service/WebGuiTest.java
+++ b/src/test/java/org/myrobotlab/service/WebGuiTest.java
@@ -13,10 +13,12 @@
 import org.myrobotlab.codec.CodecUtils;
 import org.myrobotlab.framework.MRLListener;
 import org.myrobotlab.framework.Service;
+import org.myrobotlab.framework.StaticType;
 import org.myrobotlab.framework.TimeoutException;
 import org.myrobotlab.logging.LoggerFactory;
 import org.myrobotlab.net.Http;
 import org.myrobotlab.test.AbstractTest;
+import org.myrobotlab.utils.ObjectTypePair;
 import org.slf4j.Logger;
 
 public class WebGuiTest extends AbstractTest {
@@ -148,8 +150,11 @@ public void urlEncodingTest() {
   public void sendBlockingTest() throws InterruptedException, TimeoutException {
     String retVal = "retVal";
     // Put directly in blocking list because sendBlocking() won't use it for local services
-    Runtime.getInstance().getInbox().blockingList.put("runtime.onBlocking", new Object[1]);
-    Object[] blockingListRet = Runtime.getInstance().getInbox().blockingList.get("runtime.onBlocking");
+    Runtime.getInstance().getInbox().blockingList.put(
+            "runtime.onBlocking",
+                    new ObjectTypePair<>(null, new StaticType(){})
+    );
+    ObjectTypePair blockingListRet = Runtime.getInstance().getInbox().blockingList.get("runtime.onBlocking");
 
     // Delay in a new thread so we can get our wait() call in first
     new Thread(() -> {
@@ -168,7 +173,7 @@ public void sendBlockingTest() throws InterruptedException, TimeoutException {
       }
     }
 
-    assertEquals(retVal, blockingListRet[0]);
+    assertEquals(retVal, blockingListRet.first);
   }