All Downloads are FREE. Search and download functionalities are using the official Maven repository.

io.joynr.android.binder.BinderMessageProcessor Maven / Gradle / Ivy

There is a newer version: 1.25.0
Show newest version
/*-
 * #%L
 * %%
 * Copyright (C) 2019 BMW Car IT GmbH
 * %%
 * 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
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * 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.
 * #L%
 */
package io.joynr.android.binder;

import android.os.Handler;
import android.os.Looper;

import com.google.inject.Injector;
import com.google.inject.Key;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.util.ArrayList;
import java.util.List;
import java.util.Set;

import io.joynr.android.AndroidBinderRuntime;
import io.joynr.messaging.JoynrMessageProcessor;
import io.joynr.messaging.routing.MessageRouter;
import joynr.ImmutableMessage;

public class BinderMessageProcessor {

    private static final Logger logger = LoggerFactory.getLogger(BinderMessageProcessor.class);

    private static final long DELAY_MILLISECOND_FOR_RETRY_CHECK_CC = 1000L;
    private static final int MESSAGE_WHAT = 0;

    protected List messagesWaitingProcessing;

    private boolean mainTransport;

    public BinderMessageProcessor(boolean mainTransport) {
        this.mainTransport = mainTransport;
        messagesWaitingProcessing = new ArrayList();
    }

    /**
     * Process's a message passed as parameter sending it to the message processors
     * available in the guice injector.
     * @param injector the guice injector being used
     * @param message the message that you want to process
     */
    protected void processMessage(Injector injector, ImmutableMessage message) {
        Set messageProcessors = injector.getInstance(new Key>() {
        });
        if (messageProcessors != null) {
            for (JoynrMessageProcessor processor : messageProcessors) {
                message = processor.processIncoming(message);
            }
        }

        if (mainTransport) {
            /*
             * On LibJoynr side, prevent message loops by marking the messages as
             * received from global. The LibJoynrMessageRouter prevents sending
             * back the message to the CC.
             */
            message.setReceivedFromGlobal(true);
        }

        MessageRouter messageRouter = AndroidBinderRuntime.getInjector().getInstance(MessageRouter.class);

        messageRouter.routeIn(message);
    }

    protected Handler createInjectorRuntimeAvailabilityHandler() {
        return new Handler(Looper.getMainLooper()) {

            @Override
            public void handleMessage(android.os.Message msg) {
                try {
                    Injector injector = AndroidBinderRuntime.getInjector();
                    if (injector != null) {
                        messagesWaitingProcessing.forEach(immutableMessage -> {
                            processMessage(injector, immutableMessage);
                        });
                        messagesWaitingProcessing.clear();
                        logger.debug("BinderService Runtime available. Processing {} messages", messagesWaitingProcessing.size());
                    } else {
                        logger.debug("BinderService Runtime not available. Checking again in 1 second...");
                        removeAndSendMessageDelayedToHandler();
                    }
                } catch (Exception error) {
                    logger.error("BinderService handleMessage message not processed: {}", error.getMessage());
                }
            }
        };
    }

    /**
     * Handler responsible for checking if the guice injector used by the joynr runtime
     * gets available so that messages received before the runtime gets initialized are processed.
     */
    protected Handler checkInjectorRuntimeAvailabilityHandler = createInjectorRuntimeAvailabilityHandler();

    /**
     * Helper function that removes possible messages waiting to be processed in the handler
     * and send a new fresh message delayed so that we can have only one message to be processed
     * in the queue.
     */
    protected void removeAndSendMessageDelayedToHandler() {
        checkInjectorRuntimeAvailabilityHandler.removeMessages(MESSAGE_WHAT);
        checkInjectorRuntimeAvailabilityHandler.sendEmptyMessageDelayed(MESSAGE_WHAT, DELAY_MILLISECOND_FOR_RETRY_CHECK_CC);
    }

    protected void addToWaitingProcessing(ImmutableMessage message) {
        messagesWaitingProcessing.add(message);
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy