const assert = require('assert')
const http = require('http')
const path = require('path')
const qs = require('querystring')
const {closeWindow} = require('./window-helpers')
const {remote} = require('electron')
const {BrowserWindow, ipcMain, protocol, session, webContents} = remote
// The RPC API doesn't seem to support calling methods on remote objects very
// well. In order to test stream protocol, we must work around this limitation
// and use Stream instances created in the browser process.
const stream = remote.require('stream')
/* The whole protocol API doesn't use standard callbacks */
/* eslint-disable standard/no-callback-literal */
describe('protocol module', () => {
const protocolName = 'sp'
const text = 'valar morghulis'
const postData = {
name: 'post test',
type: 'string'
}
function delay (ms) {
return new Promise((resolve) => {
setTimeout(resolve, ms)
})
}
function getStream (chunkSize = text.length, data = text) {
const body = stream.PassThrough()
async function sendChunks () {
let buf = Buffer.from(data)
for (;;) {
body.push(buf.slice(0, chunkSize))
buf = buf.slice(chunkSize)
if (!buf.length) {
break
}
// emulate network delay
await delay(50)
}
body.push(null)
}
sendChunks()
return body
}
afterEach((done) => {
protocol.unregisterProtocol(protocolName, () => {
protocol.uninterceptProtocol('http', () => done())
})
})
describe('protocol.register(Any)Protocol', () => {
const emptyHandler = (request, callback) => callback()
it('throws error when scheme is already registered', (done) => {
protocol.registerStringProtocol(protocolName, emptyHandler, (error) => {
assert.equal(error, null)
protocol.registerBufferProtocol(protocolName, emptyHandler, (error) => {
assert.notEqual(error, null)
done()
})
})
})
it('does not crash when handler is called twice', (done) => {
const doubleHandler = (request, callback) => {
try {
callback(text)
callback()
} catch (error) {
// Ignore error
}
}
protocol.registerStringProtocol(protocolName, doubleHandler, (error) => {
if (error) return done(error)
$.ajax({
url: protocolName + '://fake-host',
cache: false,
success: (data) => {
assert.equal(data, text)
done()
},
error: (xhr, errorType, error) => done(error)
})
})
})
it('sends error when callback is called with nothing', (done) => {
protocol.registerBufferProtocol(protocolName, emptyHandler, (error) => {
if (error) return done(error)
$.ajax({
url: protocolName + '://fake-host',
cache: false,
success: () => done('request succeeded but it should not'),
error: (xhr, errorType) => {
assert.equal(errorType, 'error')
return done()
}
})
})
})
it('does not crash when callback is called in next tick', (done) => {
const handler = (request, callback) => {
setImmediate(() => callback(text))
}
protocol.registerStringProtocol(protocolName, handler, (error) => {
if (error) return done(error)
$.ajax({
url: protocolName + '://fake-host',
cache: false,
success: (data) => {
assert.equal(data, text)
done()
},
error: (xhr, errorType, error) => done(error)
})
})
})
})
describe('protocol.unregisterProtocol', () => {
it('returns error when scheme does not exist', (done) => {
protocol.unregisterProtocol('not-exist', (error) => {
assert.notEqual(error, null)
done()
})
})
})
describe('protocol.registerStringProtocol', () => {
it('sends string as response', (done) => {
const handler = (request, callback) => callback(text)
protocol.registerStringProtocol(protocolName, handler, (error) => {
if (error) return done(error)
$.ajax({
url: protocolName + '://fake-host',
cache: false,
success: (data) => {
assert.equal(data, text)
done()
},
error: (xhr, errorType, error) => done(error)
})
})
})
it('sets Access-Control-Allow-Origin', (done) => {
const handler = (request, callback) => callback(text)
protocol.registerStringProtocol(protocolName, handler, (error) => {
if (error) return done(error)
$.ajax({
url: protocolName + '://fake-host',
cache: false,
success: (data, status, request) => {
assert.equal(data, text)
assert.equal(request.getResponseHeader('Access-Control-Allow-Origin'), '*')
done()
},
error: (xhr, errorType, error) => done(error)
})
})
})
it('sends object as response', (done) => {
const handler = (request, callback) => {
callback({
data: text,
mimeType: 'text/html'
})
}
protocol.registerStringProtocol(protocolName, handler, (error) => {
if (error) return done(error)
$.ajax({
url: protocolName + '://fake-host',
cache: false,
success: (data) => {
assert.equal(data, text)
done()
},
error: (xhr, errorType, error) => done(error)
})
})
})
it('fails when sending object other than string', (done) => {
const handler = (request, callback) => callback(new Date())
protocol.registerBufferProtocol(protocolName, handler, (error) => {
if (error) return done(error)
$.ajax({
url: protocolName + '://fake-host',
cache: false,
success: () => done('request succeeded but it should not'),
error: (xhr, errorType) => {
assert.equal(errorType, 'error')
done()
}
})
})
})
})
describe('protocol.registerBufferProtocol', () => {
const buffer = Buffer.from(text)
it('sends Buffer as response', (done) => {
const handler = (request, callback) => callback(buffer)
protocol.registerBufferProtocol(protocolName, handler, (error) => {
if (error) return done(error)
$.ajax({
url: protocolName + '://fake-host',
cache: false,
success: (data) => {
assert.equal(data, text)
done()
},
error: (xhr, errorType, error) => done(error)
})
})
})
it('sets Access-Control-Allow-Origin', (done) => {
const handler = (request, callback) => callback(buffer)
protocol.registerBufferProtocol(protocolName, handler, (error) => {
if (error) return done(error)
$.ajax({
url: protocolName + '://fake-host',
cache: false,
success: (data, status, request) => {
assert.equal(data, text)
assert.equal(request.getResponseHeader('Access-Control-Allow-Origin'), '*')
done()
},
error: (xhr, errorType, error) => done(error)
})
})
})
it('sends object as response', (done) => {
const handler = (request, callback) => {
callback({
data: buffer,
mimeType: 'text/html'
})
}
protocol.registerBufferProtocol(protocolName, handler, (error) => {
if (error) return done(error)
$.ajax({
url: protocolName + '://fake-host',
cache: false,
success: (data) => {
assert.equal(data, text)
done()
},
error: (xhr, errorType, error) => done(error)
})
})
})
it('fails when sending string', (done) => {
const handler = (request, callback) => callback(text)
protocol.registerBufferProtocol(protocolName, handler, (error) => {
if (error) return done(error)
$.ajax({
url: protocolName + '://fake-host',
cache: false,
success: () => done('request succeeded but it should not'),
error: (xhr, errorType) => {
assert.equal(errorType, 'error')
done()
}
})
})
})
})
describe('protocol.registerFileProtocol', () => {
const filePath = path.join(__dirname, 'fixtures', 'asar', 'a.asar', 'file1')
const fileContent = require('fs').readFileSync(filePath)
const normalPath = path.join(__dirname, 'fixtures', 'pages', 'a.html')
const normalContent = require('fs').readFileSync(normalPath)
it('sends file path as response', (done) => {
const handler = (request, callback) => callback(filePath)
protocol.registerFileProtocol(protocolName, handler, (error) => {
if (error) return done(error)
$.ajax({
url: protocolName + '://fake-host',
cache: false,
success: (data) => {
assert.equal(data, String(fileContent))
return done()
},
error: (xhr, errorType, error) => done(error)
})
})
})
it('sets Access-Control-Allow-Origin', (done) => {
const handler = (request, callback) => callback(filePath)
protocol.registerFileProtocol(protocolName, handler, (error) => {
if (error) return done(error)
$.ajax({
url: protocolName + '://fake-host',
cache: false,
success: (data, status, request) => {
assert.equal(data, String(fileContent))
assert.equal(request.getResponseHeader('Access-Control-Allow-Origin'), '*')
done()
},
error: (xhr, errorType, error) => {
done(error)
}
})
})
})
it('sends object as response', (done) => {
const handler = (request, callback) => callback({ path: filePath })
protocol.registerFileProtocol(protocolName, handler, (error) => {
if (error) return done(error)
$.ajax({
url: protocolName + '://fake-host',
cache: false,
success: (data) => {
assert.equal(data, String(fileContent))
done()
},
error: (xhr, errorType, error) => done(error)
})
})
})
it('can send normal file', (done) => {
const handler = (request, callback) => callback(normalPath)
protocol.registerFileProtocol(protocolName, handler, (error) => {
if (error) return done(error)
$.ajax({
url: protocolName + '://fake-host',
cache: false,
success: (data) => {
assert.equal(data, String(normalContent))
done()
},
error: (xhr, errorType, error) => done(error)
})
})
})
it('fails when sending unexist-file', (done) => {
const fakeFilePath = path.join(__dirname, 'fixtures', 'asar', 'a.asar', 'not-exist')
const handler = (request, callback) => callback(fakeFilePath)
protocol.registerFileProtocol(protocolName, handler, (error) => {
if (error) return done(error)
$.ajax({
url: protocolName + '://fake-host',
cache: false,
success: () => done('request succeeded but it should not'),
error: (xhr, errorType) => {
assert.equal(errorType, 'error')
done()
}
})
})
})
it('fails when sending unsupported content', (done) => {
const handler = (request, callback) => callback(new Date())
protocol.registerFileProtocol(protocolName, handler, (error) => {
if (error) return done(error)
$.ajax({
url: protocolName + '://fake-host',
cache: false,
success: () => done('request succeeded but it should not'),
error: (xhr, errorType) => {
assert.equal(errorType, 'error')
done()
}
})
})
})
})
describe('protocol.registerHttpProtocol', () => {
it('sends url as response', (done) => {
const server = http.createServer((req, res) => {
assert.notEqual(req.headers.accept, '')
res.end(text)
server.close()
})
server.listen(0, '127.0.0.1', () => {
const port = server.address().port
const url = 'http://127.0.0.1:' + port
const handler = (request, callback) => callback({url})
protocol.registerHttpProtocol(protocolName, handler, (error) => {
if (error) return done(error)
$.ajax({
url: protocolName + '://fake-host',
cache: false,
success: (data) => {
assert.equal(data, text)
done()
},
error: (xhr, errorType, error) => done(error)
})
})
})
})
it('fails when sending invalid url', (done) => {
const handler = (request, callback) => callback({url: 'url'})
protocol.registerHttpProtocol(protocolName, handler, (error) => {
if (error) return done(error)
$.ajax({
url: protocolName + '://fake-host',
cache: false,
success: () => done('request succeeded but it should not'),
error: (xhr, errorType) => {
assert.equal(errorType, 'error')
done()
}
})
})
})
it('fails when sending unsupported content', (done) => {
const handler = (request, callback) => callback(new Date())
protocol.registerHttpProtocol(protocolName, handler, (error) => {
if (error) return done(error)
$.ajax({
url: protocolName + '://fake-host',
cache: false,
success: () => {
done('request succeeded but it should not')
},
error: (xhr, errorType) => {
assert.equal(errorType, 'error')
done()
}
})
})
})
it('works when target URL redirects', (done) => {
let contents = null
const server = http.createServer((req, res) => {
if (req.url === '/serverRedirect') {
res.statusCode = 301
res.setHeader('Location', `http://${req.rawHeaders[1]}`)
res.end()
} else {
res.end(text)
}
})
server.listen(0, '127.0.0.1', () => {
const port = server.address().port
const url = `${protocolName}://fake-host`
const redirectURL = `http://127.0.0.1:${port}/serverRedirect`
const handler = (request, callback) => callback({url: redirectURL})
protocol.registerHttpProtocol(protocolName, handler, (error) => {
if (error) return done(error)
contents = webContents.create({})
contents.on('did-finish-load', () => {
assert.equal(contents.getURL(), url)
server.close()
contents.destroy()
done()
})
contents.loadURL(url)
})
})
})
})
xdescribe('protocol.registerStreamProtocol', () => {
it('sends Stream as response', (done) => {
const handler = (request, callback) => callback(getStream())
protocol.registerStreamProtocol(protocolName, handler, (error) => {
if (error) return done(error)
$.ajax({
url: protocolName + '://fake-host',
cache: false,
success: (data) => {
assert.equal(data, text)
done()
},
error: (xhr, errorType, error) => {
done(error || new Error(`Request failed: ${xhr.status}`))
}
})
})
})
it('sends object as response', (done) => {
const handler = (request, callback) => callback({data: getStream()})
protocol.registerStreamProtocol(protocolName, handler, (error) => {
if (error) return done(error)
$.ajax({
url: protocolName + '://fake-host',
cache: false,
success: (data, _, request) => {
assert.equal(request.status, 200)
assert.equal(data, text)
done()
},
error: (xhr, errorType, error) => {
done(error || new Error(`Request failed: ${xhr.status}`))
}
})
})
})
it('sends custom response headers', (done) => {
const handler = (request, callback) => callback({
data: getStream(3),
headers: {
'x-electron': ['a', 'b']
}
})
protocol.registerStreamProtocol(protocolName, handler, (error) => {
if (error) return done(error)
$.ajax({
url: protocolName + '://fake-host',
cache: false,
success: (data, _, request) => {
assert.equal(request.status, 200)
assert.equal(request.getResponseHeader('x-electron'), 'a,b')
assert.equal(data, text)
done()
},
error: (xhr, errorType, error) => {
done(error || new Error(`Request failed: ${xhr.status}`))
}
})
})
})
it('sends custom status code', (done) => {
const handler = (request, callback) => callback({
statusCode: 204,
data: null
})
protocol.registerStreamProtocol(protocolName, handler, (error) => {
if (error) return done(error)
$.ajax({
url: protocolName + '://fake-host',
cache: false,
success: (data, _, request) => {
assert.equal(request.status, 204)
assert.equal(data, undefined)
done()
},
error: (xhr, errorType, error) => {
done(error || new Error(`Request failed: ${xhr.status}`))
}
})
})
})
it('receives request headers', (done) => {
const handler = (request, callback) => {
callback({
headers: {
'content-type': 'application/json'
},
data: getStream(5, JSON.stringify(Object.assign({}, request.headers)))
})
}
protocol.registerStreamProtocol(protocolName, handler, (error) => {
if (error) return done(error)
$.ajax({
url: protocolName + '://fake-host',
headers: {
'x-return-headers': 'yes'
},
cache: false,
success: (data) => {
assert.equal(data['x-return-headers'], 'yes')
done()
},
error: (xhr, errorType, error) => {
done(error || new Error(`Request failed: ${xhr.status}`))
}
})
})
})
})
describe('protocol.isProtocolHandled', () => {
it('returns true for about:', (done) => {
protocol.isProtocolHandled('about', (result) => {
assert.equal(result, true)
done()
})
})
it('returns true for file:', (done) => {
protocol.isProtocolHandled('file', (result) => {
assert.equal(result, true)
done()
})
})
it('returns true for http:', (done) => {
protocol.isProtocolHandled('http', (result) => {
assert.equal(result, true)
done()
})
})
it('returns true for https:', (done) => {
protocol.isProtocolHandled('https', (result) => {
assert.equal(result, true)
done()
})
})
it('returns false when scheme is not registered', (done) => {
protocol.isProtocolHandled('no-exist', (result) => {
assert.equal(result, false)
done()
})
})
it('returns true for custom protocol', (done) => {
const emptyHandler = (request, callback) => callback()
protocol.registerStringProtocol(protocolName, emptyHandler, (error) => {
assert.equal(error, null)
protocol.isProtocolHandled(protocolName, (result) => {
assert.equal(result, true)
done()
})
})
})
it('returns true for intercepted protocol', (done) => {
const emptyHandler = (request, callback) => callback()
protocol.interceptStringProtocol('http', emptyHandler, (error) => {
assert.equal(error, null)
protocol.isProtocolHandled('http', (result) => {
assert.equal(result, true)
done()
})
})
})
})
describe('protocol.intercept(Any)Protocol', () => {
const emptyHandler = (request, callback) => callback()
it('throws error when scheme is already intercepted', (done) => {
protocol.interceptStringProtocol('http', emptyHandler, (error) => {
assert.equal(error, null)
protocol.interceptBufferProtocol('http', emptyHandler, (error) => {
assert.notEqual(error, null)
done()
})
})
})
it('does not crash when handler is called twice', (done) => {
var doubleHandler = (request, callback) => {
try {
callback(text)
callback()
} catch (error) {
// Ignore error
}
}
protocol.interceptStringProtocol('http', doubleHandler, (error) => {
if (error) return done(error)
$.ajax({
url: 'http://fake-host',
cache: false,
success: (data) => {
assert.equal(data, text)
done()
},
error: (xhr, errorType, error) => done(error)
})
})
})
it('sends error when callback is called with nothing', function (done) {
protocol.interceptBufferProtocol('http', emptyHandler, (error) => {
if (error) return done(error)
$.ajax({
url: 'http://fake-host',
cache: false,
success: () => done('request succeeded but it should not'),
error: (xhr, errorType) => {
assert.equal(errorType, 'error')
done()
}
})
})
})
})
describe('protocol.interceptStringProtocol', () => {
it('can intercept http protocol', (done) => {
const handler = (request, callback) => callback(text)
protocol.interceptStringProtocol('http', handler, (error) => {
if (error) return done(error)
$.ajax({
url: 'http://fake-host',
cache: false,
success: (data) => {
assert.equal(data, text)
done()
},
error: (xhr, errorType, error) => done(error)
})
})
})
it('can set content-type', (done) => {
const handler = (request, callback) => {
callback({
mimeType: 'application/json',
data: '{"value": 1}'
})
}
protocol.interceptStringProtocol('http', handler, (error) => {
if (error) return done(error)
$.ajax({
url: 'http://fake-host',
cache: false,
success: (data) => {
assert.equal(typeof data, 'object')
assert.equal(data.value, 1)
done()
},
error: (xhr, errorType, error) => done(error)
})
})
})
it('can receive post data', (done) => {
const handler = (request, callback) => {
const uploadData = request.uploadData[0].bytes.toString()
callback({data: uploadData})
}
protocol.interceptStringProtocol('http', handler, (error) => {
if (error) return done(error)
$.ajax({
url: 'http://fake-host',
cache: false,
type: 'POST',
data: postData,
success: (data) => {
assert.deepEqual(qs.parse(data), postData)
done()
},
error: (xhr, errorType, error) => done(error)
})
})
})
})
describe('protocol.interceptBufferProtocol', () => {
it('can intercept http protocol', (done) => {
const handler = (request, callback) => callback(Buffer.from(text))
protocol.interceptBufferProtocol('http', handler, (error) => {
if (error) return done(error)
$.ajax({
url: 'http://fake-host',
cache: false,
success: (data) => {
assert.equal(data, text)
done()
},
error: (xhr, errorType, error) => done(error)
})
})
})
it('can receive post data', (done) => {
const handler = (request, callback) => {
const uploadData = request.uploadData[0].bytes
callback(uploadData)
}
protocol.interceptBufferProtocol('http', handler, (error) => {
if (error) return done(error)
$.ajax({
url: 'http://fake-host',
cache: false,
type: 'POST',
data: postData,
success: (data) => {
assert.equal(data, $.param(postData))
done()
},
error: (xhr, errorType, error) => done(error)
})
})
})
})
describe('protocol.interceptHttpProtocol', () => {
it('can send POST request', (done) => {
const server = http.createServer((req, res) => {
let body = ''
req.on('data', (chunk) => {
body += chunk
})
req.on('end', () => {
res.end(body)
})
server.close()
})
server.listen(0, '127.0.0.1', () => {
const port = server.address().port
const url = `http://127.0.0.1:${port}`
const handler = (request, callback) => {
const data = {
url: url,
method: 'POST',
uploadData: {
contentType: 'application/x-www-form-urlencoded',
data: request.uploadData[0].bytes.toString()
},
session: null
}
callback(data)
}
protocol.interceptHttpProtocol('http', handler, (error) => {
if (error) return done(error)
$.ajax({
url: 'http://fake-host',
cache: false,
type: 'POST',
data: postData,
success: (data) => {
assert.deepEqual(qs.parse(data), postData)
done()
},
error: (xhr, errorType, error) => done(error)
})
})
})
})
it('can use custom session', (done) => {
const customSession = session.fromPartition('custom-ses', {cache: false})
customSession.webRequest.onBeforeRequest((details, callback) => {
assert.equal(details.url, 'http://fake-host/')
callback({cancel: true})
})
const handler = (request, callback) => {
callback({
url: request.url,
session: customSession
})
}
protocol.interceptHttpProtocol('http', handler, (error) => {
if (error) return done(error)
fetch('http://fake-host').then(() => {
done('request succeeded but it should not')
}).catch(() => {
customSession.webRequest.onBeforeRequest(null)
done()
})
})
})
})
xdescribe('protocol.interceptStreamProtocol', () => {
it('can intercept http protocol', (done) => {
const handler = (request, callback) => callback(getStream())
protocol.interceptStreamProtocol('http', handler, (error) => {
if (error) return done(error)
$.ajax({
url: 'http://fake-host',
cache: false,
success: (data) => {
assert.equal(data, text)
done()
},
error: (xhr, errorType, error) => {
done(error || new Error(`Request failed: ${xhr.status}`))
}
})
})
})
it('can receive post data', (done) => {
const handler = (request, callback) => {
callback(getStream(3, request.uploadData[0].bytes.toString()))
}
protocol.interceptStreamProtocol('http', handler, (error) => {
if (error) return done(error)
$.ajax({
url: 'http://fake-host',
cache: false,
type: 'POST',
data: postData,
success: (data) => {
assert.deepEqual(qs.parse(data), postData)
done()
},
error: (xhr, errorType, error) => {
done(error || new Error(`Request failed: ${xhr.status}`))
}
})
})
})
it('can execute redirects', (done) => {
const handler = (request, callback) => {
if (request.url.indexOf('http://fake-host') === 0) {
setTimeout(() => {
callback({
data: null,
statusCode: 302,
headers: {
Location: 'http://fake-redirect'
}
})
}, 300)
} else {
assert.equal(request.url.indexOf('http://fake-redirect'), 0)
callback(getStream(1, 'redirect'))
}
}
protocol.interceptStreamProtocol('http', handler, (error) => {
if (error) return done(error)
$.ajax({
url: 'http://fake-host',
cache: false,
success: (data) => {
assert.equal(data, 'redirect')
done()
},
error: (xhr, errorType, error) => {
done(error || new Error(`Request failed: ${xhr.status}`))
}
})
})
})
})
describe('protocol.uninterceptProtocol', () => {
it('returns error when scheme does not exist', (done) => {
protocol.uninterceptProtocol('not-exist', (error) => {
assert.notEqual(error, null)
done()
})
})
it('returns error when scheme is not intercepted', (done) => {
protocol.uninterceptProtocol('http', (error) => {
assert.notEqual(error, null)
done()
})
})
})
describe('protocol.registerStandardSchemes', () => {
const standardScheme = remote.getGlobal('standardScheme')
const origin = `${standardScheme}://fake-host`
const imageURL = `${origin}/test.png`
const filePath = path.join(__dirname, 'fixtures', 'pages', 'b.html')
const fileContent = ''
let w = null
let success = null
beforeEach(() => {
w = new BrowserWindow({show: false})
success = false
})
afterEach((done) => {
protocol.unregisterProtocol(standardScheme, () => {
closeWindow(w).then(() => {
w = null
done()
})
})
})
it('resolves relative resources', (done) => {
const handler = (request, callback) => {
if (request.url === imageURL) {
success = true
callback()
} else {
callback(filePath)
}
}
protocol.registerFileProtocol(standardScheme, handler, (error) => {
if (error) return done(error)
w.webContents.on('did-finish-load', () => {
assert(success)
done()
})
w.loadURL(origin)
})
})
it('resolves absolute resources', (done) => {
const handler = (request, callback) => {
if (request.url === imageURL) {
success = true
callback()
} else {
callback({
data: fileContent,
mimeType: 'text/html'
})
}
}
protocol.registerStringProtocol(standardScheme, handler, (error) => {
if (error) return done(error)
w.webContents.on('did-finish-load', () => {
assert(success)
done()
})
w.loadURL(origin)
})
})
it('can have fetch working in it', (done) => {
const content = ''
const handler = (request, callback) => callback({data: content, mimeType: 'text/html'})
protocol.registerStringProtocol(standardScheme, handler, (error) => {
if (error) return done(error)
w.webContents.on('crashed', () => done('WebContents crashed'))
w.webContents.on('did-finish-load', () => done())
w.loadURL(origin)
})
})
it('can access files through the FileSystem API', (done) => {
let filePath = path.join(__dirname, 'fixtures', 'pages', 'filesystem.html')
const handler = (request, callback) => callback({path: filePath})
protocol.registerFileProtocol(standardScheme, handler, (error) => {
if (error) return done(error)
w.loadURL(origin)
})
ipcMain.once('file-system-error', (event, err) => done(err))
ipcMain.once('file-system-write-end', () => done())
})
it('registers secure, when {secure: true}', (done) => {
let filePath = path.join(__dirname, 'fixtures', 'pages', 'cache-storage.html')
const handler = (request, callback) => callback({path: filePath})
ipcMain.once('success', () => done())
ipcMain.once('failure', (event, err) => done(err))
protocol.registerFileProtocol(standardScheme, handler, (error) => {
if (error) return done(error)
w.loadURL(origin)
})
})
})
})