Что ж, мне придется оставить это для кого-то, кто более опытен, чем я, потому что сейчас у меня больше дел. У меня был вопрос от читателя об исходном коде Java EE 7 Developer Handbook . Работает ли JMS 2.0 как клиент? Из этого я понял, что разработчик должен иметь возможность запускать GlassFish 4 и получать доступ к очередям и темам JMS из отдельного контейнера. Раньше я делал это для инвестиционных банков, использующих J2EE 1.4 с серверами приложений WebSphere и WebLogic, и поэтому я уверен, что код, с которым я работал несколько месяцев, обязательно сработает, вопрос читателя — скорее мой собственный надзор. Должно быть, я мечтал об этих результатах.
Извините, я потратил несколько дней на то, чтобы заставить GlassFish Embedded 4.0 и затем управляемый сервер работать вместе, чтобы продемонстрировать простой пример JMS 2.0. Я также использовал тест Arquillian для работы, а затем также попробовал не-Arquillian версию, и теперь я очень обеспокоен. Казалось бы, никто не знает в Сети, как точно настроить автономную JMS 2.0 как приложение Java SE.
Это насколько я понимаю:
package je7hb.jms.essentials;
import static org.junit.Assert.*;
import javax.annotation.Resource;
import javax.ejb.*;
import javax.jms.*;
import javax.naming.Context;
import javax.naming.InitialContext;
import javax.naming.NamingException;
import org.jboss.arquillian.container.test.api.Deployment;
import org.jboss.arquillian.container.test.api.RunAsClient;
import org.jboss.arquillian.junit.Arquillian;
import org.jboss.shrinkwrap.api.ArchivePaths;
import org.jboss.shrinkwrap.api.ShrinkWrap;
import org.jboss.shrinkwrap.api.asset.EmptyAsset;
import org.jboss.shrinkwrap.api.spec.JavaArchive;
import org.jboss.shrinkwrap.api.spec.WebArchive;
import org.junit.*;
import org.junit.runner.RunWith;
import java.io.File;
import java.util.ArrayList;
import java.util.List;
import java.util.Properties;
@RunWith(Arquillian.class)
public class AsynchronousJMSMessageArquillianTest {
@Deployment
public static WebArchive createDeployment() {
WebArchive webArchive = ShrinkWrap.create(WebArchive.class, "asyncjms.war")
.addClasses(PayloadCheck.class)
.addAsWebInfResource(
new File("src/test/resources-glassfish-managed/glassfish-resources.xml"),
"glassfish-resources.xml")
.addAsWebInfResource(
EmptyAsset.INSTANCE, "beans.xml");
System.out.println(webArchive.toString(true));
return webArchive;
}
private List<String> messages = new ArrayList<>();
private CompletionListener completionListener = new CompletionListener() {
@Override
public void onCompletion(Message msg) {
TextMessage textMsg = (TextMessage)msg;
try {
System.out.printf("%s.onCompletion(%s) Thread: %s\n",
getClass().getSimpleName(), textMsg.getText(), Thread.currentThread());
} catch (JMSException e) {
e.printStackTrace(System.err);
}
}
@Override
public void onException(Message msg, Exception ex) {
ex.printStackTrace(System.err);
}
};
@Test
@RunAsClient
public void shouldFire() throws JMSException, InterruptedException, NamingException {
Properties properties = new Properties();
// properties.put("com.sun.appserv.iiop.endpoints", "localhost:7676");
// properties.put("org.omg.CORBA.ORBInitialHost", "localhost");
// properties.put("org.omg.CORBA.ORBInitialPort", "3700");
// properties.put(Context.INITIAL_CONTEXT_FACTORY, "com.sun.enterprise.naming.SerialInitContextFactory");
// properties.put(Context.PROVIDER_URL, "mq://localhost:7676/");
// properties.put("java.naming.factory.url.pkgs", "com.sun.enterprise.naming");
// properties.put("java.naming.factory.state", "com.sun.corba.ee.impl.presentation.rmi.JNDIStateFactoryImpl");
properties.put(Context.INITIAL_CONTEXT_FACTORY, "com.sun.enterprise.naming.SerialInitContextFactory");
properties.put(Context.URL_PKG_PREFIXES, "com.sun.enterprise.naming");
properties.put(Context.STATE_FACTORIES, "com.sun.corba.ee.impl.presentation.rmi.JNDIStateFactoryImpl");
// properties.put("org.omg.CORBA.ORBInitialHost", "localhost");
// properties.put("org.omg.CORBA.ORBInitialPort", "3700");
properties.put(Context.PROVIDER_URL, "mq://localhost:7676"); // vm://localhost:
// properties.put(Context.PROVIDER_URL, "iiop://localhost:7676"); // vm://localhost:
InitialContext jndiContext = new InitialContext(properties);
System.out.printf("\t jndiContext=%s\n", jndiContext);
ConnectionFactory connectionFactory =
(ConnectionFactory)jndiContext.lookup("jms/demoConnectionFactory");
Queue queue = (Queue)jndiContext.lookup("jms/demoQueue");
assertNotNull(connectionFactory);
assertNotNull(queue);
Connection connection = connectionFactory.createConnection();
JMSContext context = connectionFactory.createContext(
Session.AUTO_ACKNOWLEDGE );
JMSProducer producer = context.createProducer();
messages.clear();
producer.setAsync(completionListener);
producer.send(queue, "hello");
producer.send(queue, "world");
producer.send(queue, "asynchronously");
Thread.sleep(2000); // Delay before shutdown
System.out.println("Done");
}
}
Доказательства есть в коде, и вы можете просмотреть закомментированные строки, потому что каждый, а некоторые из них были опробованы
Причина в том, что определенные вызовы, такие как setAsync , запрещены в Java EE и веб-контейнере. Поэтому, если вам нужен асинхронный JMS-ридер или писатель в EE, вы можете забыть об этом.
Файл конфигурации ресурса Glassfish:
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE resources PUBLIC "-//GlassFish.org//DTD GlassFish Application Server 3.1 Resource Definitions//EN" "http://glassfish.org/dtds/glassfish-resources_1_5.dtd">
<resources>
<admin-object-resource enabled="true" jndi-name="jms/demoQueue" object-type="user" res-adapter="jmsra" res-type="javax.jms.Queue">
<description/>
<property name="Name" value="PhysicalQueue"/>
</admin-object-resource>
<admin-object-resource enabled="true" jndi-name="jms/demoTopic" object-type="user" res-adapter="jmsra" res-type="javax.jms.Topic">
<description/>
<property name="Name" value="PhysicalTopic"/>
</admin-object-resource>
<connector-connection-pool name="jms/demoDestinationFactoryPool"
connection-definition-name="javax.jms.QueueConnectionFactory"
resource-adapter-name="jmsra" />
<connector-resource enabled="true"
jndi-name="jms/demoConnectionFactory"
pool-name="jms/demoDestinationFactoryPool"
object-type="system-all" />
</resources>
На самом деле, это такое неловкое положение, я бы рекомендовал не смотреть на GlassFish для отдельного клиента JMS, а, возможно, подождать WildFly или Tom EE . Очевидно, что использование JMS 2.0 вне контейнера EE очень непривлекательно, и я думаю, что это будет чертовски легко для новых разработчиков, в противном случае они хотят коснуться Java EE с помощью штанги баржи. Если технический эксперт не может разработать простой пример использования JMS вне контейнера, то какой в этом смысл? Официальная страница JMS 2.0 также бесполезна.
Частично проблема заключается в том, что нет стандартного JMS API на стороне клиента для Java SE, JAX RS 2.0 теперь имеет клиентский API. Другое заключается в том, что некоторые из артефактов GlassFish Maven имеют огромные зависимости, которые вызывают загрузку всех модулей сервера приложений (я указал пальцем на это gf-client:4.0-SNAPSHOT). Наконец, эта трудная проблема на самом деле не задокументирована, и, к сожалению, пример учебного кода по Java EE 7 доступен только как часть пакета, а не как независимый для быстрого доступа.
Поэтому я планирую покинуть этот JMS async проект как часть распространения кода книги. Извините, что код не работает вообще. Этот код должен работать как единый интеграционный тест в ссылочной реализации GlassFish и Arquillian. У меня больше нет времени на это, но я исключу его создание (потому что тест не пройден) в новых усовершенствованиях подпроекта Gradle, которые я сейчас пишу.
