跳到主要内容

protocol

注册自定义协议并拦截现有协议请求。

进程: 主进程

实现与 file:// 协议具有相同效果的协议的示例

const { app, protocol, net } = require('electron')
const path = require('node:path')
const url = require('node:url')

app.whenReady().then(() => {
protocol.handle('atom', (request) => {
const filePath = request.url.slice('atom://'.length)
return net.fetch(url.pathToFileURL(path.join(__dirname, filePath)).toString())
})
})

注意: 除非另有说明,所有方法只能在 app 模块的 ready 事件发出后使用。

protocol 与自定义 partitionsession 结合使用

协议注册到特定的 Electron session 对象。 如果您未指定会话,则您的 protocol 将应用于 Electron 使用的默认会话。 但是,如果您在 browserWindowwebPreferences 上定义了 partitionsession,则该窗口将使用不同的会话,并且如果您仅使用 electron.protocol.XXX,则您的自定义协议将不起作用。

为了使您的自定义协议与自定义会话结合使用,您需要将其显式注册到该会话。

const { app, BrowserWindow, net, protocol, session } = require('electron')
const path = require('node:path')
const url = require('url')

app.whenReady().then(() => {
const partition = 'persist:example'
const ses = session.fromPartition(partition)

ses.protocol.handle('atom', (request) => {
const filePath = request.url.slice('atom://'.length)
return net.fetch(url.pathToFileURL(path.resolve(__dirname, filePath)).toString())
})

const mainWindow = new BrowserWindow({ webPreferences: { partition } })
})

方法

protocol 模块具有以下方法

protocol.registerSchemesAsPrivileged(customSchemes)

注意: 此方法只能在 app 模块的 ready 事件发出之前使用,并且只能调用一次。

scheme 注册为标准、安全,绕过资源的 内容安全策略,允许注册 ServiceWorker,支持 fetch API、流媒体视频/音频和 V8 代码缓存。 使用值 true 指定权限以启用该功能。

注册特权 scheme 的示例,该 scheme 绕过内容安全策略

const { protocol } = require('electron')
protocol.registerSchemesAsPrivileged([
{ scheme: 'foo', privileges: { bypassCSP: true } }
])

标准 scheme 遵循 RFC 3986 所谓的 通用 URI 语法。 例如,httphttps 是标准 scheme,而 file 不是。

将 scheme 注册为标准允许在提供服务时正确解析相对和绝对资源。 否则,scheme 的行为将类似于 file 协议,但不具备解析相对 URL 的能力。

例如,当您使用自定义协议加载以下页面而未将其注册为标准 scheme 时,图像将不会加载,因为非标准 scheme 无法识别相对 URL

<body>
<img src='test.png'>
</body>

将 scheme 注册为标准将允许通过 FileSystem API 访问文件。 否则,渲染器将为该 scheme 抛出安全错误。

默认情况下,Web 存储 API(localStorage、sessionStorage、webSQL、indexedDB、cookies)对于非标准 scheme 处于禁用状态。 因此,通常,如果您想注册自定义协议来替换 http 协议,则必须将其注册为标准 scheme。

使用流的协议(http 和 stream 协议)应设置 stream: true<video><audio> HTML 元素期望协议默认缓冲其响应。 stream 标志配置这些元素以正确期望流式响应。

protocol.handle(scheme, handler)

  • scheme string - 要处理的 scheme,例如 httpsmy-app。 这是 URL 中 : 之前的位。
  • handler Function<GlobalResponse | Promise<GlobalResponse>>

scheme 注册协议处理程序。 对使用此 scheme 的 URL 发出的请求将委托给此处理程序,以确定应发送什么响应。

可以返回 ResponsePromise<Response>

示例

const { app, net, protocol } = require('electron')
const path = require('node:path')
const { pathToFileURL } = require('url')

protocol.registerSchemesAsPrivileged([
{
scheme: 'app',
privileges: {
standard: true,
secure: true,
supportFetchAPI: true
}
}
])

app.whenReady().then(() => {
protocol.handle('app', (req) => {
const { host, pathname } = new URL(req.url)
if (host === 'bundle') {
if (pathname === '/') {
return new Response('<h1>hello, world</h1>', {
headers: { 'content-type': 'text/html' }
})
}
// NB, this checks for paths that escape the bundle, e.g.
// app://bundle/../../secret_file.txt
const pathToServe = path.resolve(__dirname, pathname)
const relativePath = path.relative(__dirname, pathToServe)
const isSafe = relativePath && !relativePath.startsWith('..') && !path.isAbsolute(relativePath)
if (!isSafe) {
return new Response('bad', {
status: 400,
headers: { 'content-type': 'text/html' }
})
}

return net.fetch(pathToFileURL(pathToServe).toString())
} else if (host === 'api') {
return net.fetch('https://api.my-server.com/' + pathname, {
method: req.method,
headers: req.headers,
body: req.body
})
}
})
})

有关更多详细信息,请参阅 MDN 文档中的 RequestResponse

protocol.unhandle(scheme)

  • scheme string - 要移除处理程序的 scheme。

移除使用 protocol.handle 注册的协议处理程序。

protocol.isProtocolHandled(scheme)

  • scheme string

返回 boolean - scheme 是否已被处理。

protocol.registerFileProtocol(scheme, handler) 已弃用

历史记录

返回 boolean - 协议是否成功注册

注册 scheme 协议,该协议将发送文件作为响应。 将使用 requestcallback 调用 handler,其中 request 是针对 scheme 的传入请求。

要处理 request,应使用文件路径或具有 path 属性的对象调用 callback,例如 callback(filePath)callback({ path: filePath })filePath 必须是绝对路径。

默认情况下,scheme 被视为类似于 http:,这与遵循“通用 URI 语法”(如 file:)的协议的解析方式不同。

protocol.registerBufferProtocol(scheme, handler) 已弃用

历史记录

返回 boolean - 协议是否成功注册

注册 scheme 协议,该协议将发送 Buffer 作为响应。

用法与 registerFileProtocol 相同,不同之处在于应使用 Buffer 对象或具有 data 属性的对象调用 callback

示例

protocol.registerBufferProtocol('atom', (request, callback) => {
callback({ mimeType: 'text/html', data: Buffer.from('<h5>Response</h5>') })
})

protocol.registerStringProtocol(scheme, handler) 已弃用

历史记录

返回 boolean - 协议是否成功注册

注册 scheme 协议,该协议将发送 string 作为响应。

用法与 registerFileProtocol 相同,不同之处在于应使用 string 或具有 data 属性的对象调用 callback

protocol.registerHttpProtocol(scheme, handler) 已弃用

历史记录

返回 boolean - 协议是否成功注册

注册 scheme 协议,该协议将发送 HTTP 请求作为响应。

用法与 registerFileProtocol 相同,不同之处在于应使用具有 url 属性的对象调用 callback

protocol.registerStreamProtocol(scheme, handler) 已弃用

历史记录

返回 boolean - 协议是否成功注册

注册 scheme 协议,该协议将发送流作为响应。

用法与 registerFileProtocol 相同,不同之处在于应使用 ReadableStream 对象或具有 data 属性的对象调用 callback

示例

const { protocol } = require('electron')
const { PassThrough } = require('stream')

function createStream (text) {
const rv = new PassThrough() // PassThrough is also a Readable stream
rv.push(text)
rv.push(null)
return rv
}

protocol.registerStreamProtocol('atom', (request, callback) => {
callback({
statusCode: 200,
headers: {
'content-type': 'text/html'
},
data: createStream('<h5>Response</h5>')
})
})

可以传递任何实现可读流 API(发出 data/end/error 事件)的对象。 例如,以下是如何返回文件的方法

protocol.registerStreamProtocol('atom', (request, callback) => {
callback(fs.createReadStream('index.html'))
})

protocol.unregisterProtocol(scheme) 已弃用

历史记录
  • scheme string

返回 boolean - 协议是否成功注销

注销 scheme 的自定义协议。

protocol.isProtocolRegistered(scheme) 已弃用

历史记录
  • scheme string

返回 boolean - scheme 是否已被注册。

protocol.interceptFileProtocol(scheme, handler) 已弃用

历史记录

返回 boolean - 协议是否成功被拦截

拦截 scheme 协议,并使用 handler 作为协议的新处理程序,该处理程序发送文件作为响应。

protocol.interceptStringProtocol(scheme, handler) 已弃用

历史记录

返回 boolean - 协议是否成功被拦截

拦截 scheme 协议,并使用 handler 作为协议的新处理程序,该处理程序发送 string 作为响应。

protocol.interceptBufferProtocol(scheme, handler) 已弃用

历史记录

返回 boolean - 协议是否成功被拦截

拦截 scheme 协议,并使用 handler 作为协议的新处理程序,该处理程序发送 Buffer 作为响应。

protocol.interceptHttpProtocol(scheme, handler) 已弃用

历史记录

返回 boolean - 协议是否成功被拦截

拦截 scheme 协议,并使用 handler 作为协议的新处理程序,该处理程序发送新的 HTTP 请求作为响应。

protocol.interceptStreamProtocol(scheme, handler) 已弃用

历史记录

返回 boolean - 协议是否成功被拦截

protocol.registerStreamProtocol 相同,不同之处在于它替换了现有的协议处理程序。

protocol.uninterceptProtocol(scheme) 已弃用

历史记录
  • scheme string

返回 boolean - 协议是否成功取消拦截

移除为 scheme 安装的拦截器,并恢复其原始处理程序。

protocol.isProtocolIntercepted(scheme) 已弃用

历史记录
  • scheme string

返回 boolean - scheme 是否已被拦截。