2021-06-15 11:08:13 +03:00
|
|
|
#!/usr/bin/env node
|
2021-11-29 19:42:19 +03:00
|
|
|
/**
|
|
|
|
* Copyright (c) Microsoft Corporation.
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
2021-06-15 11:08:13 +03:00
|
|
|
|
|
|
|
import * as path from 'path';
|
|
|
|
import * as URL from 'url';
|
|
|
|
import * as fs from 'fs';
|
|
|
|
import * as http from 'http';
|
|
|
|
import * as https from 'https';
|
2021-06-17 01:13:51 +03:00
|
|
|
import * as os from 'os';
|
|
|
|
import * as util from 'util';
|
|
|
|
import * as child_process from 'child_process';
|
2021-06-15 11:08:13 +03:00
|
|
|
|
|
|
|
const existsAsync = path => new Promise(resolve => fs.stat(path, err => resolve(!err)));
|
|
|
|
|
|
|
|
const __filename = URL.fileURLToPath(import.meta.url);
|
|
|
|
const __dirname = path.dirname(__filename);
|
|
|
|
|
2021-08-31 14:04:30 +03:00
|
|
|
if (process.argv[2] === '--help' || process.argv[2] === '-h') {
|
|
|
|
console.log(`usage: ${path.basename(process.argv[1])} [firefox|ff|firefox-beta|ff-beta] [build number] [build platform]`);
|
|
|
|
console.log(``);
|
|
|
|
console.log(`Repackages Firefox with tip-of-tree Juggler implementation`);
|
|
|
|
process.exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
let browserName = '';
|
|
|
|
if (process.argv[2] === 'firefox' || process.argv[2] === 'ff') {
|
|
|
|
browserName = 'firefox';
|
|
|
|
} else if (process.argv[2] === 'firefox-beta' || process.argv[2] === 'ff-beta') {
|
|
|
|
browserName = 'firefox-beta';
|
|
|
|
} else {
|
|
|
|
console.error('ERROR: unknown firefox to repackage - either "firefox", "ff", "firefox-beta" or "ff-beta" is allowed as first argument');
|
|
|
|
process.exit(1);
|
|
|
|
}
|
|
|
|
|
2021-06-15 11:08:13 +03:00
|
|
|
// Path to jar.mn in the juggler
|
2021-08-31 14:04:30 +03:00
|
|
|
const JARMN_PATH = path.join(__dirname, browserName, 'juggler', 'jar.mn');
|
2021-06-15 11:08:13 +03:00
|
|
|
// Workdir for Firefox repackaging
|
|
|
|
const BUILD_DIRECTORY = `/tmp/repackaged-firefox`;
|
|
|
|
// Information about currently downloaded build
|
|
|
|
const BUILD_INFO_PATH = path.join(BUILD_DIRECTORY, 'build-info.json');
|
|
|
|
// Backup OMNI.JA - the original one before repackaging.
|
|
|
|
const OMNI_BACKUP_PATH = path.join(BUILD_DIRECTORY, 'omni.ja.backup');
|
|
|
|
// Workdir to extract omni.ja
|
|
|
|
const OMNI_EXTRACT_DIR = path.join(BUILD_DIRECTORY, 'omni');
|
|
|
|
// Path inside omni.ja to juggler
|
|
|
|
const OMNI_JUGGLER_DIR = path.join(OMNI_EXTRACT_DIR, 'chrome', 'juggler');
|
|
|
|
|
|
|
|
const EXECUTABLE_PATHS = {
|
|
|
|
'ubuntu18.04': ['firefox', 'firefox'],
|
|
|
|
'ubuntu20.04': ['firefox', 'firefox'],
|
|
|
|
'mac10.14': ['firefox', 'Nightly.app', 'Contents', 'MacOS', 'firefox'],
|
|
|
|
'mac10.15': ['firefox', 'Nightly.app', 'Contents', 'MacOS', 'firefox'],
|
|
|
|
'mac11': ['firefox', 'Nightly.app', 'Contents', 'MacOS', 'firefox'],
|
|
|
|
'mac11-arm64': ['firefox', 'Nightly.app', 'Contents', 'MacOS', 'firefox'],
|
|
|
|
'win64': ['firefox', 'firefox.exe'],
|
|
|
|
};
|
|
|
|
|
2021-06-17 01:13:51 +03:00
|
|
|
const DOWNLOAD_URLS = {
|
2021-08-31 14:04:30 +03:00
|
|
|
'firefox': {
|
|
|
|
'ubuntu18.04': 'https://playwright.azureedge.net/builds/firefox/%s/firefox-ubuntu-18.04.zip',
|
|
|
|
'ubuntu20.04': 'https://playwright.azureedge.net/builds/firefox/%s/firefox-ubuntu-20.04.zip',
|
2021-10-16 02:06:52 +03:00
|
|
|
'mac10.14': 'https://playwright.azureedge.net/builds/firefox/%s/firefox-mac-11.zip',
|
|
|
|
'mac10.15': 'https://playwright.azureedge.net/builds/firefox/%s/firefox-mac-11.zip',
|
|
|
|
'mac11': 'https://playwright.azureedge.net/builds/firefox/%s/firefox-mac-11.zip',
|
2021-12-10 22:45:43 +03:00
|
|
|
'mac11-arm64': 'https://playwright.azureedge.net/builds/firefox/%s/firefox-mac-11-arm64.zip',
|
2021-08-31 14:04:30 +03:00
|
|
|
'win64': 'https://playwright.azureedge.net/builds/firefox/%s/firefox-win64.zip',
|
|
|
|
},
|
|
|
|
'firefox-beta': {
|
|
|
|
'ubuntu18.04': 'https://playwright.azureedge.net/builds/firefox-beta/%s/firefox-beta-ubuntu-18.04.zip',
|
|
|
|
'ubuntu20.04': 'https://playwright.azureedge.net/builds/firefox-beta/%s/firefox-beta-ubuntu-20.04.zip',
|
2021-10-16 02:06:52 +03:00
|
|
|
'mac10.14': 'https://playwright.azureedge.net/builds/firefox-beta/%s/firefox-beta-mac-11.zip',
|
|
|
|
'mac10.15': 'https://playwright.azureedge.net/builds/firefox-beta/%s/firefox-beta-mac-11.zip',
|
|
|
|
'mac11': 'https://playwright.azureedge.net/builds/firefox-beta/%s/firefox-beta-mac-11.zip',
|
2021-12-10 22:45:43 +03:00
|
|
|
'mac11-arm64': 'https://playwright.azureedge.net/builds/firefox-beta/%s/firefox-beta-mac-11-arm64.zip',
|
2021-08-31 14:04:30 +03:00
|
|
|
'win64': 'https://playwright.azureedge.net/builds/firefox-beta/%s/firefox-beta-win64.zip',
|
|
|
|
},
|
2021-06-17 01:13:51 +03:00
|
|
|
};
|
|
|
|
|
2021-08-31 14:04:30 +03:00
|
|
|
async function ensureFirefoxBuild(browserName, buildNumber, buildPlatform) {
|
2021-06-17 03:32:37 +03:00
|
|
|
if (!buildNumber)
|
2021-08-31 14:04:30 +03:00
|
|
|
buildNumber = (await fs.promises.readFile(path.join(__dirname, browserName, 'BUILD_NUMBER'), 'utf8')).split('\n').shift();
|
2021-06-17 03:32:37 +03:00
|
|
|
if (!buildPlatform)
|
|
|
|
buildPlatform = getHostPlatform();
|
2021-08-31 14:04:30 +03:00
|
|
|
const currentBuildInfo = await fs.promises.readFile(BUILD_INFO_PATH).then(text => JSON.parse(text)).catch(e => ({ buildPlatform: '', buildNumber: '', browserName: '' }));
|
2021-06-15 11:08:13 +03:00
|
|
|
|
2021-08-31 14:04:30 +03:00
|
|
|
if (currentBuildInfo.buildPlatform === buildPlatform && currentBuildInfo.buildNumber === buildNumber && currentBuildInfo.browserName === browserName)
|
|
|
|
return currentBuildInfo;
|
2021-11-29 19:42:19 +03:00
|
|
|
await fs.promises.rm(BUILD_DIRECTORY, { recursive: true }).catch(e => {});
|
2021-06-15 11:08:13 +03:00
|
|
|
await fs.promises.mkdir(BUILD_DIRECTORY);
|
|
|
|
const buildZipPath = path.join(BUILD_DIRECTORY, 'firefox.zip');
|
2021-06-17 01:13:51 +03:00
|
|
|
|
2021-08-31 14:04:30 +03:00
|
|
|
const urlTemplate = DOWNLOAD_URLS[browserName][buildPlatform];
|
2021-06-17 01:13:51 +03:00
|
|
|
if (!urlTemplate)
|
|
|
|
throw new Error(`ERROR: repack-juggler does not support ${buildPlatform}`);
|
|
|
|
const url = util.format(urlTemplate, buildNumber);
|
2021-08-31 14:04:30 +03:00
|
|
|
console.log(`Downloading ${browserName} r${buildNumber} for ${buildPlatform} - it might take a few minutes`);
|
2021-06-17 01:13:51 +03:00
|
|
|
await downloadFile(url, buildZipPath);
|
2021-11-29 19:42:19 +03:00
|
|
|
await spawnAsync('unzip', [ buildZipPath ], { cwd: BUILD_DIRECTORY });
|
2021-08-31 14:04:30 +03:00
|
|
|
const buildInfo = { buildNumber, buildPlatform, browserName };
|
|
|
|
await fs.promises.writeFile(BUILD_INFO_PATH, JSON.stringify(buildInfo), 'utf8');
|
|
|
|
return buildInfo;
|
2021-06-15 11:08:13 +03:00
|
|
|
}
|
|
|
|
|
2021-08-31 14:04:30 +03:00
|
|
|
async function repackageJuggler(browserName, buildInfo) {
|
2021-11-29 19:42:19 +03:00
|
|
|
const { buildNumber, buildPlatform } = buildInfo;
|
2021-06-17 03:32:37 +03:00
|
|
|
|
|
|
|
// Find all omni.ja files in the Firefox build.
|
|
|
|
const omniPaths = await spawnAsync('find', ['.', '-name', 'omni.ja'], {
|
|
|
|
cwd: BUILD_DIRECTORY,
|
2021-11-29 19:42:19 +03:00
|
|
|
}).then(({ stdout }) => stdout.trim().split('\n').map(aPath => path.join(BUILD_DIRECTORY, aPath)));
|
2021-06-17 03:32:37 +03:00
|
|
|
|
|
|
|
// Iterate over all omni.ja files and find one that has juggler inside.
|
|
|
|
const omniWithJugglerPath = await (async () => {
|
|
|
|
for (const omniPath of omniPaths) {
|
2021-11-29 19:42:19 +03:00
|
|
|
const { stdout } = await spawnAsync('unzip', ['-Z1', omniPath], { cwd: BUILD_DIRECTORY });
|
2021-06-17 03:32:37 +03:00
|
|
|
if (stdout.includes('chrome/juggler'))
|
|
|
|
return omniPath;
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
})();
|
|
|
|
|
|
|
|
if (!omniWithJugglerPath) {
|
|
|
|
console.error('ERROR: did not find omni.ja file with baked in Juggler!');
|
|
|
|
process.exit(1);
|
|
|
|
} else {
|
|
|
|
if (!(await existsAsync(OMNI_BACKUP_PATH)))
|
|
|
|
await fs.promises.copyFile(omniWithJugglerPath, OMNI_BACKUP_PATH);
|
2021-06-15 11:08:13 +03:00
|
|
|
}
|
|
|
|
|
2021-06-17 03:32:37 +03:00
|
|
|
// Let's repackage omni folder!
|
2021-11-29 19:42:19 +03:00
|
|
|
await fs.promises.rm(OMNI_EXTRACT_DIR, { recursive: true }).catch(e => {});
|
2021-06-17 03:32:37 +03:00
|
|
|
await fs.promises.mkdir(OMNI_EXTRACT_DIR);
|
|
|
|
|
2021-11-29 19:42:19 +03:00
|
|
|
await spawnAsync('unzip', [OMNI_BACKUP_PATH], { cwd: OMNI_EXTRACT_DIR });
|
2021-06-17 03:32:37 +03:00
|
|
|
// Remove current juggler directory
|
2021-11-29 19:42:19 +03:00
|
|
|
await fs.promises.rm(OMNI_JUGGLER_DIR, { recursive: true });
|
2021-06-17 03:32:37 +03:00
|
|
|
// Repopulate with tip-of-tree juggler files
|
|
|
|
const jarmn = await fs.promises.readFile(JARMN_PATH, 'utf8');
|
|
|
|
const jarLines = jarmn.split('\n').map(line => line.trim()).filter(line => line.startsWith('content/') && line.endsWith(')'));
|
|
|
|
for (const line of jarLines) {
|
|
|
|
const tokens = line.split(/\s+/);
|
|
|
|
const toPath = path.join(OMNI_JUGGLER_DIR, tokens[0]);
|
2021-08-31 14:04:30 +03:00
|
|
|
const fromPath = path.join(__dirname, browserName, 'juggler', tokens[1].slice(1, -1));
|
2021-11-29 19:42:19 +03:00
|
|
|
await fs.promises.mkdir(path.dirname(toPath), { recursive: true });
|
2021-06-17 03:32:37 +03:00
|
|
|
await fs.promises.copyFile(fromPath, toPath);
|
|
|
|
}
|
2021-06-15 11:08:13 +03:00
|
|
|
|
2021-08-31 14:04:30 +03:00
|
|
|
await fs.promises.unlink(omniWithJugglerPath);
|
2021-11-29 19:42:19 +03:00
|
|
|
await spawnAsync('zip', ['-0', '-qr9XD', omniWithJugglerPath, '.'], { cwd: OMNI_EXTRACT_DIR, stdio: 'inherit' });
|
2021-06-15 11:08:13 +03:00
|
|
|
|
2021-08-31 14:04:30 +03:00
|
|
|
const module = await import(path.join(__dirname, browserName, 'install-preferences.js'));
|
|
|
|
await module.default.installFirefoxPreferences(path.join(BUILD_DIRECTORY, 'firefox'));
|
|
|
|
|
2021-06-17 03:32:37 +03:00
|
|
|
// Output executable path to be used in test.
|
|
|
|
console.log(`
|
2021-08-31 14:04:30 +03:00
|
|
|
browser: ${browserName}
|
2021-06-17 03:32:37 +03:00
|
|
|
buildNumber: ${buildNumber}
|
|
|
|
buildPlatform: ${buildPlatform}
|
|
|
|
executablePath: ${path.join(BUILD_DIRECTORY, ...EXECUTABLE_PATHS[buildPlatform])}
|
|
|
|
`);
|
|
|
|
}
|
2021-06-15 11:08:13 +03:00
|
|
|
|
|
|
|
|
|
|
|
function httpRequest(url, method, response) {
|
2021-11-29 19:42:19 +03:00
|
|
|
const options = URL.parse(url);
|
2021-06-15 11:08:13 +03:00
|
|
|
options.method = method;
|
|
|
|
|
|
|
|
const requestCallback = res => {
|
|
|
|
if (res.statusCode >= 300 && res.statusCode < 400 && res.headers.location)
|
|
|
|
httpRequest(res.headers.location, method, response);
|
|
|
|
else
|
|
|
|
response(res);
|
|
|
|
};
|
|
|
|
const request = options.protocol === 'https:' ?
|
|
|
|
https.request(options, requestCallback) :
|
|
|
|
http.request(options, requestCallback);
|
|
|
|
request.end();
|
|
|
|
return request;
|
|
|
|
}
|
|
|
|
|
|
|
|
function downloadFile(url, destinationPath, progressCallback) {
|
|
|
|
let downloadedBytes = 0;
|
|
|
|
let totalBytes = 0;
|
|
|
|
|
2021-12-10 22:45:43 +03:00
|
|
|
let fulfill, reject;
|
|
|
|
const promise = new Promise((x, y) => { fulfill = x; reject = y; });
|
2021-06-15 11:08:13 +03:00
|
|
|
|
|
|
|
const request = httpRequest(url, 'GET', response => {
|
|
|
|
if (response.statusCode !== 200) {
|
|
|
|
const error = new Error(`Download failed: server returned code ${response.statusCode}. URL: ${url}`);
|
|
|
|
// consume response data to free up memory
|
|
|
|
response.resume();
|
2021-12-10 22:45:43 +03:00
|
|
|
reject(error);
|
2021-06-15 11:08:13 +03:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
const file = fs.createWriteStream(destinationPath);
|
2021-12-10 22:45:43 +03:00
|
|
|
file.on('finish', () => fulfill());
|
|
|
|
file.on('error', error => reject(error));
|
2021-06-15 11:08:13 +03:00
|
|
|
response.pipe(file);
|
|
|
|
totalBytes = parseInt(response.headers['content-length'], 10);
|
|
|
|
if (progressCallback)
|
|
|
|
response.on('data', onData);
|
|
|
|
});
|
2021-12-10 22:45:43 +03:00
|
|
|
request.on('error', error => reject(error));
|
2021-06-15 11:08:13 +03:00
|
|
|
return promise;
|
|
|
|
|
|
|
|
function onData(chunk) {
|
|
|
|
downloadedBytes += chunk.length;
|
|
|
|
progressCallback(downloadedBytes, totalBytes);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function spawnAsync(cmd, args, options) {
|
|
|
|
// console.log(cmd, ...args, 'CWD:', options.cwd);
|
2021-06-17 01:13:51 +03:00
|
|
|
const process = child_process.spawn(cmd, args, options);
|
2021-06-15 11:08:13 +03:00
|
|
|
|
|
|
|
return new Promise(resolve => {
|
|
|
|
let stdout = '';
|
|
|
|
let stderr = '';
|
|
|
|
if (process.stdout)
|
|
|
|
process.stdout.on('data', data => stdout += data);
|
|
|
|
if (process.stderr)
|
|
|
|
process.stderr.on('data', data => stderr += data);
|
2021-11-29 19:42:19 +03:00
|
|
|
process.on('close', code => resolve({ stdout, stderr, code }));
|
|
|
|
process.on('error', error => resolve({ stdout, stderr, code: 0, error }));
|
2021-06-15 11:08:13 +03:00
|
|
|
});
|
|
|
|
}
|
2021-06-17 01:13:51 +03:00
|
|
|
|
|
|
|
function getUbuntuVersionSync() {
|
|
|
|
if (os.platform() !== 'linux')
|
|
|
|
return '';
|
|
|
|
try {
|
|
|
|
let osReleaseText;
|
|
|
|
if (fs.existsSync('/etc/upstream-release/lsb-release'))
|
|
|
|
osReleaseText = fs.readFileSync('/etc/upstream-release/lsb-release', 'utf8');
|
|
|
|
else
|
|
|
|
osReleaseText = fs.readFileSync('/etc/os-release', 'utf8');
|
|
|
|
if (!osReleaseText)
|
|
|
|
return '';
|
|
|
|
return getUbuntuVersionInternal(osReleaseText);
|
2021-11-29 19:42:19 +03:00
|
|
|
} catch (e) {
|
2021-06-17 01:13:51 +03:00
|
|
|
return '';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function getUbuntuVersionInternal(osReleaseText) {
|
|
|
|
const fields = new Map();
|
|
|
|
for (const line of osReleaseText.split('\n')) {
|
|
|
|
const tokens = line.split('=');
|
|
|
|
const name = tokens.shift();
|
|
|
|
let value = tokens.join('=').trim();
|
|
|
|
if (value.startsWith('"') && value.endsWith('"'))
|
|
|
|
value = value.substring(1, value.length - 1);
|
|
|
|
if (!name)
|
|
|
|
continue;
|
|
|
|
fields.set(name.toLowerCase(), value);
|
|
|
|
}
|
|
|
|
// For Linux mint
|
|
|
|
if (fields.get('distrib_id') && fields.get('distrib_id').toLowerCase() === 'ubuntu')
|
|
|
|
return fields.get('distrib_release') || '';
|
|
|
|
if (!fields.get('name') || fields.get('name').toLowerCase() !== 'ubuntu')
|
|
|
|
return '';
|
|
|
|
return fields.get('version_id') || '';
|
|
|
|
}
|
|
|
|
|
|
|
|
function getHostPlatform() {
|
|
|
|
const platform = os.platform();
|
|
|
|
if (platform === 'darwin') {
|
|
|
|
const [major, minor] = child_process.execSync('sw_vers -productVersion', {
|
|
|
|
stdio: ['ignore', 'pipe', 'ignore']
|
|
|
|
}).toString('utf8').trim().split('.').map(x => parseInt(x, 10));
|
|
|
|
let arm64 = false;
|
|
|
|
// BigSur is the first version that might run on Apple Silicon.
|
|
|
|
if (major >= 11) {
|
|
|
|
arm64 = child_process.execSync('/usr/sbin/sysctl -in hw.optional.arm64', {
|
|
|
|
stdio: ['ignore', 'pipe', 'ignore']
|
|
|
|
}).toString().trim() === '1';
|
|
|
|
}
|
|
|
|
const LAST_STABLE_MAC_MAJOR_VERSION = 11;
|
|
|
|
// All new MacOS releases increase major version.
|
|
|
|
let macVersion = `${major}`;
|
|
|
|
if (major === 10) {
|
|
|
|
// Pre-BigSur MacOS was increasing minor version every release.
|
|
|
|
macVersion = `${major}.${minor}`;
|
|
|
|
} else if (major > LAST_STABLE_MAC_MAJOR_VERSION) {
|
|
|
|
// Best-effort support for MacOS beta versions.
|
|
|
|
macVersion = LAST_STABLE_MAC_MAJOR_VERSION + '';
|
|
|
|
}
|
|
|
|
const archSuffix = arm64 ? '-arm64' : '';
|
|
|
|
return `mac${macVersion}${archSuffix}`;
|
|
|
|
}
|
|
|
|
if (platform === 'linux') {
|
|
|
|
const ubuntuVersion = getUbuntuVersionSync();
|
|
|
|
if (parseInt(ubuntuVersion, 10) <= 19)
|
|
|
|
return 'ubuntu18.04';
|
|
|
|
return 'ubuntu20.04';
|
|
|
|
}
|
|
|
|
if (platform === 'win32')
|
2021-10-28 03:20:23 +03:00
|
|
|
return 'win64';
|
2021-06-17 01:13:51 +03:00
|
|
|
return platform;
|
|
|
|
}
|
2021-08-31 14:04:30 +03:00
|
|
|
|
|
|
|
async function main() {
|
2021-12-10 22:45:43 +03:00
|
|
|
const buildInfo = await ensureFirefoxBuild(browserName, process.argv[3], process.argv[4]).catch(e => {
|
|
|
|
console.log(e.message);
|
|
|
|
process.exit(1);
|
|
|
|
});
|
2021-08-31 14:04:30 +03:00
|
|
|
await repackageJuggler(browserName, buildInfo);
|
|
|
|
}
|
|
|
|
|
|
|
|
await main();
|
|
|
|
|