blob: a3051241c4efc32186e5ece63bfe93fd947283c5 [file] [log] [blame]
/*
* Copyright 2000-2009 JetBrains s.r.o.
*
* 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.
*/
/*
* @author max
*/
package com.intellij.util.messages.impl;
import com.intellij.openapi.diagnostic.Logger;
import com.intellij.util.SmartFMap;
import com.intellij.util.messages.MessageBusConnection;
import com.intellij.util.messages.MessageHandler;
import com.intellij.util.messages.Topic;
import org.jetbrains.annotations.NotNull;
import java.lang.reflect.Method;
import java.util.Queue;
public class MessageBusConnectionImpl implements MessageBusConnection {
private static final Logger LOG = Logger.getInstance("#com.intellij.util.messages.impl.MessageBusConnectionImpl");
private final MessageBusImpl myBus;
@SuppressWarnings("SSBasedInspection")
private final ThreadLocal<Queue<Message>> myPendingMessages = MessageBusImpl.createThreadLocalQueue();
private MessageHandler myDefaultHandler;
private volatile SmartFMap<Topic, Object> mySubscriptions = SmartFMap.emptyMap();
public MessageBusConnectionImpl(@NotNull MessageBusImpl bus) {
myBus = bus;
}
@Override
public <L> void subscribe(@NotNull Topic<L> topic, @NotNull L handler) throws IllegalStateException {
synchronized (myPendingMessages) {
if (mySubscriptions.get(topic) != null) {
throw new IllegalStateException("Subscription to " + topic + " already exists");
}
mySubscriptions = mySubscriptions.plus(topic, handler);
}
myBus.notifyOnSubscription(this, topic);
}
@Override
@SuppressWarnings("unchecked")
public <L> void subscribe(@NotNull Topic<L> topic) throws IllegalStateException {
if (myDefaultHandler == null) {
throw new IllegalStateException("Connection must have default handler installed prior to any anonymous subscriptions. "
+ "Target topic: " + topic);
}
if (topic.getListenerClass().isInstance(myDefaultHandler)) {
throw new IllegalStateException("Can't subscribe to the topic '" + topic +"'. Default handler has incompatible type - expected: '" +
topic.getListenerClass() + "', actual: '" + myDefaultHandler.getClass() + "'");
}
subscribe(topic, (L)myDefaultHandler);
}
@Override
public void setDefaultHandler(MessageHandler handler) {
myDefaultHandler = handler;
}
@Override
public void disconnect() {
Queue<Message> jobs = myPendingMessages.get();
myPendingMessages.remove();
myBus.notifyConnectionTerminated(this);
if (!jobs.isEmpty()) {
LOG.error("Not delivered events in the queue: " + jobs);
}
}
@Override
public void dispose() {
disconnect();
}
@Override
public void deliverImmediately() {
Queue<Message> messages = myPendingMessages.get();
while (!messages.isEmpty()) {
myBus.deliverSingleMessage();
}
}
void deliverMessage(@NotNull Message message) {
final Message messageOnLocalQueue = myPendingMessages.get().poll();
assert messageOnLocalQueue == message;
final Topic topic = message.getTopic();
final Object handler = mySubscriptions.get(topic);
try {
Method listenerMethod = message.getListenerMethod();
if (handler == myDefaultHandler) {
myDefaultHandler.handle(listenerMethod, message.getArgs());
}
else {
listenerMethod.invoke(handler, message.getArgs());
}
}
catch (AbstractMethodError e) {
//Do nothing. This listener just does not implement something newly added yet.
}
catch(Throwable e) {
LOG.error(e.getCause() == null ? e : e.getCause());
}
}
void scheduleMessageDelivery(Message message) {
myPendingMessages.get().offer(message);
}
public String toString() {
return mySubscriptions.toString();
}
MessageBusImpl getBus() {
return myBus;
}
}