Search in sources :

Example 1 with InputStreamHandle

use of ch.ethz.iks.r_osgi.streams.InputStreamHandle in project ecf by eclipse.

the class ChannelEndpointImpl method handleMessage.

/**
 * message handler method.
 *
 * @param msg
 *            the incoming message.
 * @return if reply is created, null otherwise.
 * @throws RemoteOSGiException
 *             if something goes wrong.
 */
RemoteOSGiMessage handleMessage(final RemoteOSGiMessage msg) throws RemoteOSGiException {
    trace("handleMessage(msg=" + msg + ";remoteAddress=" + networkChannel.getRemoteAddress() + ")");
    switch(msg.getFuncID()) {
        // requests
        case RemoteOSGiMessage.LEASE:
            {
                final LeaseMessage lease = (LeaseMessage) msg;
                processLease(lease);
                populateLease(lease, RemoteOSGiServiceImpl.getServices(networkChannel.getProtocol()), RemoteOSGiServiceImpl.getTopics());
                return lease;
            }
        case RemoteOSGiMessage.REQUEST_SERVICE:
            {
                final RequestServiceMessage reqSrv = (RequestServiceMessage) msg;
                final String serviceID = reqSrv.getServiceID();
                final RemoteServiceRegistration reg = getServiceRegistration(serviceID);
                final DeliverServiceMessage m = reg.getDeliverServiceMessage();
                m.setXID(reqSrv.getXID());
                m.setServiceID(reqSrv.getServiceID());
                return m;
            }
        case RemoteOSGiMessage.LEASE_UPDATE:
            {
                final LeaseUpdateMessage suMsg = (LeaseUpdateMessage) msg;
                final String serviceID = suMsg.getServiceID();
                final short stateUpdate = suMsg.getType();
                final String serviceURI = getRemoteAddress().resolve("#" + serviceID).toString();
                switch(stateUpdate) {
                    case LeaseUpdateMessage.TOPIC_UPDATE:
                        {
                            // There is an older r-OSGi version that incorrectly sends an ArrayList
                            // (1.0.0.RC4_v20131016-1848)
                            Object topicsAdded = suMsg.getPayload()[0];
                            if (topicsAdded instanceof List) {
                                topicsAdded = ((List) topicsAdded).toArray(new String[0]);
                            }
                            Object topicsRemoved = suMsg.getPayload()[1];
                            if (topicsRemoved instanceof List) {
                                topicsRemoved = ((List) topicsRemoved).toArray(new String[0]);
                            }
                            updateTopics((String[]) topicsAdded, (String[]) topicsRemoved);
                            return null;
                        }
                    case LeaseUpdateMessage.SERVICE_ADDED:
                        {
                            final Dictionary properties = (Dictionary) suMsg.getPayload()[1];
                            sanitizeServiceProperties(properties, serviceURI);
                            final RemoteServiceReferenceImpl ref = new RemoteServiceReferenceImpl((String[]) suMsg.getPayload()[0], serviceID, properties, this);
                            remoteServices.put(serviceURI, ref);
                            RemoteOSGiServiceImpl.notifyRemoteServiceListeners(new RemoteServiceEvent(RemoteServiceEvent.REGISTERED, ref));
                            return null;
                        }
                    case LeaseUpdateMessage.SERVICE_MODIFIED:
                        {
                            final Dictionary properties = (Dictionary) suMsg.getPayload()[1];
                            sanitizeServiceProperties(properties, serviceURI);
                            final ServiceRegistration reg = (ServiceRegistration) proxiedServices.get(serviceID);
                            if (reg != null) {
                                reg.setProperties(properties);
                            }
                            // $NON-NLS-1$
                            final RemoteServiceReferenceImpl ref = getRemoteReference(serviceURI);
                            // (see https://bugs.eclipse.org/420433)
                            if (ref == null && reg == null) {
                                return null;
                            }
                            ref.setProperties(properties);
                            RemoteOSGiServiceImpl.notifyRemoteServiceListeners(new RemoteServiceEvent(RemoteServiceEvent.MODIFIED, ref));
                            return null;
                        }
                    case LeaseUpdateMessage.SERVICE_REMOVED:
                        {
                            if (networkChannel == null) {
                                return null;
                            }
                            final RemoteServiceReference ref = (RemoteServiceReference) remoteServices.remove(serviceURI);
                            if (ref != null) {
                                RemoteOSGiServiceImpl.notifyRemoteServiceListeners(new RemoteServiceEvent(RemoteServiceEvent.UNREGISTERING, ref));
                            }
                            final Bundle bundle = (Bundle) proxyBundles.remove(serviceID);
                            if (bundle != null) {
                                try {
                                    bundle.uninstall();
                                } catch (final BundleException be) {
                                    be.printStackTrace();
                                }
                                proxiedServices.remove(serviceID);
                                // $NON-NLS-1$
                                remoteServices.remove(serviceURI);
                            }
                            return null;
                        }
                }
                return null;
            }
        case RemoteOSGiMessage.REMOTE_CALL:
            {
                final RemoteCallMessage invMsg = (RemoteCallMessage) msg;
                try {
                    RemoteServiceRegistration serv = (RemoteServiceRegistration) localServices.get(invMsg.getServiceID());
                    if (serv == null) {
                        final RemoteServiceRegistration reg = getServiceRegistration(invMsg.getServiceID());
                        if (reg == null) {
                            throw new IllegalStateException(toString() + "Could not get " + // $NON-NLS-1$
                            invMsg.getServiceID() + ", known services " + // $NON-NLS-1$
                            localServices);
                        } else {
                            serv = reg;
                        }
                    }
                    // get the invocation arguments and the local method
                    final Object[] arguments = invMsg.getArgs();
                    for (int i = 0; i < arguments.length; i++) {
                        if (arguments[i] instanceof InputStreamHandle) {
                            arguments[i] = getInputStreamProxy((InputStreamHandle) arguments[i]);
                        } else if (arguments[i] instanceof OutputStreamHandle) {
                            arguments[i] = getOutputStreamProxy((OutputStreamHandle) arguments[i]);
                        }
                    }
                    final Method method = serv.getMethod(invMsg.getMethodSignature());
                    // invoke method
                    try {
                        Object result = method.invoke(serv.getServiceObject(), arguments);
                        final RemoteCallResultMessage m = new RemoteCallResultMessage();
                        m.setXID(invMsg.getXID());
                        Class returnType = method.getReturnType();
                        if (result instanceof InputStream) {
                            m.setResult(getInputStreamPlaceholder((InputStream) result));
                        } else if (result instanceof OutputStream) {
                            m.setResult(getOutputStreamPlaceholder((OutputStream) result));
                        } else if (serv.isOSGiAsync() && AsyncReturnUtil.isAsyncType(returnType)) {
                            m.setResult(AsyncReturnUtil.convertAsyncToReturn(result, returnType, serv.getOSGiTimeout()));
                        } else
                            m.setResult(result);
                        return m;
                    } catch (final InvocationTargetException t) {
                        t.printStackTrace();
                        throw t.getTargetException();
                    }
                } catch (final Throwable t) {
                    // TODO: send to log
                    t.printStackTrace();
                    final RemoteCallResultMessage m = new RemoteCallResultMessage();
                    m.setXID(invMsg.getXID());
                    m.setException(t);
                    return m;
                }
            }
        case RemoteOSGiMessage.REMOTE_EVENT:
            {
                final RemoteEventMessage eventMsg = (RemoteEventMessage) msg;
                final Dictionary properties = eventMsg.getProperties();
                // transform the event timestamps
                final Long remoteTs;
                if ((remoteTs = (Long) properties.get(EventConstants.TIMESTAMP)) != null) {
                    properties.put(EventConstants.TIMESTAMP, getOffset().transform(remoteTs));
                }
                final Event event = new Event(eventMsg.getTopic(), properties);
                // and deliver the event to the local framework
                if (RemoteOSGiServiceImpl.eventAdminTracker.getTrackingCount() > 0) {
                    ((EventAdmin) RemoteOSGiServiceImpl.eventAdminTracker.getService()).postEvent(event);
                } else {
                    // TODO: to log
                    System.err.println(// $NON-NLS-1$
                    "Could not deliver received event: " + event + // $NON-NLS-1$
                    ". No EventAdmin available.");
                }
                return null;
            }
        case RemoteOSGiMessage.TIME_OFFSET:
            {
                // add timestamp to the message and return the message to sender
                ((TimeOffsetMessage) msg).timestamp();
                return msg;
            }
        case RemoteOSGiMessage.STREAM_REQUEST:
            {
                final StreamRequestMessage reqMsg = (StreamRequestMessage) msg;
                try {
                    // fetch stream object
                    final Object stream = streams.get(new Integer(reqMsg.getStreamID()));
                    if (stream == null) {
                        throw new IllegalStateException(// $NON-NLS-1$
                        "Could not get stream with ID " + reqMsg.getStreamID());
                    }
                    // invoke operation on stream
                    switch(reqMsg.getOp()) {
                        case StreamRequestMessage.READ:
                            {
                                final int result = ((InputStream) stream).read();
                                final StreamResultMessage m = new StreamResultMessage();
                                m.setXID(reqMsg.getXID());
                                m.setResult((short) result);
                                return m;
                            }
                        case StreamRequestMessage.READ_ARRAY:
                            {
                                final byte[] b = new byte[reqMsg.getLenOrVal()];
                                final int len = ((InputStream) stream).read(b, 0, reqMsg.getLenOrVal());
                                final StreamResultMessage m = new StreamResultMessage();
                                m.setXID(reqMsg.getXID());
                                m.setResult(StreamResultMessage.RESULT_ARRAY);
                                m.setLen(len);
                                if (len > 0) {
                                    m.setData(b);
                                }
                                return m;
                            }
                        case StreamRequestMessage.WRITE:
                            {
                                ((OutputStream) stream).write(reqMsg.getLenOrVal());
                                final StreamResultMessage m = new StreamResultMessage();
                                m.setXID(reqMsg.getXID());
                                m.setResult(StreamResultMessage.RESULT_WRITE_OK);
                                return m;
                            }
                        case StreamRequestMessage.WRITE_ARRAY:
                            {
                                ((OutputStream) stream).write(reqMsg.getData());
                                final StreamResultMessage m = new StreamResultMessage();
                                m.setXID(reqMsg.getXID());
                                m.setResult(StreamResultMessage.RESULT_WRITE_OK);
                                return m;
                            }
                        default:
                            throw new RemoteOSGiException(// $NON-NLS-1$
                            "Unimplemented op code for stream request " + msg);
                    }
                } catch (final IOException e) {
                    final StreamResultMessage m = new StreamResultMessage();
                    m.setXID(reqMsg.getXID());
                    m.setResult(StreamResultMessage.RESULT_EXCEPTION);
                    m.setException(e);
                    return m;
                }
            }
        case RemoteOSGiMessage.REQUEST_BUNDLE:
            final RequestBundleMessage reqB = (RequestBundleMessage) msg;
            try {
                final String serviceID = reqB.getServiceID();
                final RemoteServiceRegistration reg = getServiceRegistration(serviceID);
                final byte[] bytes = RemoteOSGiServiceImpl.getBundle(reg.getReference().getBundle());
                final DeliverBundlesMessage delB = new DeliverBundlesMessage();
                delB.setXID(reqB.getXID());
                delB.setDependencies(new byte[][] { bytes });
                return delB;
            } catch (IOException ioe) {
                ioe.printStackTrace();
                return null;
            }
        case RemoteOSGiMessage.REQUEST_DEPENDENCIES:
            final RequestDependenciesMessage reqDeps = (RequestDependenciesMessage) msg;
            try {
                final byte[][] bundleBytes = RemoteOSGiServiceImpl.getBundlesForPackages(reqDeps.getPackages());
                final DeliverBundlesMessage delDeps = new DeliverBundlesMessage();
                delDeps.setXID(reqDeps.getXID());
                delDeps.setDependencies(bundleBytes);
                return delDeps;
            } catch (IOException ioe) {
                ioe.printStackTrace();
                return null;
            }
        default:
            // $NON-NLS-1$
            throw new RemoteOSGiException("Unimplemented message " + msg);
    }
}
Also used : RequestServiceMessage(ch.ethz.iks.r_osgi.messages.RequestServiceMessage) Dictionary(java.util.Dictionary) DeliverServiceMessage(ch.ethz.iks.r_osgi.messages.DeliverServiceMessage) OutputStream(java.io.OutputStream) RequestBundleMessage(ch.ethz.iks.r_osgi.messages.RequestBundleMessage) RemoteCallResultMessage(ch.ethz.iks.r_osgi.messages.RemoteCallResultMessage) StreamRequestMessage(ch.ethz.iks.r_osgi.messages.StreamRequestMessage) InputStreamHandle(ch.ethz.iks.r_osgi.streams.InputStreamHandle) RemoteEventMessage(ch.ethz.iks.r_osgi.messages.RemoteEventMessage) RemoteOSGiException(ch.ethz.iks.r_osgi.RemoteOSGiException) List(java.util.List) ArrayList(java.util.ArrayList) BundleException(org.osgi.framework.BundleException) ServiceRegistration(org.osgi.framework.ServiceRegistration) OutputStreamHandle(ch.ethz.iks.r_osgi.streams.OutputStreamHandle) RequestDependenciesMessage(ch.ethz.iks.r_osgi.messages.RequestDependenciesMessage) Bundle(org.osgi.framework.Bundle) ByteArrayInputStream(java.io.ByteArrayInputStream) InputStream(java.io.InputStream) RemoteServiceEvent(ch.ethz.iks.r_osgi.RemoteServiceEvent) Method(java.lang.reflect.Method) StreamResultMessage(ch.ethz.iks.r_osgi.messages.StreamResultMessage) IOException(java.io.IOException) LeaseUpdateMessage(ch.ethz.iks.r_osgi.messages.LeaseUpdateMessage) LeaseMessage(ch.ethz.iks.r_osgi.messages.LeaseMessage) InvocationTargetException(java.lang.reflect.InvocationTargetException) ChannelEndpoint(ch.ethz.iks.r_osgi.channels.ChannelEndpoint) RemoteCallMessage(ch.ethz.iks.r_osgi.messages.RemoteCallMessage) RemoteServiceReference(ch.ethz.iks.r_osgi.RemoteServiceReference) RemoteServiceEvent(ch.ethz.iks.r_osgi.RemoteServiceEvent) Event(org.osgi.service.event.Event) DeliverBundlesMessage(ch.ethz.iks.r_osgi.messages.DeliverBundlesMessage)

Example 2 with InputStreamHandle

use of ch.ethz.iks.r_osgi.streams.InputStreamHandle in project ecf by eclipse.

the class ChannelEndpointImpl method asyncRemoteCall.

void asyncRemoteCall(final String fragment, final String methodSignature, final Object[] args, final AsyncRemoteCallCallback callback) {
    if (networkChannel == null) {
        // $NON-NLS-1$
        throw new RemoteOSGiException("Channel is closed");
    }
    // check arguments for streams and replace with placeholder
    for (int i = 0; i < args.length; i++) {
        if (args[i] instanceof InputStream) {
            args[i] = getInputStreamPlaceholder((InputStream) args[i]);
        } else if (args[i] instanceof OutputStream) {
            args[i] = getOutputStreamPlaceholder((OutputStream) args[i]);
        }
    }
    final Integer xid = new Integer(RemoteOSGiServiceImpl.nextXid());
    synchronized (callbacks) {
        callbacks.put(xid, new AsyncCallback() {

            public void result(final RemoteOSGiMessage msg) {
                final RemoteCallResultMessage resultMsg = (RemoteCallResultMessage) msg;
                if (resultMsg.causedException()) {
                    callback.remoteCallResult(false, resultMsg.getException());
                }
                final Object result = resultMsg.getResult();
                final Object res;
                if (result instanceof InputStreamHandle) {
                    res = getInputStreamProxy((InputStreamHandle) result);
                } else if (result instanceof OutputStreamHandle) {
                    res = getOutputStreamProxy((OutputStreamHandle) result);
                } else {
                    res = result;
                }
                callback.remoteCallResult(true, res);
            }
        });
    }
    final RemoteCallMessage invokeMsg = new RemoteCallMessage();
    invokeMsg.setServiceID(fragment);
    invokeMsg.setMethodSignature(methodSignature);
    invokeMsg.setArgs(args);
    invokeMsg.setXID(xid.shortValue());
    try {
        send(invokeMsg);
    } catch (final RemoteOSGiException e) {
        callbacks.remove(xid);
        callback.remoteCallResult(false, new RemoteOSGiException(// $NON-NLS-1$
        "Method invocation of " + getRemoteAddress() + "#" + fragment + " " + methodSignature + " failed.", // $NON-NLS-1$ //$NON-NLS-2$ //$NON-NLS-3$
        e));
    }
}
Also used : RemoteCallResultMessage(ch.ethz.iks.r_osgi.messages.RemoteCallResultMessage) OutputStreamHandle(ch.ethz.iks.r_osgi.streams.OutputStreamHandle) RemoteCallMessage(ch.ethz.iks.r_osgi.messages.RemoteCallMessage) RemoteOSGiException(ch.ethz.iks.r_osgi.RemoteOSGiException) InputStreamHandle(ch.ethz.iks.r_osgi.streams.InputStreamHandle) RemoteOSGiMessage(ch.ethz.iks.r_osgi.messages.RemoteOSGiMessage) ByteArrayInputStream(java.io.ByteArrayInputStream) InputStream(java.io.InputStream) OutputStream(java.io.OutputStream) ChannelEndpoint(ch.ethz.iks.r_osgi.channels.ChannelEndpoint)

Example 3 with InputStreamHandle

use of ch.ethz.iks.r_osgi.streams.InputStreamHandle in project ecf by eclipse.

the class ChannelEndpointImpl method invokeMethod.

/**
 * invoke a method on the remote host. This function is used by all proxy
 * bundles.
 *
 * @param service
 *            the service uri.
 * @param methodSignature
 *            the method signature.
 * @param args
 *            the method parameter.
 * @throws Throwable
 *             can throw any exception that the original method can throw,
 *             plus RemoteOSGiException.
 * @return the result of the remote method invocation.
 * @see ch.ethz.iks.r_osgi.channels.ChannelEndpoint#invokeMethod(java.lang.String,
 *      java.lang.String, java.lang.Object[])
 * @category ChannelEndpoint
 */
public Object invokeMethod(final String service, final String methodSignature, final Object[] args) throws Throwable {
    if (networkChannel == null) {
        // $NON-NLS-1$
        throw new RemoteOSGiException("Channel is closed");
    }
    // check arguments for streams and replace with placeholder
    for (int i = 0; i < args.length; i++) {
        if (args[i] instanceof InputStream) {
            args[i] = getInputStreamPlaceholder((InputStream) args[i]);
        } else if (args[i] instanceof OutputStream) {
            args[i] = getOutputStreamPlaceholder((OutputStream) args[i]);
        }
    }
    final RemoteCallMessage invokeMsg = new RemoteCallMessage();
    invokeMsg.setServiceID(URI.create(service).getFragment());
    invokeMsg.setMethodSignature(methodSignature);
    invokeMsg.setArgs(args);
    try {
        // send the message and get a MethodResultMessage in return
        final RemoteCallResultMessage resultMsg = (RemoteCallResultMessage) sendAndWait(invokeMsg);
        if (resultMsg.causedException()) {
            throw resultMsg.getException();
        }
        final Object result = resultMsg.getResult();
        if (result instanceof InputStreamHandle) {
            return getInputStreamProxy((InputStreamHandle) result);
        } else if (result instanceof OutputStreamHandle) {
            return getOutputStreamProxy((OutputStreamHandle) result);
        } else {
            if (result != null) {
                String returnType = Type.getReturnType(methodSignature).getClassName();
                RemoteServiceReferenceImpl refImpl = getRemoteReference(URI.create(service).toString());
                if (refImpl != null && refImpl.isOSGiAsync() && AsyncReturnUtil.isAsyncType(returnType))
                    return AsyncReturnUtil.convertReturnToAsync(result, returnType);
            }
            return result;
        }
    } catch (final RemoteOSGiException e) {
        throw new RemoteOSGiException(// $NON-NLS-1$
        "Method invocation of " + service + " " + methodSignature + " failed.", // $NON-NLS-1$ //$NON-NLS-2$
        e);
    }
}
Also used : RemoteCallResultMessage(ch.ethz.iks.r_osgi.messages.RemoteCallResultMessage) OutputStreamHandle(ch.ethz.iks.r_osgi.streams.OutputStreamHandle) RemoteCallMessage(ch.ethz.iks.r_osgi.messages.RemoteCallMessage) RemoteOSGiException(ch.ethz.iks.r_osgi.RemoteOSGiException) InputStreamHandle(ch.ethz.iks.r_osgi.streams.InputStreamHandle) ByteArrayInputStream(java.io.ByteArrayInputStream) InputStream(java.io.InputStream) OutputStream(java.io.OutputStream) ChannelEndpoint(ch.ethz.iks.r_osgi.channels.ChannelEndpoint)

Example 4 with InputStreamHandle

use of ch.ethz.iks.r_osgi.streams.InputStreamHandle in project ecf by eclipse.

the class ChannelEndpointImpl method getInputStreamPlaceholder.

/**
 * creates a placeholder for an InputStream that can be sent to the other
 * party and will be converted to an InputStream proxy there.
 *
 * @param origIS
 *            the instance of InputStream that needs to be remoted
 * @return the placeholder object that is sent to the actual client
 */
private InputStreamHandle getInputStreamPlaceholder(final InputStream origIS) {
    final InputStreamHandle sp = new InputStreamHandle(nextStreamID());
    streams.put(new Integer(sp.getStreamID()), origIS);
    return sp;
}
Also used : InputStreamHandle(ch.ethz.iks.r_osgi.streams.InputStreamHandle)

Aggregations

InputStreamHandle (ch.ethz.iks.r_osgi.streams.InputStreamHandle)4 RemoteOSGiException (ch.ethz.iks.r_osgi.RemoteOSGiException)3 ChannelEndpoint (ch.ethz.iks.r_osgi.channels.ChannelEndpoint)3 RemoteCallMessage (ch.ethz.iks.r_osgi.messages.RemoteCallMessage)3 RemoteCallResultMessage (ch.ethz.iks.r_osgi.messages.RemoteCallResultMessage)3 OutputStreamHandle (ch.ethz.iks.r_osgi.streams.OutputStreamHandle)3 ByteArrayInputStream (java.io.ByteArrayInputStream)3 InputStream (java.io.InputStream)3 OutputStream (java.io.OutputStream)3 RemoteServiceEvent (ch.ethz.iks.r_osgi.RemoteServiceEvent)1 RemoteServiceReference (ch.ethz.iks.r_osgi.RemoteServiceReference)1 DeliverBundlesMessage (ch.ethz.iks.r_osgi.messages.DeliverBundlesMessage)1 DeliverServiceMessage (ch.ethz.iks.r_osgi.messages.DeliverServiceMessage)1 LeaseMessage (ch.ethz.iks.r_osgi.messages.LeaseMessage)1 LeaseUpdateMessage (ch.ethz.iks.r_osgi.messages.LeaseUpdateMessage)1 RemoteEventMessage (ch.ethz.iks.r_osgi.messages.RemoteEventMessage)1 RemoteOSGiMessage (ch.ethz.iks.r_osgi.messages.RemoteOSGiMessage)1 RequestBundleMessage (ch.ethz.iks.r_osgi.messages.RequestBundleMessage)1 RequestDependenciesMessage (ch.ethz.iks.r_osgi.messages.RequestDependenciesMessage)1 RequestServiceMessage (ch.ethz.iks.r_osgi.messages.RequestServiceMessage)1