”工欲善其事,必先利其器。“—孔子《论语.录灵公》
首页 > 编程 > 使用 Deno 发送 Web 推送消息

使用 Deno 发送 Web 推送消息

发布于2024-08-01
浏览:481

Send Web Push messages with Deno

最近,我一直在开发一个仅使用 Web API 从头开始​​构建的新 webpush 包。这使得(至少在理论上)直接从浏览器发送 Web Push 消息成为可能。

这篇博文旨在解释什么是 Web Push 协议,它是如何工作的
(RFC 8291) 以及如何使用我的库发送 Web 推送消息。

什么是网络推送协议?

Web 推送协议是一种中间协议,允许应用程序向用户代理(通常是浏览器)发送消息。

它与服务器发送事件(SSE)类似,消息被推送到用户代理,但它有不同的目的。 Web 推送消息不需要网站作为服务工作者打开一个选项卡
可以监听推送消息。它在后台运行。

Web 推送协议如何工作?

Web Push 协议涉及三个参与者:

  • 用户代理:通常是您的网站访问者浏览器
  • 推送服务:由 Google、Mozilla 或 Apple 维护和拥有的推送服务器,具体取决于浏览器
  • 应用程序服务器:您的服务器

以下是它们之间交互的概述:

     -------             --------------         ------------- 
    |  UA   |           | Push Service |       | Application |
     -------             --------------         ------------- 
        |                      |                      |
        |        Setup         |                      |
        ||                      |
        |           Provide Subscription              |
        |-------------------------------------------->|
        |                      |                      |
        :                      :                      :
        |                      |     Push Message     |
        |    Push Message      |



由于多种原因需要中间推送服务。

首先,它减少了带宽和电池使用量,因为用户代理仅为所有网站维护一个连接,而不是每个网站一个连接。

它还提高了可扩展性和可靠性,因为主要浏览器的推送服务旨在处理数百万用户。由于如果用户代理离线,则必须保留推送消息,因此构建推送服务需要大量工程、弹性且冗余的基础设施

最后,对于小型网络公司来说,构建、部署和维护自定义推送服务通常过于复杂且耗费资源。这会给大公司带来不公平的竞争优势,因为他们将拥有必要的资源来开发和完善自己的推送服务。

如果您是像我一样关注隐私的用户,请参阅中介服务
接收所有消息都会引发危险信号。为了解决这个问题,Web Push
消息通过 HTTP 加密内容编码进行保护(请参阅我的
http-ece 包),确保
敏感信息仍然受到保护,任何第三方都无法读取
过境服务。

设置

您可能已经注意到设置箭头与上面 ASCII 图中的其他箭头不同。这是因为设置阶段依赖于实现。所有主流浏览器都实现了 javascript

中推送 API 不同的方式。返回标准
的 PushManager.subscribe() 方法 PushSubscription 已公开。

订阅始终包含与推送订阅关联的唯一 URL 端点以及用于加密消息的公钥。

创建订阅时,可以提供可选的applicationServerKey来标识应用服务器推送消息。这是自愿应用程序服务器标识 (VAPID) 身份验证方法
(RFC 8292)。 VAPID 密钥用于减轻对推送服务的 DDOS 攻击。此外,在应用程序服务器和推送服务之间添加身份验证可以降低泄露订阅端点的风险。由于这些原因,它们在 Firefox 中是强制性的。

提供订阅

第二步是将订阅发送到应用程序服务器,以便它可以开始发送消息。

应用程序服务器通常会将订阅存储在数据库中以供以后重用。

推送消息

最后,为了推送消息,应用程序服务器使用 vapid 身份验证方案发送加密的 HTTP 请求,前提是提供了 applicationServerKey 来创建订阅。

如果推送服务收到消息时用户代理在线,则
转发。否则,它将被存储,直到用户代理上线或消息过期。

当用户代理收到消息时,它会执行主要用于显示通知的推送事件处理程序。

使用 webpush 设置应用程序服务器

首先,您必须生成 VAPID 密钥,因为某些浏览器强制要求它们:

$ deno run https://raw.githubusercontent.com/negrel/webpush/master/cmd/generate-vapid-keys.ts

复制输出并将其保存到文件中,您不需要再次生成 VAPID 密钥。

在您的应用程序服务器代码中,您可以按如下方式加载它们:

import * as webpush from "jsr:@negrel/webpush";

// Read generated VAPID file.
const vapidKeysJson = Deno.readTextFileSync("./path/to/vapid.json");

// Import VAPID keys.
webpush.importVapidKeys(JSON.parse(vapidKeysJson));

然后,您需要创建一个ApplicationServer对象实例。

// adminEmail is used by Push services maintainer to contact you in case there
// is problem with your application server.
const adminEmail = "[email protected]";

// Create an application server object.
const appServer = await webpush.ApplicationServer.new({
  contactInformation: "mailto:"   adminEmail,
  vapidKeys,
});

然后要发送推送消息,只需创建一个 PushSubscriber 并调用它的
pushMessage()/pushTextMessage()方法如下:

const subsribers = [];

// HTTP handler for user agent sending their subscription.
function subscribeHandler(req) {
  // Extract subscription send by user agent.
  const subscription = await req.json();

  // Store subscription in db.
  // ...

  // Create a subscriber object.
  const sub = appServer.subscribe(subscription);

  // Store subscriber in memory.
  subscribers.push(sub);
}

// Helper method to send message to all subscribers.
function broadcastMessage(msg) {
  for (const sub of subscribes) {
    sub.pushTextMessage(msg, {});
  }
}

就是这样,您向订阅者发送推送消息!

webpush 存储库包含一个交互式示例,其中包含可以在本地运行的类似代码。它还包含客户端 JavaScript 代码,所以一定要检查一下!

版本声明 本文转载于:https://dev.to/negrel/send-web-push-messages-with-deno-1kcl?1如有侵犯,请联系[email protected]删除
最新教程 更多>
  • PHP与C++函数重载处理的区别
    PHP与C++函数重载处理的区别
    作为经验丰富的C开发人员脱离谜题,您可能会遇到功能超载的概念。这个概念虽然在C中普遍,但在PHP中构成了独特的挑战。让我们深入研究PHP功能过载的复杂性,并探索其提供的可能性。在PHP中理解php的方法在PHP中,函数超载的概念(如C等语言)不存在。函数签名仅由其名称定义,而与他们的参数列表无关。...
    编程 发布于2025-05-12
  • 为什么PYTZ最初显示出意外的时区偏移?
    为什么PYTZ最初显示出意外的时区偏移?
    与pytz 最初从pytz获得特定的偏移。例如,亚洲/hong_kong最初显示一个七个小时37分钟的偏移: 差异源利用本地化将时区分配给日期,使用了适当的时区名称和偏移量。但是,直接使用DateTime构造器分配时区不允许进行正确的调整。 example pytz.timezone(...
    编程 发布于2025-05-12
  • Go语言垃圾回收如何处理切片内存?
    Go语言垃圾回收如何处理切片内存?
    Garbage Collection in Go Slices: A Detailed AnalysisIn Go, a slice is a dynamic array that references an underlying array.使用切片时,了解垃圾收集行为至关重要,以避免潜在的内存泄...
    编程 发布于2025-05-12
  • 同实例无需转储复制MySQL数据库方法
    同实例无需转储复制MySQL数据库方法
    在同一实例上复制一个MySQL数据库而无需转储在同一mySQL实例上复制数据库,而无需创建InterMediate sqql script。以下方法为传统的转储和IMPORT过程提供了更简单的替代方法。 直接管道数据 MySQL手动概述了一种允许将mysqldump直接输出到MySQL clie...
    编程 发布于2025-05-12
  • Spark DataFrame添加常量列的妙招
    Spark DataFrame添加常量列的妙招
    在Spark Dataframe ,将常数列添加到Spark DataFrame,该列具有适用于所有行的任意值的Spark DataFrame,可以通过多种方式实现。使用文字值(SPARK 1.3)在尝试提供直接值时,用于此问题时,旨在为此目的的column方法可能会导致错误。 df.withCo...
    编程 发布于2025-05-12
  • 如何克服PHP的功能重新定义限制?
    如何克服PHP的功能重新定义限制?
    克服PHP的函数重新定义限制 但是,PHP工具腰带中有一个隐藏的宝石:runkit扩展。它使您能够灵活地重新定义函数。 runkit_function_renction_rename() runkit_function_redefine() //重新定义'this'以返回“新和改...
    编程 发布于2025-05-12
  • 如何使用node-mysql在单个查询中执行多个SQL语句?
    如何使用node-mysql在单个查询中执行多个SQL语句?
    Multi-Statement Query Support in Node-MySQLIn Node.js, the question arises when executing multiple SQL statements in a single query using the node-mys...
    编程 发布于2025-05-12
  • JavaScript计算两个日期之间天数的方法
    JavaScript计算两个日期之间天数的方法
    How to Calculate the Difference Between Dates in JavascriptAs you attempt to determine the difference between two dates in Javascript, consider this s...
    编程 发布于2025-05-12
  • 在GO中构造SQL查询时,如何安全地加入文本和值?
    在GO中构造SQL查询时,如何安全地加入文本和值?
    在go中构造文本sql查询时,在go sql queries 中,在使用conting and contement和contement consem per时,尤其是在使用integer per当per当per时,per per per当per. [&​​&&&&&&&&&&&&&&&默元组方法在...
    编程 发布于2025-05-12
  • 如何使用Python有效地以相反顺序读取大型文件?
    如何使用Python有效地以相反顺序读取大型文件?
    在python 中,如果您使用一个大文件,并且需要从最后一行读取其内容,则在第一行到第一行,Python的内置功能可能不合适。这是解决此任务的有效解决方案:反向行读取器生成器 == ord('\ n'): 缓冲区=缓冲区[:-1] ...
    编程 发布于2025-05-12
  • 为什么HTML无法打印页码及解决方案
    为什么HTML无法打印页码及解决方案
    无法在html页面上打印页码? @page规则在@Media内部和外部都无济于事。 HTML:Customization:@page { margin: 10%; @top-center { font-family: sans-serif; font-weight: bo...
    编程 发布于2025-05-12
  • 如何有效地选择熊猫数据框中的列?
    如何有效地选择熊猫数据框中的列?
    在处理数据操作任务时,在Pandas DataFrames 中选择列时,选择特定列的必要条件是必要的。在Pandas中,选择列的各种选项。选项1:使用列名 如果已知列索引,请使用ILOC函数选择它们。请注意,python索引基于零。 df1 = df.iloc [:,0:2]#使用索引0和1 c...
    编程 发布于2025-05-12
  • 如何在无序集合中为元组实现通用哈希功能?
    如何在无序集合中为元组实现通用哈希功能?
    在未订购的集合中的元素要纠正此问题,一种方法是手动为特定元组类型定义哈希函数,例如: template template template 。 struct std :: hash { size_t operator()(std :: tuple const&tuple)const {...
    编程 发布于2025-05-12
  • MySQL中如何高效地根据两个条件INSERT或UPDATE行?
    MySQL中如何高效地根据两个条件INSERT或UPDATE行?
    在两个条件下插入或更新或更新 solution:的答案在于mysql的插入中...在重复键更新语法上。如果不存在匹配行或更新现有行,则此功能强大的功能可以通过插入新行来进行有效的数据操作。如果违反了唯一的密钥约束。实现所需的行为,该表必须具有唯一的键定义(在这种情况下为'名称'...
    编程 发布于2025-05-12
  • 如何使用不同数量列的联合数据库表?
    如何使用不同数量列的联合数据库表?
    合并列数不同的表 当尝试合并列数不同的数据库表时,可能会遇到挑战。一种直接的方法是在列数较少的表中,为缺失的列追加空值。 例如,考虑两个表,表 A 和表 B,其中表 A 的列数多于表 B。为了合并这些表,同时处理表 B 中缺失的列,请按照以下步骤操作: 确定表 B 中缺失的列,并将它们添加到表的末...
    编程 发布于2025-05-12

免责声明: 提供的所有资源部分来自互联网,如果有侵犯您的版权或其他权益,请说明详细缘由并提供版权或权益证明然后发到邮箱:[email protected] 我们会第一时间内为您处理。

Copyright© 2022 湘ICP备2022001581号-3