Может ли сторона сервера EJB отправлять событие стороне клиента EJB? - PullRequest
4 голосов
/ 16 ноября 2011

Интересно, можно ли что-то сделать в этом направлении:

1) Сторона сервера (класс EJB)

@Statefull
public class SomeEJB implements SomeEJBRemote {

@Resource
private SessionContext sessionContext;

//(...)
public void someMethod(Record record){
    //(...)
    //Situation that requires the server to alert all registered listeners
    if(specialSituation){
        sessionContext.fireEventToListeners(new SomeEvent());
    }

    //Do something else...
}
}

2) Клиентская сторона

//(...) 
SomeEJBRemote ejb = initialContext.lookup("SomeEJBRemote");
ejb.addListener(this);

void EJBEventListener(SomeEvent event){
    System.out.println("EJB server has sent an event!");
}

A) Есть ли в спецификации Java EE что-нибудь, позволяющее это сделать?

B) Я знаю, что JBoss разрешает двунаправленную связь, но могу ли я сделать так же, как этот пример?

C) Есть ли что-нибудь подобное в OpenEJB (или TOMEE)?

Ответы [ 2 ]

4 голосов
/ 28 ноября 2011

Это легко сделать с помощью встроенного контейнера EJB и MDB в клиенте. У нас есть пример, который делает именно это.

Проверьте monitor модуль , этот пример .

На 10 000 футов этот пример делает следующее:

на стороне сервера:

  • @ Bean-компонент без состояния, который оборачивает доступ к EntityManager
  • JMS Сообщение отправлено в тему обо всех операциях добавления / удаления

на стороне клиента:

  • Встроенный EJB-контейнер / получение сообщений MDB
  • По получении уведомление выдается пользователю через java.awt.SystemTray

Итак, интересная особенность этого метода в том, что он полностью транзакционный - EntityManager обновляет и отправленные сообщения JMS, все являются частью транзакции. Если обновление базы данных завершится неудачно, сообщение JMS не будет отправлено.

Вот 100% клиентского кода из этого примера. Не требуется много, чтобы сделать то, что описано.

клиент "основного" класса

import javax.naming.InitialContext;
import javax.naming.NamingException;
import java.awt.AWTException;
import java.awt.Image;
import java.awt.MenuItem;
import java.awt.PopupMenu;
import java.awt.SystemTray;
import java.awt.Toolkit;
import java.awt.TrayIcon;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.net.MalformedURLException;
import java.net.URL;

public class NotificationMonitor {
    private static TrayIcon trayIcon;

    public static void main(String[] args) throws NamingException, InterruptedException, AWTException, MalformedURLException {
        addSystemTrayIcon();

        // Boot the embedded EJB Container 
        new InitialContext();

        System.out.println("Starting monitor...");
    }

    private static void addSystemTrayIcon() throws AWTException, MalformedURLException {
        SystemTray tray = SystemTray.getSystemTray();

        URL moviepng = NotificationMonitor.class.getClassLoader().getResource("movie.png");
        Image image = Toolkit.getDefaultToolkit().getImage(moviepng);

        ActionListener exitListener = new ActionListener() {
            public void actionPerformed(ActionEvent e) {
                System.out.println("Exiting monitor...");
                System.exit(0);
            }
        };

        PopupMenu popup = new PopupMenu();
        MenuItem defaultItem = new MenuItem("Exit");
        defaultItem.addActionListener(exitListener);
        popup.add(defaultItem);

        trayIcon = new TrayIcon(image, "Notification Monitor", popup);
        trayIcon.setImageAutoSize(true);
        tray.add(trayIcon);
    }

    public static void showAlert(String message) {
        synchronized (trayIcon) {
            trayIcon.displayMessage("Alert received", message, TrayIcon.MessageType.WARNING);
        }
    }
}

Клиентский MDB

import javax.ejb.ActivationConfigProperty;
import javax.ejb.MessageDriven;
import javax.jms.JMSException;
import javax.jms.Message;
import javax.jms.MessageListener;
import javax.jms.TextMessage;

@MessageDriven(activationConfig = {
        @ActivationConfigProperty(propertyName = "destinationType", propertyValue = "javax.jms.Topic"),
        @ActivationConfigProperty(propertyName = "destination", propertyValue = "notifications")})
public class NotificationsBean implements MessageListener {

    public void onMessage(Message message) {
        try {
            TextMessage textMessage = (TextMessage) message;
            String text = textMessage.getText();

            NotificationMonitor.showAlert(text);
        } catch (JMSException e) {
            e.printStackTrace();
        }
    }
}

Клиентский файл jndi.properties

Это настраивает встроенный контейнер EJB. Вы можете сделать это и в коде.

java.naming.factory.initial=org.apache.openejb.client.LocalInitialContextFactory
Default\ JMS\ Resource\ Adapter=new://Resource?type=ActiveMQResourceAdapter
Default\ JMS\ Resource\ Adapter.BrokerXmlConfig=broker:vm://localhost
Default\ JMS\ Resource\ Adapter.ServerUrl=tcp://localhost:61616
1 голос
/ 17 ноября 2011

Нет, в EJB ничего подобного нет.Я бы посоветовал клиенту прослушать очередь / тему JMS.Альтернативно, клиент может экспортировать объект слушателя в RMI (фактически становясь самим сервером), а затем передать ссылку на сервер;этот метод потребует немного больше инфраструктуры.

...