145 lines
4.9 KiB
Java
145 lines
4.9 KiB
Java
|
/*
|
||
|
* Licensed to the Apache Software Foundation (ASF) under one or more
|
||
|
* contributor license agreements. See the NOTICE file distributed with
|
||
|
* this work for additional information regarding copyright ownership.
|
||
|
* The ASF licenses this file to You 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.
|
||
|
*/
|
||
|
package async;
|
||
|
|
||
|
import java.io.IOException;
|
||
|
import java.io.PrintWriter;
|
||
|
import java.util.Iterator;
|
||
|
import java.util.concurrent.ConcurrentLinkedQueue;
|
||
|
import java.util.concurrent.atomic.AtomicInteger;
|
||
|
|
||
|
import javax.servlet.AsyncContext;
|
||
|
import javax.servlet.AsyncEvent;
|
||
|
import javax.servlet.AsyncListener;
|
||
|
import javax.servlet.ServletContext;
|
||
|
import javax.servlet.ServletException;
|
||
|
import javax.servlet.http.HttpServlet;
|
||
|
import javax.servlet.http.HttpServletRequest;
|
||
|
import javax.servlet.http.HttpServletResponse;
|
||
|
|
||
|
import async.Stockticker.Stock;
|
||
|
import async.Stockticker.TickListener;
|
||
|
|
||
|
public class AsyncStockServlet extends HttpServlet implements TickListener, AsyncListener{
|
||
|
|
||
|
private static final long serialVersionUID = 1L;
|
||
|
|
||
|
private static final ConcurrentLinkedQueue<AsyncContext> clients =
|
||
|
new ConcurrentLinkedQueue<>();
|
||
|
private static final AtomicInteger clientcount = new AtomicInteger(0);
|
||
|
|
||
|
public AsyncStockServlet() {
|
||
|
System.out.println("AsyncStockServlet created");
|
||
|
}
|
||
|
|
||
|
|
||
|
@Override
|
||
|
protected void service(HttpServletRequest req, HttpServletResponse resp)
|
||
|
throws ServletException, IOException {
|
||
|
if (req.isAsyncStarted()) {
|
||
|
req.getAsyncContext().complete();
|
||
|
} else if (req.isAsyncSupported()) {
|
||
|
AsyncContext actx = req.startAsync();
|
||
|
actx.addListener(this);
|
||
|
resp.setContentType("text/plain");
|
||
|
clients.add(actx);
|
||
|
if (clientcount.incrementAndGet()==1) {
|
||
|
Stockticker ticker = (Stockticker) req.getServletContext().getAttribute(
|
||
|
AsyncStockContextListener.STOCK_TICKER_KEY);
|
||
|
ticker.addTickListener(this);
|
||
|
}
|
||
|
} else {
|
||
|
new Exception("Async Not Supported").printStackTrace();
|
||
|
resp.sendError(400,"Async is not supported.");
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
@Override
|
||
|
public void tick(Stock stock) {
|
||
|
Iterator<AsyncContext> it = clients.iterator();
|
||
|
while (it.hasNext()) {
|
||
|
AsyncContext actx = it.next();
|
||
|
try {
|
||
|
writeStock(actx, stock);
|
||
|
} catch (Exception e) {
|
||
|
// Ignore. The async error handling will deal with this.
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
public void writeStock(AsyncContext actx, Stock stock) throws IOException {
|
||
|
HttpServletResponse response = (HttpServletResponse)actx.getResponse();
|
||
|
PrintWriter writer = response.getWriter();
|
||
|
writer.write("STOCK#");//make client parsing easier
|
||
|
writer.write(stock.getSymbol());
|
||
|
writer.write("#");
|
||
|
writer.write(stock.getValueAsString());
|
||
|
writer.write("#");
|
||
|
writer.write(stock.getLastChangeAsString());
|
||
|
writer.write("#");
|
||
|
writer.write(String.valueOf(stock.getCnt()));
|
||
|
writer.write("\n");
|
||
|
writer.flush();
|
||
|
response.flushBuffer();
|
||
|
}
|
||
|
|
||
|
|
||
|
@Override
|
||
|
public void shutdown() {
|
||
|
// The web application is shutting down. Complete any AsyncContexts
|
||
|
// associated with an active client.
|
||
|
Iterator<AsyncContext> it = clients.iterator();
|
||
|
while (it.hasNext()) {
|
||
|
AsyncContext actx = it.next();
|
||
|
try {
|
||
|
actx.complete();
|
||
|
} catch (Exception e) {
|
||
|
// Ignore. The async error handling will deal with this.
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
@Override
|
||
|
public void onComplete(AsyncEvent event) throws IOException {
|
||
|
if (clients.remove(event.getAsyncContext()) && clientcount.decrementAndGet()==0) {
|
||
|
ServletContext sc = event.getAsyncContext().getRequest().getServletContext();
|
||
|
Stockticker ticker = (Stockticker) sc.getAttribute(
|
||
|
AsyncStockContextListener.STOCK_TICKER_KEY);
|
||
|
ticker.removeTickListener(this);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
@Override
|
||
|
public void onError(AsyncEvent event) throws IOException {
|
||
|
event.getAsyncContext().complete();
|
||
|
}
|
||
|
|
||
|
@Override
|
||
|
public void onTimeout(AsyncEvent event) throws IOException {
|
||
|
event.getAsyncContext().complete();
|
||
|
}
|
||
|
|
||
|
|
||
|
@Override
|
||
|
public void onStartAsync(AsyncEvent event) throws IOException {
|
||
|
// NOOP
|
||
|
}
|
||
|
}
|