Convert settings.js to typescript

Change-type: patch
This commit is contained in:
Alexis Svinartchouk 2020-01-09 15:00:58 +01:00 committed by Lorenzo Alberto Maria Ambrosi
parent c85896845f
commit c0eb9bd1e9
8 changed files with 190 additions and 276 deletions

View File

@ -32,6 +32,7 @@ const messages = require('../../shared/messages')
const store = require('./models/store') const store = require('./models/store')
const packageJSON = require('../../../package.json') const packageJSON = require('../../../package.json')
const flashState = require('./models/flash-state') const flashState = require('./models/flash-state')
// eslint-disable-next-line node/no-missing-require
const settings = require('./models/settings') const settings = require('./models/settings')
// eslint-disable-next-line node/no-missing-require // eslint-disable-next-line node/no-missing-require
const windowProgress = require('./os/window-progress') const windowProgress = require('./os/window-progress')

View File

@ -1,234 +0,0 @@
/*
* Copyright 2016 balena.io
*
* 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.
*/
'use strict'
/**
* @module Etcher.Models.Settings
*/
const _ = require('lodash')
const Bluebird = require('bluebird')
// eslint-disable-next-line node/no-missing-require
const localSettings = require('./local-settings')
// eslint-disable-next-line node/no-missing-require
const errors = require('../../../shared/errors')
const packageJSON = require('../../../../package.json')
const debug = require('debug')('etcher:models:settings')
/**
* @summary Default settings
* @constant
* @type {Object}
*/
const DEFAULT_SETTINGS = {
unsafeMode: false,
errorReporting: true,
unmountOnSuccess: true,
validateWriteOnSuccess: true,
trim: false,
updatesEnabled: packageJSON.updates.enabled && !_.includes([ 'rpm', 'deb' ], packageJSON.packageType),
lastSleptUpdateNotifier: null,
lastSleptUpdateNotifierVersion: null,
desktopNotifications: true
}
/**
* @summary Settings state
* @type {Object}
* @private
*/
let settings = _.cloneDeep(DEFAULT_SETTINGS)
/**
* @summary Reset settings to their default values
* @function
* @public
*
* @returns {Promise}
*
* @example
* settings.reset().then(() => {
* console.log('Done!');
* });
*/
exports.reset = () => {
debug('reset')
// TODO: Remove default settings from config file (?)
settings = _.cloneDeep(DEFAULT_SETTINGS)
return localSettings.writeAll(settings)
}
/**
* @summary Extend the current settings
* @function
* @public
*
* @param {Object} value - value
* @returns {Promise}
*
* @example
* settings.assign({
* foo: 'bar'
* }).then(() => {
* console.log('Done!');
* });
*/
exports.assign = (value) => {
debug('assign', value)
if (_.isNil(value)) {
return Bluebird.reject(errors.createError({
title: 'Missing settings'
}))
}
if (!_.isPlainObject(value)) {
return Bluebird.reject(errors.createError({
title: 'Settings must be an object'
}))
}
const newSettings = _.assign({}, settings, value)
return localSettings.writeAll(newSettings)
.then((updatedSettings) => {
// NOTE: Only update in memory settings when successfully written
settings = updatedSettings
})
}
/**
* @summary Extend the application state with the local settings
* @function
* @public
*
* @returns {Promise}
*
* @example
* settings.load().then(() => {
* console.log('Done!');
* });
*/
exports.load = () => {
debug('load')
return localSettings.readAll().then((loadedSettings) => {
return _.assign(settings, loadedSettings)
})
}
/**
* @summary Set a setting value
* @function
* @public
*
* @param {String} key - setting key
* @param {*} value - setting value
* @returns {Promise}
*
* @example
* settings.set('unmountOnSuccess', true).then(() => {
* console.log('Done!');
* });
*/
exports.set = (key, value) => {
debug('set', key, value)
if (_.isNil(key)) {
return Bluebird.reject(errors.createError({
title: 'Missing setting key'
}))
}
if (!_.isString(key)) {
return Bluebird.reject(errors.createError({
title: `Invalid setting key: ${key}`
}))
}
const previousValue = settings[key]
settings[key] = value
return localSettings.writeAll(settings)
.catch((error) => {
// Revert to previous value if persisting settings failed
settings[key] = previousValue
throw error
})
}
/**
* @summary Get a setting value
* @function
* @public
*
* @param {String} key - setting key
* @returns {*} setting value
*
* @example
* const value = settings.get('unmountOnSuccess');
*/
exports.get = (key) => {
return _.cloneDeep(_.get(settings, [ key ]))
}
/**
* @summary Check if setting value exists
* @function
* @public
*
* @param {String} key - setting key
* @returns {Boolean} exists
*
* @example
* const hasValue = settings.has('unmountOnSuccess');
*/
exports.has = (key) => {
/* eslint-disable no-eq-null */
return settings[key] != null
}
/**
* @summary Get all setting values
* @function
* @public
*
* @returns {Object} all setting values
*
* @example
* const allSettings = settings.getAll();
* console.log(allSettings.unmountOnSuccess);
*/
exports.getAll = () => {
debug('getAll')
return _.cloneDeep(settings)
}
/**
* @summary Get the default setting values
* @function
* @public
*
* @returns {Object} all setting values
*
* @example
* const defaults = settings.getDefaults();
* console.log(defaults.unmountOnSuccess);
*/
exports.getDefaults = () => {
debug('getDefaults')
return _.cloneDeep(DEFAULT_SETTINGS)
}

View File

@ -0,0 +1,142 @@
/*
* Copyright 2016 balena.io
*
* 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.
*/
import * as _debug from 'debug';
import * as _ from 'lodash';
import * as packageJSON from '../../../../package.json';
import * as errors from '../../../shared/errors';
import { Dictionary } from '../../../shared/utils';
import * as localSettings from './local-settings';
const debug = _debug('etcher:models:settings');
const DEFAULT_SETTINGS: Dictionary<any> = {
unsafeMode: false,
errorReporting: true,
unmountOnSuccess: true,
validateWriteOnSuccess: true,
trim: false,
updatesEnabled:
packageJSON.updates.enabled &&
!_.includes(['rpm', 'deb'], packageJSON.packageType),
lastSleptUpdateNotifier: null,
lastSleptUpdateNotifierVersion: null,
desktopNotifications: true,
};
let settings = _.cloneDeep(DEFAULT_SETTINGS);
/**
* @summary Reset settings to their default values
*/
export async function reset(): Promise<void> {
debug('reset');
// TODO: Remove default settings from config file (?)
settings = _.cloneDeep(DEFAULT_SETTINGS);
return await localSettings.writeAll(settings);
}
/**
* @summary Extend the current settings
*/
export async function assign(value: Dictionary<any>): Promise<void> {
debug('assign', value);
if (_.isNil(value)) {
throw errors.createError({
title: 'Missing settings',
});
}
if (!_.isPlainObject(value)) {
throw errors.createError({
title: 'Settings must be an object',
});
}
const newSettings = _.assign({}, settings, value);
const updatedSettings = await localSettings.writeAll(newSettings);
// NOTE: Only update in memory settings when successfully written
settings = updatedSettings;
}
/**
* @summary Extend the application state with the local settings
*/
export async function load(): Promise<void> {
debug('load');
const loadedSettings = await localSettings.readAll();
_.assign(settings, loadedSettings);
}
/**
* @summary Set a setting value
*/
export async function set(key: string, value: any): Promise<void> {
debug('set', key, value);
if (_.isNil(key)) {
throw errors.createError({
title: 'Missing setting key',
});
}
if (!_.isString(key)) {
throw errors.createError({
title: `Invalid setting key: ${key}`,
});
}
const previousValue = settings[key];
settings[key] = value;
try {
await localSettings.writeAll(settings);
} catch (error) {
// Revert to previous value if persisting settings failed
settings[key] = previousValue;
throw error;
}
}
/**
* @summary Get a setting value
*/
export function get(key: string): any {
return _.cloneDeep(_.get(settings, [key]));
}
/**
* @summary Check if setting value exists
*/
export function has(key: string): boolean {
return settings[key] != null;
}
/**
* @summary Get all setting values
*/
export function getAll() {
debug('getAll');
return _.cloneDeep(settings);
}
/**
* @summary Get the default setting values
*/
export function getDefaults() {
debug('getDefaults');
return _.cloneDeep(DEFAULT_SETTINGS);
}

View File

@ -30,6 +30,7 @@ const errors = require('../../../shared/errors')
const fileExtensions = require('../../../shared/file-extensions') const fileExtensions = require('../../../shared/file-extensions')
// eslint-disable-next-line node/no-missing-require // eslint-disable-next-line node/no-missing-require
const utils = require('../../../shared/utils') const utils = require('../../../shared/utils')
// eslint-disable-next-line node/no-missing-require
const settings = require('./settings') const settings = require('./settings')
/** /**

View File

@ -23,6 +23,7 @@ const os = require('os')
const ipc = require('node-ipc') const ipc = require('node-ipc')
const electron = require('electron') const electron = require('electron')
const store = require('../models/store') const store = require('../models/store')
// eslint-disable-next-line node/no-missing-require
const settings = require('../models/settings') const settings = require('../models/settings')
const flashState = require('../models/flash-state') const flashState = require('../models/flash-state')
// eslint-disable-next-line node/no-missing-require // eslint-disable-next-line node/no-missing-require

View File

@ -26,6 +26,7 @@ const semver = require('semver')
const EXIT_CODES = require('../shared/exit-codes') const EXIT_CODES = require('../shared/exit-codes')
// eslint-disable-next-line node/no-missing-require // eslint-disable-next-line node/no-missing-require
const { buildWindowMenu } = require('./menu') const { buildWindowMenu } = require('./menu')
// eslint-disable-next-line node/no-missing-require
const settings = require('./app/models/settings') const settings = require('./app/models/settings')
// eslint-disable-next-line node/no-missing-require // eslint-disable-next-line node/no-missing-require
const analytics = require('./app/modules/analytics') const analytics = require('./app/modules/analytics')

View File

@ -19,10 +19,21 @@
const m = require('mochainon') const m = require('mochainon')
const _ = require('lodash') const _ = require('lodash')
const Bluebird = require('bluebird') const Bluebird = require('bluebird')
// eslint-disable-next-line node/no-missing-require
const settings = require('../../../lib/gui/app/models/settings') const settings = require('../../../lib/gui/app/models/settings')
// eslint-disable-next-line node/no-missing-require // eslint-disable-next-line node/no-missing-require
const localSettings = require('../../../lib/gui/app/models/local-settings') const localSettings = require('../../../lib/gui/app/models/local-settings')
const checkError = async (promise, fn) => {
try {
await promise
} catch (error) {
fn(error)
return
}
throw new Error('Expected error was not thrown')
}
describe('Browser: settings', function () { describe('Browser: settings', function () {
beforeEach(function () { beforeEach(function () {
return settings.reset() return settings.reset()
@ -74,11 +85,10 @@ describe('Browser: settings', function () {
}) })
describe('.assign()', function () { describe('.assign()', function () {
it('should throw if no settings', function (done) { it('should throw if no settings', async function () {
settings.assign().asCallback((error) => { await checkError(settings.assign(), (error) => {
m.chai.expect(error).to.be.an.instanceof(Error) m.chai.expect(error).to.be.an.instanceof(Error)
m.chai.expect(error.message).to.equal('Missing settings') m.chai.expect(error.message).to.equal('Missing settings')
done()
}) })
}) })
@ -109,23 +119,19 @@ describe('Browser: settings', function () {
}) })
}) })
it('should not change the application state if storing to the local machine results in an error', function (done) { it('should not change the application state if storing to the local machine results in an error', async function () {
settings.set('foo', 'bar').then(() => { await settings.set('foo', 'bar')
m.chai.expect(settings.get('foo')).to.equal('bar')
const localSettingsWriteAllStub = m.sinon.stub(localSettings, 'writeAll')
localSettingsWriteAllStub.returns(Bluebird.reject(new Error('localSettings error')))
await checkError(settings.assign({ foo: 'baz' }), (error) => {
m.chai.expect(error).to.be.an.instanceof(Error)
m.chai.expect(error.message).to.equal('localSettings error')
localSettingsWriteAllStub.restore()
m.chai.expect(settings.get('foo')).to.equal('bar') m.chai.expect(settings.get('foo')).to.equal('bar')
})
const localSettingsWriteAllStub = m.sinon.stub(localSettings, 'writeAll')
localSettingsWriteAllStub.returns(Bluebird.reject(new Error('localSettings error')))
settings.assign({
foo: 'baz'
}).asCallback((error) => {
m.chai.expect(error).to.be.an.instanceof(Error)
m.chai.expect(error.message).to.equal('localSettings error')
localSettingsWriteAllStub.restore()
m.chai.expect(settings.get('foo')).to.equal('bar')
done()
})
}).catch(done)
}) })
}) })
@ -161,27 +167,24 @@ describe('Browser: settings', function () {
}) })
}) })
it('should reject if no key', function (done) { it('should reject if no key', async function () {
settings.set(null, true).asCallback((error) => { await checkError(settings.set(null, true), (error) => {
m.chai.expect(error).to.be.an.instanceof(Error) m.chai.expect(error).to.be.an.instanceof(Error)
m.chai.expect(error.message).to.equal('Missing setting key') m.chai.expect(error.message).to.equal('Missing setting key')
done()
}) })
}) })
it('should throw if key is not a string', function (done) { it('should throw if key is not a string', async function () {
settings.set(1234, true).asCallback((error) => { await checkError(settings.set(1234, true), (error) => {
m.chai.expect(error).to.be.an.instanceof(Error) m.chai.expect(error).to.be.an.instanceof(Error)
m.chai.expect(error.message).to.equal('Invalid setting key: 1234') m.chai.expect(error.message).to.equal('Invalid setting key: 1234')
done()
}) })
}) })
it('should throw if setting an array', function (done) { it('should throw if setting an array', async function () {
settings.assign([ 1, 2, 3 ]).asCallback((error) => { await checkError(settings.assign([ 1, 2, 3 ]), (error) => {
m.chai.expect(error).to.be.an.instanceof(Error) m.chai.expect(error).to.be.an.instanceof(Error)
m.chai.expect(error.message).to.equal('Settings must be an object') m.chai.expect(error.message).to.equal('Settings must be an object')
done()
}) })
}) })
@ -203,21 +206,19 @@ describe('Browser: settings', function () {
}) })
}) })
it('should not change the application state if storing to the local machine results in an error', function (done) { it('should not change the application state if storing to the local machine results in an error', async function () {
settings.set('foo', 'bar').then(() => { await settings.set('foo', 'bar')
m.chai.expect(settings.get('foo')).to.equal('bar')
const localSettingsWriteAllStub = m.sinon.stub(localSettings, 'writeAll')
localSettingsWriteAllStub.returns(Bluebird.reject(new Error('localSettings error')))
await checkError(settings.set('foo', 'baz'), (error) => {
m.chai.expect(error).to.be.an.instanceof(Error)
m.chai.expect(error.message).to.equal('localSettings error')
localSettingsWriteAllStub.restore()
m.chai.expect(settings.get('foo')).to.equal('bar') m.chai.expect(settings.get('foo')).to.equal('bar')
})
const localSettingsWriteAllStub = m.sinon.stub(localSettings, 'writeAll')
localSettingsWriteAllStub.returns(Bluebird.reject(new Error('localSettings error')))
settings.set('foo', 'baz').asCallback((error) => {
m.chai.expect(error).to.be.an.instanceof(Error)
m.chai.expect(error.message).to.equal('localSettings error')
localSettingsWriteAllStub.restore()
m.chai.expect(settings.get('foo')).to.equal('bar')
done()
})
}).catch(done)
}) })
}) })

View File

@ -1,6 +1,7 @@
'use strict' 'use strict'
const m = require('mochainon') const m = require('mochainon')
// eslint-disable-next-line node/no-missing-require
const settings = require('../../../lib/gui/app/models/settings') const settings = require('../../../lib/gui/app/models/settings')
// eslint-disable-next-line node/no-missing-require // eslint-disable-next-line node/no-missing-require
const progressStatus = require('../../../lib/gui/app/modules/progress-status') const progressStatus = require('../../../lib/gui/app/modules/progress-status')