這幾天在作web端實時展現服務端日誌文件新增內容的功能。要知足實時的需求,我選擇的方案是在web端跟服務端創建一個websocket連接,由服務端經過tail -f 命令將文件新增內容發送給web端。html
關於websocket的介紹,能夠參考這篇博文:http://www.cnblogs.com/lizhenghn/p/5155933.html(連接僅用於學習交流,若有版權問題請及時告知)。這裏我主要想介紹的是在spring-boot框架下如何發佈websocket服務。java
1、在服務端發佈websocket服務nginx
服務端發佈websocket服務有幾種方式,包括Servlet容器掃描初始化、Spring掃描初始化。我使用的是第二種方式,能夠將websocket服務定義爲一個單獨的類實例。
web
Spring掃描初始化時,須要先定義一個Bean:ServerEndpointExporter,以聲明服務端。我把這個Bean獨立放到一個websocket 配置類中。spring
import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.web.socket.server.standard.ServerEndpointExporter; @Configuration public class WebSocketConfig { @Bean public ServerEndpointExporter serverEndpointExporter (){ return new ServerEndpointExporter(); } }
接下來是定義websocket服務接口,並使用關鍵字 @ServerEndpoint("/websocketPath") 聲明一個接口的訪問路徑。websocket
import java.io.BufferedReader; import java.io.File; import java.io.IOException; import java.io.InputStream; import java.io.InputStreamReader; import java.util.concurrent.CopyOnWriteArraySet; import javax.websocket.OnClose; import javax.websocket.OnError; import javax.websocket.OnMessage; import javax.websocket.OnOpen; import javax.websocket.Session; import javax.websocket.server.ServerEndpoint; import org.springframework.stereotype.Component; import org.springframework.web.bind.annotation.RestController; @ServerEndpoint("/logWebsocket") @Component @RestController public class LogWebSocketHandle { private Session session;//每一個websocket會話鏈接對應一個session private static CopyOnWriteArraySet<LogWebSocketHandle> webSocketSet = new CopyOnWriteArraySet<>(); /** * 新的WebSocket請求開啓。 * 新創建鏈接後會觸發onOpen方法 * @throws JSchException * @throws IOException */ @OnOpen public void onOpen(Session session) throws JSchException, IOException { this.session = session; webSocketSet.add(this); //服務端保留session信息,並返回結果給客戶端 //這個語句用於服務端給客戶端發送數據 session.getBasicRemote().sendText("正在獲取日誌<br>"); } /** * WebSocket請求關閉。 * websocket鏈接關閉後,會觸發onClose方法 */ @OnClose public void onClose() { webSocketSet.remove(this); } @OnError public void onError(Throwable thr) { thr.printStackTrace(); } /** * 客戶端發送消息時,服務端經過onMessage方法接收 */ @OnMessage public void onMessage (String message, Session session) throws IOException, JSchException, InterruptedException { LOG.info("來自客戶端的message:" + message); try { //process message //TODO } catch (IOException e) { e.printStackTrace(); } // 給客戶端羣發消息 // for ( Session item : webSocketSet ){ // item.getBasicRemote().sendText(message); // } } }
2、web端創建websocket鏈接
session
var websocket_host = window.location.host; //與服務端創建websocket鏈接 var websocket = new WebSocket("ws://"+websocket_host+"/項目名/logWebsocket"); //鏈接創建後,會觸發onopen方法 websocket.onopen = function(event){ console.log("opened!"); $("#chart_multiple div").append(event.data); //向服務端發送數據 websocket.send(message); }; //接收服務端的數據 websocket.onmessage = function(event){ $("#chart_multiple div").append(event.data); $("#chart_multiple").scrollTop( $("#chart_multiple div").height()-$("#chart_multiple").height() ); }
3、使用nginx轉發時的額外配置app
若是項目使用了nginx進行負載均衡,那麼須要在nginx.conf配置文件中添加一個websocket轉發配置,具體爲:
負載均衡
location /#{websocket所在的項目名}/logWebsocket { proxy_pass http://ip:port/#{websocket所在的項目名}/logWebsocket; proxy_set_header Upgrade $http_upgrade; proxy_set_header Connection "upgrade"; proxy_read_timeout 7200s; }