vue3 实时通讯 SSE

javascript 复制代码
/**
 * 原生 EventSource 轻量封装
 * 自动重连 & 任意事件监听
 * 支持自定义请求头(通过 URL 参数传递 Authorization)
 */
export default class SSE {
  private url: string;
  private es: EventSource | null;
  private retry: number;
  private headers?: Record<string, string>;

  constructor(url: string, headers?: Record<string, string>) {
    this.url = url
    this.es = null
    this.retry = 3000 // 重连间隔 ms
    this.headers = headers
    this.connect()
  }

  /* 建立连接 */
  connect(): void {
    // 如果有 headers,将 Authorization 添加到 URL 中(因为 EventSource 不支持自定义 headers)
    let finalUrl = this.url;
    if (this.headers?.Authorization) {
      const token = this.headers.Authorization.replace('Bearer ', '');
      const separator = this.url.includes('?') ? '&' : '?';
      finalUrl = `${this.url}${separator}authorization=${encodeURIComponent(token)}`;
    }

    this.es = new EventSource(finalUrl)

    this.es.addEventListener('open', () => {
      console.log('[SSE] connected')
    })

    this.es.addEventListener('error', () => {
      console.log('[SSE] disconnected, retrying...')
      this.es?.close()
      setTimeout(() => this.connect(), this.retry)
    })
  }

  /**
   * 订阅任意后端事件
   * @param {string} event  事件名(与后端 event:xxx 对应)
   * @param {Function} cb   回调 (data: any) => {}
   */
  subscribe<T = any>(event: string, cb: (data: T) => void): void {
    this.es?.addEventListener(event, (e: MessageEvent) => {
      try {
        const data = JSON.parse(e.data)
        cb(data)
      } catch (err) {
        console.error('[SSE] parse error', err)
      }
    })
  }

  /* 手动关闭连接 */
  close(): void {
    this.es?.close()
    console.log('[SSE] closed by client')
  }
}
javascript 复制代码
import { defineStore } from 'pinia';

// 生成唯一 clientId
function generateClientId(): string {
  const stored = localStorage.getItem('sse_client_id');
  if (stored) return stored;
  const newId = `client_${Date.now()}_${Math.random().toString(36).substr(2, 9)}`;
  localStorage.setItem('sse_client_id', newId);
  return newId;
}



export const useNotificationStore = defineStore('notification', {
  state: () => ({
    sse: null,
    // ...
  }),
  
  actions: {
    async initSSE() {
      if (this.sse) return;

      const clientId = generateClientId();

      const token = localStorage.getItem('token');
      if (!token) {
        console.error('[NotificationStore] No token found, cannot init SSE');
        return;
      }

      // 建立 SSE 连接,传递 Authorization 请求头
      const sseUrl = `xxx/xxx/xx?clientId=${clientId}`;
      this.sse = new SSE(sseUrl, {
        Authorization: `${token}`
      });

      /* 任务状态事件 */
      this.sse.subscribe<{ code: number; data: any; message: string }>('sse_task_status', (payload) => {
        console.log('sse监听-----------------------', payload)
        if (payload.code !== 200) return;

        // payload.data为实时数据
        // 取出数据放入store

     
      });
    },
    // 断开 SSE 连接
    disconnectSSE() {
      if (this.sse) {
        this.sse.close();
        this.sse = null;
        console.log('[NotificationStore] SSE connection closed');
      }
    },
  },
});
javascript 复制代码
// 建立连接
notificationStore.initSSE()

// 断开连接
notificationStore.disconnectSSE()
相关推荐
qq_437100665 天前
SSE 流式响应(Server-Sent Events)
前端·sse
你脸上有BUG6 天前
SSE库选型+fetch-event-source示例
前端·sse·通知订阅
赵庆明老师8 天前
vben开发入门6:tsconfig.json
json·vue3·vben
赵庆明老师8 天前
vben开发入门5:vite.config.ts
前端·html·vue3·vben
弹简特8 天前
【AI辅助趣学SpringAI】03-聊天模型之SSE流式编程
人工智能·sse·springai
沙振宇9 天前
【Web】使用Vue3+PlayCanvas开发3D游戏(十二)渲染PCD点云可视化模型
3d·vue3·点云·pcd
花千树-0109 天前
MCP HTTP 传输详解:比 SSE 简单,但有一个意外的坑
java·agent·sse·function call·ai agent·mcp·harness
是席木木啊12 天前
告别console.log!Vue3项目日志框架选型指南
前端·vue3·日志框架
程序员-南12 天前
解决 Vue3 中 keep-alive 缓存问题的方法
缓存·vue3
qq_120840937113 天前
Three.js 模型加载稳定性实战:从资源失败到可用发布的工程化方案
前端·javascript·vue.js·vue3·three.js