duanjipiao7076 2018-12-11 06:18
浏览 885
已采纳

通过websocket将kubernetes日志公开到浏览器

I am trying to use sidecar mode in kubernetes to create a logs sidecar to expose specific container logs. And I am using kubernetes client to fetch logs from kubernetes api and send it out by websocket. The code shows below:

func serveWs(w http.ResponseWriter, r *http.Request) {
    w.Header().Set("Access-Control-Allow-Origin", "*")
    conn, err := upgrader.Upgrade(w, r, nil)
    if err != nil {
        if _, ok := err.(websocket.HandshakeError); !ok {
            log.Println(err)
        }
        return
    }

    defer conn.Close()

    logsClient, err := InitKubeLogsClient(config.InCluster)
    if err != nil {
        log.Fatalln(err)
    }

    stream, err := logsClient.GetLogs(config.Namespace, config.PodName, config.ContainerName)
    if err != nil {
        log.Fatalln(err)
    }
    defer stream.Close()
    reader := bufio.NewReader(stream)

    for {
        line, err := reader.ReadString('
')
        if err != nil {
            log.Fatalln(err)
        }
        conn.WriteMessage(websocket.TextMessage, []byte(line))
    }
}

I am using https://github.com/gorilla/websocket as the websocket lib. And on the browser

Is this the best way to do what I want? Is there some better way to just expose the logs api from k8s to websocket?

  • 写回答

1条回答 默认 最新

  • duanqiao1926 2018-12-29 00:10
    关注

    Put my final code here, thanks for the tips from @Peter:

    func serveWs(w http.ResponseWriter, r *http.Request) {
        w.Header().Set("Access-Control-Allow-Origin", "*")
        conn, err := upgrader.Upgrade(w, r, nil)
        if err != nil {
            if _, ok := err.(websocket.HandshakeError); !ok {
                log.Println(err)
            }
            return
        }
    
        log.Println("create new connection")
    
        defer func() {
            conn.Close()
            log.Println("connection close")
        }()
    
        logsClient, err := InitKubeLogsClient(config.InCluster)
        if err != nil {
            log.Println(err)
            return
        }
    
        stream, err := logsClient.GetLogs(config.Namespace, config.PodName, config.ContainerName)
        if err != nil {
            log.Println(err)
            return
        }
        defer stream.Close()
        reader := bufio.NewReaderSize(stream, 16)
        lastLine := ""
        for {
            data, isPrefix, err := reader.ReadLine()
            if err != nil {
                log.Println(err)
                return
            }
    
            lines := strings.Split(string(data), "")
    
            length := len(lines)
    
            if len(lastLine) > 0 {
                lines[0] = lastLine + lines[0]
                lastLine = ""
            }
    
            if isPrefix {
                lastLine = lines[length-1]
                lines = lines[:(length - 1)]
            }
    
            for _, line := range lines {
                if err := conn.WriteMessage(websocket.TextMessage, []byte(line)); err != nil {
                    log.Println(err)
                    return
                }
            }
        }
    }
    
    本回答被题主选为最佳回答 , 对您是否有帮助呢?
    评论

报告相同问题?

悬赏问题

  • ¥15 求daily translation(DT)偏差订正方法的代码
  • ¥15 js调用html页面需要隐藏某个按钮
  • ¥15 ads仿真结果在圆图上是怎么读数的
  • ¥20 Cotex M3的调试和程序执行方式是什么样的?
  • ¥20 java项目连接sqlserver时报ssl相关错误
  • ¥15 一道python难题3
  • ¥15 牛顿斯科特系数表表示
  • ¥15 arduino 步进电机
  • ¥20 程序进入HardFault_Handler
  • ¥15 关于#python#的问题:自动化测试