0

0

使用 Go、HTMX 和 Web Socket 构建简单的实时系统监视器

霞舞

霞舞

发布时间:2024-11-17 12:30:21

|

730人浏览过

|

来源于dev.to

转载

我正在寻找一个可以与 go、htmx 和 tailwwindcss 配合使用的有趣项目,最终利用 web 套接字的功能构建了一个简单的基于 web 的实时系统监视器。这是结果。

使用 Go、HTMX 和 Web Socket 构建简单的实时系统监视器

它显示系统信息、内存、磁盘、cpu 和正在运行的进程,每 5 秒自动更新一次。

我将在这篇文章中对代码进行一些分解。

堆栈

  • 去1.23.2
  • htmx
  • tailwindcss
  • gopsutil
  • websocket
  • htmx websocket 扩展

http服务器

type httpserver struct {
    subscribermessagebuffer int
    mux                     http.servemux
    subscribersmutex        sync.mutex
    subscribers             map[*subscriber]struct{}
}

type subscriber struct {
    msgs chan []byte
}

这非常简单。 httpserver 包含一个 http.servemux 作为 http 处理程序和稍后用于 web 套接字广播的订阅者。订阅者只是有用于数据更新的消息通道。

由于它只需要提供单个 html 文件,因此它需要 url 来显示页面,以及一个用于 web 套接字连接的 url。

func newhttpserver() *httpserver {
    s := &httpserver{
        subscribermessagebuffer: 10,
        subscribers:             make(map[*subscriber]struct{}),
    }

    s.mux.handle("/", http.fileserver(http.dir("./views")))
    s.mux.handlefunc("/ws", s.subscribehandler)
    return s
}

web socket 连接和订阅者

端点 /ws 将处理 web 套接字连接并管理订阅者。首先,它将启动一个新订阅者并将其添加到 http 服务器结构中的映射中。锁将用于防止竞争条件,因为我们稍后将使用 go 例程。

func (s *httpserver) subscribehandler(w http.responsewriter, r *http.request) {
    err := s.subscribe(r.context(), w, r)
    if err != nil {
        fmt.println(err)
        return
    }
}

func (s *httpserver) addsubscriber(subscriber *subscriber) {
    s.subscribersmutex.lock()
    s.subscribers[subscriber] = struct{}{}
    s.subscribersmutex.unlock()
    fmt.println("subscriber added", subscriber)
}

web 套接字开始接受连接,并通过循环,我们将检测来自订阅者的传入通道消息并将其写入 web 套接字。

func (s *httpserver) subscribe(ctx context.context, w http.responsewriter, r *http.request) error {
    var c *websocket.conn
    subscriber := &subscriber{
        msgs: make(chan []byte, s.subscribermessagebuffer),
    }

    s.addsubscriber(subscriber)

    c, err := websocket.accept(w, r, nil)
    if err != nil {
        return err
    }

    defer c.closenow()

    ctx = c.closeread(ctx)
    for {
        select {
        case msg := <-subscriber.msgs:
            ctx, cancel := context.withtimeout(ctx, time.second)
            defer cancel()
            err := c.write(ctx, websocket.messagetext, msg)
            if err != nil {
                return err
            }
        case <-ctx.done():
            return ctx.err()
        }
    }
}

自动更新

自动更新系统信息数据由 goroutine 处理。我们将构建一个 html 响应,该响应将通过 web 套接字发送,htmx 将处理 html 端的更新。

Transor
Transor

专业的AI翻译工具,支持网页、字幕、PDF、图片实时翻译

下载
func main() {
    fmt.println("starting system monitor")
    s := server.newhttpserver()

    go func(s *server.httpserver) {
        for {
            hoststat, _ := host.info()
            timestamp := time.now().format("2006-01-02 15:04:05")
            html := `
            ` + timestamp + `
            ` + hoststat.hostname + `
            ` + hoststat.os + `
            `
            s.broadcast([]byte(html))
            time.sleep(time.second * 5)
        }
    }(s)
    // ...
}

htmx 中的语法 hx-swap-oob="innerhtml:#data-timestamp" 告诉我们在 html 中交换 data-timestamp id 内的组件。所有交换机制对于其他系统信息组件都是相同的。

所有可交换的 html 组件将通过 broadcast(msg) 方法发送,稍后将每 5 秒通过通道发送一次。

func (s *httpserver) broadcast(msg []byte) {
    s.subscribersmutex.lock()
    for subscriber := range s.subscribers {
        subscriber.msgs <- msg
    }
    s.subscribersmutex.unlock()
}

htmx 视图

这是纯 html 文件,对于 tailwindcss 我简单地使用了 cdn


htmx 和使用 cdn 的 web 套接字扩展的想法相同。



如何连接到网络套接字?

系统监视器页面预计通过 web 套接字接收所有数据,以便我可以从主 div 容器中设置它。指定 hx-ext=”ws” 告诉 htmx 使用 web 套接字扩展,指定 ws-connect=”/ws” 告诉 web 套接字通过 /ws url 连接。


    

完整代码

这是代码的完整版本 https://github.com/didikz/gosysmon-web,您可能想尝试一下自己的版本。

编码愉快!

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

398

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

575

2023.08.10

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

398

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

575

2023.08.10

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

398

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

575

2023.08.10

github中文官网入口 github中文版官网网页进入
github中文官网入口 github中文版官网网页进入

github中文官网入口https://docs.github.com/zh/get-started,GitHub 是一种基于云的平台,可在其中存储、共享并与他人一起编写代码。 通过将代码存储在GitHub 上的“存储库”中,你可以: “展示或共享”你的工作。 持续“跟踪和管理”对代码的更改。

1085

2026.01.21

http500解决方法
http500解决方法

http500解决方法有检查服务器日志、检查代码错误、检查服务器配置、检查文件和目录权限、检查资源不足、更新软件版本、重启服务器或寻求专业帮助等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

434

2023.11.09

漫画免费在线观看地址大全
漫画免费在线观看地址大全

想找免费又资源丰富的漫画网站?本合集精选2025-2026年热门平台,涵盖国漫、日漫、韩漫等多类型作品,支持高清流畅阅读与离线缓存。阅读专题下面的文章了解更多详细内容。

1

2026.01.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Sass 教程
Sass 教程

共14课时 | 0.8万人学习

Bootstrap 5教程
Bootstrap 5教程

共46课时 | 3.1万人学习

CSS教程
CSS教程

共754课时 | 25.4万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号