LNXBOX/test/setup.js
2025-01-24 06:58:00 +01:00

237 lines
8.3 KiB
JavaScript

const Application = require('spectron').Application
const { copyFileSync } = require('fs')
const fs = require('fs')
const parseTorrent = require('parse-torrent')
const path = require('path')
const PNG = require('pngjs').PNG
const rimraf = require('rimraf')
const config = require('./config')
module.exports = {
createApp,
endTest,
screenshotCreateOrCompare,
compareDownloadFolder,
compareFiles,
compareTorrentFile,
compareTorrentFiles,
waitForLoad,
wait,
resetTestDataDir,
deleteTestDataDir,
copy
}
// Runs WebTorrent Desktop.
// Returns a promise that resolves to a Spectron Application once the app has loaded.
// Takes a Tape test. Makes some basic assertions to verify that the app loaded correctly.
function createApp (t) {
const userDataDir = process.platform === 'win32'
? path.join('C:\\Windows\\Temp', 'WebTorrentTest')
: path.join('/tmp', 'WebTorrentTest')
return new Application({
path: path.join(__dirname, '..', 'node_modules', '.bin',
'electron' + (process.platform === 'win32' ? '.cmd' : '')),
args: ['-r', path.join(__dirname, 'mocks.js'), path.join(__dirname, '..')],
chromeDriverArgs: [`--user-data-dir=${userDataDir}`],
env: { NODE_ENV: 'test' },
waitTimeout: 10e3
})
}
// Starts the app, waits for it to load, returns a promise
function waitForLoad (app, t, opts) {
if (!opts) opts = {}
return app.start().then(function () {
return app.client.waitUntilWindowLoaded()
}).then(function () {
// Offline mode
if (!opts.online) app.webContents.executeJavaScript('testOfflineMode()')
}).then(function () {
// Switch to the main window. Index 0 is apparently the hidden webtorrent window...
return app.client.windowByIndex(1)
}).then(function () {
return app.client.waitUntilWindowLoaded()
}).then(function () {
return app.webContents.getTitle()
}).then(function (title) {
// Note the window title is WebTorrent, this is the HTML <title>
t.equal(title, 'Main Window', 'html title')
})
}
// Returns a promise that resolves after 'ms' milliseconds. Default: 1 second
function wait (ms) {
if (ms === undefined) ms = 1000 // Default: wait long enough for the UI to update
return new Promise(function (resolve, reject) {
setTimeout(resolve, ms)
})
}
// Quit the app, end the test, either in success (!err) or failure (err)
function endTest (app, t, err) {
return app.stop().then(function () {
t.end(err)
})
}
// Takes a screenshot of the app
// If we already have a reference under test/screenshots, assert that they're the same
// Otherwise, create the reference screenshot: test/screenshots/<platform>/<name>.png
function screenshotCreateOrCompare (app, t, name) {
const ssDir = path.join(__dirname, 'screenshots', process.platform)
// check that path exists otherwise create it
if (!fs.existsSync(ssDir)) {
fs.mkdirSync(ssDir)
}
const ssPath = path.join(ssDir, name + '.png')
let ssBuf
try {
ssBuf = fs.readFileSync(ssPath)
} catch (err) {
ssBuf = Buffer.alloc(0)
}
return app.browserWindow.focus()
.then(() => wait())
.then(() => app.browserWindow.capturePage())
.then(function (buffer) {
if (ssBuf.length === 0) {
console.log('Saving screenshot ' + ssPath)
fs.writeFileSync(ssPath, buffer)
} else {
const match = compareIgnoringTransparency(buffer, ssBuf)
t.ok(match, 'screenshot comparison ' + name)
if (!match) {
const ssFailedPath = path.join(ssDir, name + '-failed.png')
console.log('Saving screenshot, failed comparison: ' + ssFailedPath)
fs.writeFileSync(ssFailedPath, buffer)
}
}
})
}
// Compares two PNGs, ignoring any transparent regions in bufExpected.
// Returns true if they match.
function compareIgnoringTransparency (bufActual, bufExpected) {
// Common case: exact byte-for-byte match
if (Buffer.compare(bufActual, bufExpected) === 0) return true
// Otherwise, compare pixel by pixel
let sumSquareDiff = 0
let numDiff = 0
const pngA = PNG.sync.read(bufActual)
const pngE = PNG.sync.read(bufExpected)
if (pngA.width !== pngE.width || pngA.height !== pngE.height) return false
const w = pngA.width
const h = pngE.height
const da = pngA.data
const de = pngE.data
for (let y = 0; y < h; y++) {
for (let x = 0; x < w; x++) {
const i = ((y * w) + x) * 4
if (de[i + 3] === 0) continue // Skip transparent pixels
const ca = (da[i] << 16) | (da[i + 1] << 8) | da[i + 2]
const ce = (de[i] << 16) | (de[i + 1] << 8) | de[i + 2]
if (ca === ce) continue
// Add pixel diff to running sum
// This is necessary on Windows, where rendering apparently isn't quite deterministic
// and a few pixels in the screenshot will sometimes be off by 1. (Visually identical.)
numDiff++
sumSquareDiff += (da[i] - de[i]) * (da[i] - de[i])
sumSquareDiff += (da[i + 1] - de[i + 1]) * (da[i + 1] - de[i + 1])
sumSquareDiff += (da[i + 2] - de[i + 2]) * (da[i + 2] - de[i + 2])
}
}
const rms = Math.sqrt(sumSquareDiff / (numDiff + 1))
const l2Distance = Math.round(Math.sqrt(sumSquareDiff))
console.log('screenshot diff l2 distance: ' + l2Distance + ', rms: ' + rms)
return l2Distance < 5000 && rms < 100
}
// Resets the test directory, containing config.json, torrents, downloads, etc
function resetTestDataDir () {
rimraf.sync(config.TEST_DIR)
// Create TEST_DIR as well as /Downloads and /Desktop
fs.mkdirSync(config.TEST_DIR_DOWNLOAD, { recursive: true })
fs.mkdirSync(config.TEST_DIR_DESKTOP, { recursive: true })
}
function deleteTestDataDir () {
rimraf.sync(config.TEST_DIR)
}
// Checks a given folder under Downloads.
// Makes sure that the filenames match exactly.
// If `filenames` is null, asserts that the folder doesn't exist.
function compareDownloadFolder (t, dirname, filenames) {
const dirpath = path.join(config.TEST_DIR_DOWNLOAD, dirname)
try {
const actualFilenames = fs.readdirSync(dirpath)
if (filenames === null) {
return t.fail('expected download folder to be absent, but it\'s here: ' + dirpath)
}
const expectedSorted = filenames.slice().sort()
const actualSorted = actualFilenames.slice().sort()
console.log(actualSorted)
t.deepEqual(actualSorted, expectedSorted, 'download folder contents: ' + dirname)
} catch (err) {
if (err.code === 'ENOENT') {
t.equal(filenames, null, 'download folder missing: ' + dirname)
} else {
console.error(err)
t.fail('unexpected error getting download folder: ' + dirname)
}
}
}
// Makes sure two files have identical contents
function compareFiles (t, pathActual, pathExpected) {
const bufActual = fs.readFileSync(pathActual)
const bufExpected = fs.readFileSync(pathExpected)
const match = Buffer.compare(bufActual, bufExpected) === 0
t.ok(match, 'correct contents: ' + pathActual)
}
// Makes sure two torrents have the same infohash and flags
function compareTorrentFiles (t, pathActual, pathExpected) {
const bufActual = fs.readFileSync(pathActual)
const bufExpected = fs.readFileSync(pathExpected)
const fieldsActual = extractImportantFields(parseTorrent(bufActual))
const fieldsExpected = extractImportantFields(parseTorrent(bufExpected))
t.deepEqual(fieldsActual, fieldsExpected, 'torrent contents: ' + pathActual)
}
// Makes sure two torrents have the same infohash and flags
function compareTorrentFile (t, pathActual, fieldsExpected) {
const bufActual = fs.readFileSync(pathActual)
const fieldsActual = extractImportantFields(parseTorrent(bufActual))
if (Array.isArray(fieldsExpected.announce)) fieldsExpected.announce.sort()
t.deepEqual(fieldsActual, fieldsExpected, 'torrent contents: ' + pathActual)
}
function extractImportantFields (parsedTorrent) {
let { infoHash, name, announce, urlList, comment } = parsedTorrent
const priv = parsedTorrent.private // private is a reserved word in JS
announce = announce.slice().sort()
return { infoHash, name, announce, urlList, comment, private: priv }
}
function copy (pathFrom, pathTo) {
try {
copyFileSync(pathFrom, pathTo)
} catch (err) {
// Windows lets us create files and folders under C:\Windows\Temp,
// but when you try to `copySync` into one of those folders, you get EPERM
// Ignore for now...
if (process.platform !== 'win32' || err.code !== 'EPERM') throw err
console.log('ignoring windows copy EPERM error', err)
}
}