Support for alpha build channel

This commit is contained in:
Scott Nonnenberg 2021-08-06 14:21:01 -07:00 committed by GitHub
parent c0ab1dff11
commit 7ce89414bf
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
15 changed files with 337 additions and 27 deletions

View file

@ -13,7 +13,7 @@ type OptionsType = {
development: boolean; development: boolean;
devTools: boolean; devTools: boolean;
includeSetup: boolean; includeSetup: boolean;
isBeta: (version: string) => boolean; isProduction: boolean;
platform: string; platform: string;
// actions // actions
@ -42,7 +42,7 @@ export const createTemplate = (
} }
const { const {
isBeta, isProduction,
devTools, devTools,
includeSetup, includeSetup,
openContactUs, openContactUs,
@ -212,7 +212,7 @@ export const createTemplate = (
label: messages.goToSupportPage.message, label: messages.goToSupportPage.message,
click: openSupportPage, click: openSupportPage,
}, },
...(!isBeta ...(isProduction
? [ ? [
{ {
label: messages.joinTheBeta.message, label: messages.joinTheBeta.message,

View file

@ -104,7 +104,7 @@ const {
} = require('./app/protocol_filter'); } = require('./app/protocol_filter');
const { installPermissionsHandler } = require('./app/permissions'); const { installPermissionsHandler } = require('./app/permissions');
const OS = require('./ts/OS'); const OS = require('./ts/OS');
const { isBeta } = require('./ts/util/version'); const { isProduction } = require('./ts/util/version');
const { const {
isSgnlHref, isSgnlHref,
isCaptchaHref, isCaptchaHref,
@ -149,7 +149,7 @@ const defaultWebPrefs = {
devTools: devTools:
process.argv.some(arg => arg === '--enable-dev-tools') || process.argv.some(arg => arg === '--enable-dev-tools') ||
config.environment !== Environment.Production || config.environment !== Environment.Production ||
isBeta(app.getVersion()), !isProduction(app.getVersion()),
}; };
async function getSpellCheckSetting() { async function getSpellCheckSetting() {
@ -1400,7 +1400,7 @@ function setupMenu(options) {
const menuOptions = { const menuOptions = {
...options, ...options,
development, development,
isBeta: isBeta(app.getVersion()), isProduction: isProduction(app.getVersion()),
devTools: defaultWebPrefs.devTools, devTools: defaultWebPrefs.devTools,
showDebugLog: showDebugLogWindow, showDebugLog: showDebugLogWindow,
showKeyboardShortcuts, showKeyboardShortcuts,

View file

@ -24,8 +24,10 @@
"clean-module-protobuf": "rm -f ts/protobuf/compiled.d.ts ts/protobuf/compiled.js", "clean-module-protobuf": "rm -f ts/protobuf/compiled.d.ts ts/protobuf/compiled.js",
"build-protobuf": "yarn build-module-protobuf", "build-protobuf": "yarn build-module-protobuf",
"clean-protobuf": "yarn clean-module-protobuf", "clean-protobuf": "yarn clean-module-protobuf",
"prepare-beta-build": "node prepare_beta_build.js", "prepare-beta-build": "node scripts/prepare_beta_build.js",
"prepare-import-build": "node prepare_import_build.js", "prepare-alpha-build": "node scripts/prepare_alpha_build.js",
"prepare-alpha-version": "node scripts/prepare_alpha_version.js",
"prepare-windows-cert": "node scripts/prepare_windows_cert.js",
"publish-to-apt": "NAME=$npm_package_name VERSION=$npm_package_version ./aptly.sh", "publish-to-apt": "NAME=$npm_package_name VERSION=$npm_package_version ./aptly.sh",
"test": "yarn test-node && yarn test-electron", "test": "yarn test-node && yarn test-electron",
"test-electron": "yarn grunt test", "test-electron": "yarn grunt test",
@ -329,9 +331,12 @@
"node_modules/@signalapp/signal-client/build/*.node" "node_modules/@signalapp/signal-client/build/*.node"
], ],
"artifactName": "${name}-win-${version}.${ext}", "artifactName": "${name}-win-${version}.${ext}",
"certificateSubjectName": "Signal (Quiet Riddle Ventures, LLC)", "certificateSubjectName": "Signal Messenger, LLC",
"certificateSha1": "77B2AA4421E5F377454B8B91E573746592D1543D", "certificateSha1": "8C9A0B5C852EC703D83EF7BFBCEB54B796073759",
"publisherName": "Signal (Quiet Riddle Ventures, LLC)", "signingHashAlgorithms": [
"sha256"
],
"publisherName": "Signal Messenger, LLC",
"icon": "build/icons/win/icon.ico", "icon": "build/icons/win/icon.ico",
"publish": [ "publish": [
{ {

View file

@ -0,0 +1,78 @@
// Copyright 2021 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
/* eslint-disable no-console */
const fs = require('fs');
const _ = require('lodash');
const packageJson = require('../package.json');
const { isAlpha } = require('../ts/util/version');
const { version } = packageJson;
// You might be wondering why this file is necessary. It comes down to our desire to allow
// side-by-side installation of production and alpha builds. Electron-Builder uses
// top-level data from package.json for many things, like the executable name, the
// debian package name, the install directory under /opt on linux, etc. We tried
// adding the ${channel} macro to these values, but Electron-Builder didn't like that.
if (!isAlpha(version)) {
process.exit();
}
console.log('prepare_alpha_build: updating package.json');
// -------
const NAME_PATH = 'name';
const PRODUCTION_NAME = 'signal-desktop';
const ALPHA_NAME = 'signal-desktop-alpha';
const PRODUCT_NAME_PATH = 'productName';
const PRODUCTION_PRODUCT_NAME = 'Signal';
const ALPHA_PRODUCT_NAME = 'Signal Alpha';
const APP_ID_PATH = 'build.appId';
const PRODUCTION_APP_ID = 'org.whispersystems.signal-desktop';
const ALPHA_APP_ID = 'org.whispersystems.signal-desktop-alpha';
const STARTUP_WM_CLASS_PATH = 'build.linux.desktop.StartupWMClass';
const PRODUCTION_STARTUP_WM_CLASS = 'Signal';
const ALPHA_STARTUP_WM_CLASS = 'Signal Alpha';
const DESKTOP_NAME_PATH = 'desktopName';
// Note: we're avoiding dashes in our .desktop name due to xdg-settings behavior
// https://github.com/signalapp/Signal-Desktop/issues/3602
const PRODUCTION_DESKTOP_NAME = 'signal.desktop';
const ALPHA_DESKTOP_NAME = 'signalalpha.desktop';
// -------
function checkValue(object, objectPath, expected) {
const actual = _.get(object, objectPath);
if (actual !== expected) {
throw new Error(`${objectPath} was ${actual}; expected ${expected}`);
}
}
// ------
checkValue(packageJson, NAME_PATH, PRODUCTION_NAME);
checkValue(packageJson, PRODUCT_NAME_PATH, PRODUCTION_PRODUCT_NAME);
checkValue(packageJson, APP_ID_PATH, PRODUCTION_APP_ID);
checkValue(packageJson, STARTUP_WM_CLASS_PATH, PRODUCTION_STARTUP_WM_CLASS);
checkValue(packageJson, DESKTOP_NAME_PATH, PRODUCTION_DESKTOP_NAME);
// -------
_.set(packageJson, NAME_PATH, ALPHA_NAME);
_.set(packageJson, PRODUCT_NAME_PATH, ALPHA_PRODUCT_NAME);
_.set(packageJson, APP_ID_PATH, ALPHA_APP_ID);
_.set(packageJson, STARTUP_WM_CLASS_PATH, ALPHA_STARTUP_WM_CLASS);
_.set(packageJson, DESKTOP_NAME_PATH, ALPHA_DESKTOP_NAME);
// -------
fs.writeFileSync('./package.json', JSON.stringify(packageJson, null, ' '));

View file

@ -0,0 +1,33 @@
// Copyright 2021 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
/* eslint-disable no-console */
const fs = require('fs');
const { execSync } = require('child_process');
const _ = require('lodash');
const { generateAlphaVersion } = require('../ts/util/version');
const packageJson = require('../package.json');
const { version: currentVersion } = packageJson;
const shortSha = execSync('git rev-parse --short HEAD')
.toString('utf8')
.replace(/[\n\r]/g, '');
const alphaVersion = generateAlphaVersion({ currentVersion, shortSha });
console.log(
`prepare_alpha_version: updating package.json.\n Previous: ${currentVersion}\n New: ${alphaVersion}`
);
// -------
_.set(packageJson, 'version', alphaVersion);
// -------
fs.writeFileSync('./package.json', JSON.stringify(packageJson, null, ' '));

View file

@ -6,8 +6,8 @@
const fs = require('fs'); const fs = require('fs');
const _ = require('lodash'); const _ = require('lodash');
const packageJson = require('./package.json'); const packageJson = require('../package.json');
const { isBeta } = require('./ts/util/version'); const { isBeta } = require('../ts/util/version');
const { version } = packageJson; const { version } = packageJson;

View file

@ -0,0 +1,50 @@
// Copyright 2018-2020 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
/* eslint-disable no-console */
const fs = require('fs');
const _ = require('lodash');
const packageJson = require('../package.json');
// We have different windows certificates used in each of our build machines, and this
// script makes it easier to ready the app to build on a given machine.
// -------
const KEY = 'build.win.certificateSha1';
const DEFAULT_VALUE = '8C9A0B5C852EC703D83EF7BFBCEB54B796073759';
const BUILDER_A = '507769334DA990A8DDE858314B0CDFC228E7CFA1';
const BUILDER_B = 'C689B0988CA1A7DF99E4CE4433AC7EA8B82F8D41';
let targetValue = DEFAULT_VALUE;
if (process.env.WINDOWS_BUILDER === 'A') {
targetValue = BUILDER_A;
}
if (process.env.WINDOWS_BUILDER === 'B') {
targetValue = BUILDER_B;
}
// -------
function checkValue(object, objectPath, expected) {
const actual = _.get(object, objectPath);
if (actual !== expected) {
throw new Error(`${objectPath} was ${actual}; expected ${expected}`);
}
}
// ------
checkValue(packageJson, KEY, DEFAULT_VALUE);
// -------
_.set(packageJson, KEY, targetValue);
// -------
fs.writeFileSync('./package.json', JSON.stringify(packageJson, null, ' '));

View file

@ -49,7 +49,7 @@ describe('SignalMenu', () => {
}, },
}; };
const options = { const options = {
isBeta: false, isProduction: true,
devTools: true, devTools: true,
openContactUs: null, openContactUs: null,
openForums: null, openForums: null,

View file

@ -2,10 +2,31 @@
// SPDX-License-Identifier: AGPL-3.0-only // SPDX-License-Identifier: AGPL-3.0-only
import { assert } from 'chai'; import { assert } from 'chai';
import { useFakeTimers } from 'sinon';
import * as semver from 'semver';
import { isBeta } from '../../util/version'; import {
generateAlphaVersion,
isAlpha,
isBeta,
isProduction,
} from '../../util/version';
describe('version utilities', () => { describe('version utilities', () => {
describe('isProduction', () => {
it('returns false for anything non-basic version number', () => {
assert.isFalse(isProduction('1.2.3-1'));
assert.isFalse(isProduction('1.2.3-alpha.1'));
assert.isFalse(isProduction('1.2.3-beta.1'));
assert.isFalse(isProduction('1.2.3-rc'));
});
it('returns true for production version strings', () => {
assert.isTrue(isProduction('1.2.3'));
assert.isTrue(isProduction('5.10.0'));
});
});
describe('isBeta', () => { describe('isBeta', () => {
it('returns false for non-beta version strings', () => { it('returns false for non-beta version strings', () => {
assert.isFalse(isBeta('1.2.3')); assert.isFalse(isBeta('1.2.3'));
@ -19,4 +40,88 @@ describe('version utilities', () => {
assert.isTrue(isBeta('1.2.3-beta.1')); assert.isTrue(isBeta('1.2.3-beta.1'));
}); });
}); });
describe('isAlpha', () => {
it('returns false for non-alpha version strings', () => {
assert.isFalse(isAlpha('1.2.3'));
assert.isFalse(isAlpha('1.2.3-beta'));
assert.isFalse(isAlpha('1.2.3-beta.1'));
assert.isFalse(isAlpha('1.2.3-rc.1'));
});
it('returns true for Alpha version strings', () => {
assert.isTrue(isAlpha('1.2.3-alpha'));
assert.isTrue(isAlpha('1.2.3-alpha.1'));
});
});
describe('generateAlphaVersion', () => {
beforeEach(function beforeEach() {
// This isn't a hook.
// eslint-disable-next-line react-hooks/rules-of-hooks
this.clock = useFakeTimers();
});
afterEach(function afterEach() {
this.clock.restore();
});
it('uses the current date and provided shortSha', function test() {
this.clock.setSystemTime(new Date('2021-07-23T01:22:55.692Z').getTime());
const currentVersion = '5.12.0-beta.1';
const shortSha = '07f0efc45';
const expected = '5.12.0-alpha.20210723.01-07f0efc45';
const actual = generateAlphaVersion({ currentVersion, shortSha });
assert.strictEqual(expected, actual);
});
it('same production version is semver.gt', function test() {
const currentVersion = '5.12.0-beta.1';
const shortSha = '07f0efc45';
this.clock.setSystemTime(new Date('2021-07-23T01:22:55.692Z').getTime());
const actual = generateAlphaVersion({ currentVersion, shortSha });
assert.isTrue(semver.gt('5.12.0', actual));
});
it('same beta version is semver.gt', function test() {
const currentVersion = '5.12.0-beta.1';
const shortSha = '07f0efc45';
this.clock.setSystemTime(new Date('2021-07-23T01:22:55.692Z').getTime());
const actual = generateAlphaVersion({ currentVersion, shortSha });
assert.isTrue(semver.gt(currentVersion, actual));
});
it('build earlier same day is semver.lt', function test() {
const currentVersion = '5.12.0-beta.1';
const shortSha = '07f0efc45';
this.clock.setSystemTime(new Date('2021-07-23T00:22:55.692Z').getTime());
const actualEarlier = generateAlphaVersion({ currentVersion, shortSha });
this.clock.setSystemTime(new Date('2021-07-23T01:22:55.692Z').getTime());
const actualLater = generateAlphaVersion({ currentVersion, shortSha });
assert.isTrue(semver.lt(actualEarlier, actualLater));
});
it('build previous day is semver.lt', function test() {
const currentVersion = '5.12.0-beta.1';
const shortSha = '07f0efc45';
this.clock.setSystemTime(new Date('2021-07-22T01:22:55.692Z').getTime());
const actualEarlier = generateAlphaVersion({ currentVersion, shortSha });
this.clock.setSystemTime(new Date('2021-07-23T01:22:55.692Z').getTime());
const actualLater = generateAlphaVersion({ currentVersion, shortSha });
assert.isTrue(semver.lt(actualEarlier, actualLater));
});
});
}); });

View file

@ -2,7 +2,7 @@
// SPDX-License-Identifier: AGPL-3.0-only // SPDX-License-Identifier: AGPL-3.0-only
import * as OS from '../OS'; import * as OS from '../OS';
import { isBeta } from '../util/version'; import { isProduction } from '../util/version';
const MIN_WINDOWS_VERSION = '8.0.0'; const MIN_WINDOWS_VERSION = '8.0.0';
@ -57,4 +57,4 @@ export const getTitleBarVisibility = (): TitleBarVisibility =>
*/ */
export const isSystemTraySupported = (appVersion: string): boolean => export const isSystemTraySupported = (appVersion: string): boolean =>
// We eventually want to support Linux in production. // We eventually want to support Linux in production.
OS.isWindows() || (OS.isLinux() && isBeta(appVersion)); OS.isWindows() || (OS.isLinux() && !isProduction(appVersion));

View file

@ -25,6 +25,7 @@ import { app, BrowserWindow, dialog, ipcMain } from 'electron';
import { getTempPath } from '../../app/attachments'; import { getTempPath } from '../../app/attachments';
import { Dialogs } from '../types/Dialogs'; import { Dialogs } from '../types/Dialogs';
import { getUserAgent } from '../util/getUserAgent'; import { getUserAgent } from '../util/getUserAgent';
import { isAlpha, isBeta } from '../util/version';
import * as packageJson from '../../package.json'; import * as packageJson from '../../package.json';
import { getSignatureFileName } from './signature'; import { getSignatureFileName } from './signature';
@ -260,7 +261,7 @@ export function getProxyUrl(): string | undefined {
} }
export function getUpdatesFileName(): string { export function getUpdatesFileName(): string {
const prefix = isBetaChannel() ? 'beta' : 'latest'; const prefix = getChannel();
if (platform === 'darwin') { if (platform === 'darwin') {
return `${prefix}-mac.yml`; return `${prefix}-mac.yml`;
@ -269,9 +270,16 @@ export function getUpdatesFileName(): string {
return `${prefix}.yml`; return `${prefix}.yml`;
} }
const hasBeta = /beta/i; function getChannel(): string {
function isBetaChannel(): boolean { const { version } = packageJson;
return hasBeta.test(packageJson.version);
if (isAlpha(version)) {
return 'alpha';
}
if (isBeta(version)) {
return 'beta';
}
return 'latest';
} }
function isVersionNewer(newVersion: string): boolean { function isVersionNewer(newVersion: string): boolean {

View file

@ -7,7 +7,7 @@ import {
} from '@signalapp/signal-client'; } from '@signalapp/signal-client';
import { isNumber } from 'lodash'; import { isNumber } from 'lodash';
import { isBeta } from './version'; import { isProduction } from './version';
import { strictAssert } from './assert'; import { strictAssert } from './assert';
import { getSendOptions } from './getSendOptions'; import { getSendOptions } from './getSendOptions';
import { handleMessageSend } from './handleMessageSend'; import { handleMessageSend } from './handleMessageSend';
@ -124,7 +124,7 @@ export async function onRetryRequest(event: RetryRequestEvent): Promise<void> {
} }
function maybeShowDecryptionToast(logId: string) { function maybeShowDecryptionToast(logId: string) {
if (!isBeta(window.getVersion())) { if (isProduction(window.getVersion())) {
return; return;
} }

View file

@ -1,13 +1,13 @@
// Copyright 2020 Signal Messenger, LLC // Copyright 2020 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only // SPDX-License-Identifier: AGPL-3.0-only
import { isBeta } from './version'; import { isProduction } from './version';
import * as RemoteConfig from '../RemoteConfig'; import * as RemoteConfig from '../RemoteConfig';
// We can remove this function once group calling has been turned on for everyone. // We can remove this function once group calling has been turned on for everyone.
export function isGroupCallingEnabled(): boolean { export function isGroupCallingEnabled(): boolean {
return ( return (
RemoteConfig.isEnabled('desktop.groupCalling') || RemoteConfig.isEnabled('desktop.groupCalling') ||
isBeta(window.getVersion()) !isProduction(window.getVersion())
); );
} }

View file

@ -2,7 +2,7 @@
// SPDX-License-Identifier: AGPL-3.0-only // SPDX-License-Identifier: AGPL-3.0-only
import * as RemoteConfig from '../RemoteConfig'; import * as RemoteConfig from '../RemoteConfig';
import { isBeta } from './version'; import { isProduction } from './version';
// We can remove this function once screen sharing has been turned on for everyone // We can remove this function once screen sharing has been turned on for everyone
export function isScreenSharingEnabled(): boolean { export function isScreenSharingEnabled(): boolean {
@ -12,6 +12,6 @@ export function isScreenSharingEnabled(): boolean {
return Boolean( return Boolean(
RemoteConfig.isEnabled('desktop.internalUser') || RemoteConfig.isEnabled('desktop.internalUser') ||
RemoteConfig.isEnabled('desktop.screensharing2') || RemoteConfig.isEnabled('desktop.screensharing2') ||
(version && isBeta(version)) (version && !isProduction(version))
); );
} }

View file

@ -2,6 +2,37 @@
// SPDX-License-Identifier: AGPL-3.0-only // SPDX-License-Identifier: AGPL-3.0-only
import * as semver from 'semver'; import * as semver from 'semver';
import moment from 'moment';
export const isProduction = (version: string): boolean => {
const parsed = semver.parse(version);
if (!parsed) {
return false;
}
return !parsed.prerelease.length && !parsed.build.length;
};
export const isBeta = (version: string): boolean => export const isBeta = (version: string): boolean =>
semver.parse(version)?.prerelease[0] === 'beta'; semver.parse(version)?.prerelease[0] === 'beta';
export const isAlpha = (version: string): boolean =>
semver.parse(version)?.prerelease[0] === 'alpha';
export const generateAlphaVersion = (options: {
currentVersion: string;
shortSha: string;
}): string => {
const { currentVersion, shortSha } = options;
const parsed = semver.parse(currentVersion);
if (!parsed) {
throw new Error(`generateAlphaVersion: Invalid version ${currentVersion}`);
}
const formattedDate = moment().utc().format('YYYYMMDD.HH');
const formattedVersion = `${parsed.major}.${parsed.minor}.${parsed.patch}`;
return `${formattedVersion}-alpha.${formattedDate}-${shortSha}`;
};