grails アプリケーションで Atmosphere を使用しています。私のIDE(IntelliJ Idea)からアプリケーションを実行するときは、すべて問題ありません。しかし、Tomcat (7.0) にデプロイすると、次の例外が発生します。
2013-07-08 09:07:19,118 [ajp-nio-8009-exec-13] ERROR cpr.AtmosphereFramework - AtmosphereFramework exception
java.lang.IllegalStateException: Not supported.
at org.atmosphere.cpr.AtmosphereRequest.startAsync(AtmosphereRequest.java:594)
at org.atmosphere.container.Servlet30CometSupport.suspend(Servlet30CometSupport.java:138)
at org.atmosphere.container.Servlet30CometSupport.service(Servlet30CometSupport.java:104)
at org.atmosphere.container.Tomcat7Servlet30SupportWithWebSocket.doService(Tomcat7Servlet30SupportWithWebSocket.java:65)
at org.atmosphere.container.TomcatWebSocketUtil.doService(TomcatWebSocketUtil.java:87)
at org.atmosphere.container.Tomcat7Servlet30SupportWithWebSocket.service(Tomcat7Servlet30SupportWithWebSocket.java:61)
at org.atmosphere.cpr.AtmosphereFramework.doCometSupport(AtmosphereFramework.java:1571)
at org.atmosphere.cpr.AtmosphereServlet.doPost(AtmosphereServlet.java:176)
at org.atmosphere.cpr.AtmosphereServlet.doGet(AtmosphereServlet.java:162)
at com.googlecode.psiprobe.Tomcat70AgentValve.invoke(Tomcat70AgentValve.java:38)
at java.util.concurrent.ThreadPoolExecutor$Worker.runTask(ThreadPoolExecutor.java:886)
at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:908)
at java.lang.Thread.run(Thread.java:662)
web.xml の myservlet 構成は次のとおりです。
<servlet>
<description>MeteorServlet</description>
<servlet-name>MeteorServlet</servlet-name>
<servlet-class>org.grails.plugin.platform.events.push.GrailsMeteorServlet</servlet-class>
<init-param>
<param-name>org.atmosphere.cpr.broadcaster.shareableThreadPool</param-name>
<param-value>true</param-value>
</init-param>
<init-param>
<param-name>org.atmosphere.cpr.broadcaster.maxProcessingThreads</param-name>
<param-value>20</param-value>
</init-param>
<init-param>
<param-name>org.atmosphere.cpr.broadcaster.maxAsyncWriteThreads</param-name>
<param-value>20</param-value>
</init-param>
<load-on-startup>0</load-on-startup>
<async-supported>true</async-supported>
</servlet>
使い方は
var receivedOrders = new Array();
var grailsEvents = new grails.Events("${rootPath}",
{
transport: 'sse',
fallbackTransport: 'long-polling',
timeout: 10000,
onMessage: function(data){
try{
if(data.responseBody.length > 0){
var order = jQuery.parseJSON(data.responseBody).body;
if(order.id){
if (receivedOrders.indexOf(order.id) == -1) {
receivedOrders[receivedOrders.length] = order.id;
var url = "<g:createLink controller="orderAdministration" action="orderNotification"/>";
$.ajax({
type: "POST",
url: url,
data: { id: order.id }
}).done(function (response) {
if (response != "0") {
$.msgGrowl({
type: 'info', sticky: true, 'title': '${message(code: 'order.notification.title')}', 'text': response, lifetime: 5000
});
}
});
}
}
}
} catch (e) {
// Atmosphere sends commented out data to WebKit based browsers
}
}
});
grailsEvents.on('order_event', function(data){});
Tomcat の設定に問題があるようです。何か案が?
編集:
私はそれをテストしました。しかし、動作しません。
grails.Events でオプションを指定したために問題が発生しました。これに変更することで、例外が解決しました。
var receivedOrders = new Array();
var grailsEvents = new grails.Events("${rootPath}");
function handleOrderEvent(data){
try{
if(data.id){
if (receivedOrders.indexOf(data.id) == -1) {
receivedOrders[receivedOrders.length] = data.id;
var url = "<g:createLink controller="orderAdministration" action="orderNotification"/>";
$.ajax({
type: "POST",
url: url,
data: { id: data.id }
}).done(function (response) {
if (response != "0") {
$.msgGrowl({
type: 'info', sticky: true, 'title': '${message(code: 'order.notification.title')}', 'text': response, lifetime: 5000
});
}
});
}
}
}catch (e) {
// Atmosphere sends commented out data to WebKit based browsers
}
}
grailsEvents.on('order_event', handleOrderEvent, {transport:'long-polling', fallbackTransport:'polling'});
しかし、まだイベントはクライアントに伝播されていません!
Tomcatの前にApache Webサーバーがあります。イベントはサービスで発生しましたが、JavaScript では発生しませんでした。