spring-boot框架下的websocket服務

        這幾天在作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;
}
相關文章
相關標籤/搜索