| /* |
| * 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.concurrent.ConcurrentLinkedQueue; |
| import java.util.concurrent.atomic.AtomicInteger; |
| |
| import jakarta.servlet.AsyncContext; |
| import jakarta.servlet.AsyncEvent; |
| import jakarta.servlet.AsyncListener; |
| import jakarta.servlet.ServletContext; |
| import jakarta.servlet.ServletException; |
| import jakarta.servlet.http.HttpServlet; |
| import jakarta.servlet.http.HttpServletRequest; |
| import jakarta.servlet.http.HttpServletResponse; |
| |
| import org.apache.juli.logging.Log; |
| import org.apache.juli.logging.LogFactory; |
| |
| 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 Log log = LogFactory.getLog(AsyncStockServlet.class); |
| |
| private static final ConcurrentLinkedQueue<AsyncContext> clients = |
| new ConcurrentLinkedQueue<>(); |
| private static final AtomicInteger clientcount = new AtomicInteger(0); |
| |
| public AsyncStockServlet() { |
| log.info("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) { |
| for (AsyncContext actx : clients) { |
| 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. |
| for (AsyncContext actx : clients) { |
| 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 |
| } |
| } |