mirror of
https://github.com/balena-os/balena-supervisor.git
synced 2025-01-19 03:06:27 +00:00
652b596c80
We add a bunch of additional unit tests, and also a coverage report using istanbul. The tests are not meant to cover everything, but they're a first attempt at having *some* unit testing on the supervisor. There's much to improve but hopefully it helps catch obvious errors. Change-Type: patch Signed-off-by: Pablo Carranza Velez <pablo@resin.io>
153 lines
5.6 KiB
CoffeeScript
153 lines
5.6 KiB
CoffeeScript
Promise = require 'bluebird'
|
|
prepare = require './lib/prepare'
|
|
|
|
m = require 'mochainon'
|
|
{ expect } = m.chai
|
|
{ stub, spy } = m.sinon
|
|
|
|
fsUtils = require '../src/lib/fs-utils'
|
|
|
|
DeviceConfig = require '../src/device-config'
|
|
|
|
childProcess = require 'child_process'
|
|
|
|
describe 'DeviceConfig', ->
|
|
before ->
|
|
@timeout(5000)
|
|
prepare()
|
|
@fakeDB = {}
|
|
@fakeConfig = {}
|
|
@fakeLogger = {
|
|
logSystemMessage: spy()
|
|
}
|
|
@deviceConfig = new DeviceConfig({ logger: @fakeLogger, db: @fakeDB, config: @fakeConfig })
|
|
|
|
|
|
# Test that the format for special values like initramfs and array variables is parsed correctly
|
|
it 'allows getting boot config with getBootConfig', ->
|
|
stub(@deviceConfig, 'readBootConfig').resolves('\
|
|
initramfs initramf.gz 0x00800000\n\
|
|
dtparam=i2c=on\n\
|
|
dtparam=audio=on\n\
|
|
dtoverlay=ads7846\n\
|
|
dtoverlay=lirc-rpi,gpio_out_pin=17,gpio_in_pin=13\n\
|
|
foobar=baz\n\
|
|
')
|
|
@deviceConfig.getBootConfig('raspberry-pi')
|
|
.then (conf) =>
|
|
@deviceConfig.readBootConfig.restore()
|
|
expect(conf).to.deep.equal({
|
|
RESIN_HOST_CONFIG_initramfs: 'initramf.gz 0x00800000'
|
|
RESIN_HOST_CONFIG_dtparam: '"i2c=on","audio=on"'
|
|
RESIN_HOST_CONFIG_dtoverlay: '"ads7846","lirc-rpi,gpio_out_pin=17,gpio_in_pin=13"'
|
|
RESIN_HOST_CONFIG_foobar: 'baz'
|
|
})
|
|
|
|
it 'properly reads a real config.txt file', ->
|
|
@deviceConfig.getBootConfig('raspberrypi3')
|
|
.then (conf) ->
|
|
expect(conf).to.deep.equal({
|
|
RESIN_HOST_CONFIG_dtparam: '"i2c_arm=on","spi=on","audio=on"'
|
|
RESIN_HOST_CONFIG_enable_uart: '1'
|
|
RESIN_HOST_CONFIG_disable_splash: '1'
|
|
RESIN_HOST_CONFIG_avoid_warnings: '1'
|
|
RESIN_HOST_CONFIG_gpu_mem: '16'
|
|
})
|
|
|
|
it 'correctly transforms environments to boot config objects', ->
|
|
bootConfig = @deviceConfig.envToBootConfig({
|
|
RESIN_HOST_CONFIG_initramfs: 'initramf.gz 0x00800000'
|
|
RESIN_HOST_CONFIG_dtparam: '"i2c=on","audio=on"'
|
|
RESIN_HOST_CONFIG_dtoverlay: '"ads7846","lirc-rpi,gpio_out_pin=17,gpio_in_pin=13"'
|
|
RESIN_HOST_CONFIG_foobar: 'baz'
|
|
})
|
|
expect(bootConfig).to.deep.equal({
|
|
initramfs: 'initramf.gz 0x00800000'
|
|
dtparam: [ 'i2c=on', 'audio=on' ]
|
|
dtoverlay: [ 'ads7846', 'lirc-rpi,gpio_out_pin=17,gpio_in_pin=13' ]
|
|
foobar: 'baz'
|
|
})
|
|
# Test that the format for special values like initramfs and array variables is preserved
|
|
it 'does not allow setting forbidden keys', ->
|
|
current = {
|
|
RESIN_HOST_CONFIG_initramfs: 'initramf.gz 0x00800000'
|
|
RESIN_HOST_CONFIG_dtparam: '"i2c=on","audio=on"'
|
|
RESIN_HOST_CONFIG_dtoverlay: '"ads7846","lirc-rpi,gpio_out_pin=17,gpio_in_pin=13"'
|
|
RESIN_HOST_CONFIG_foobar: 'baz'
|
|
}
|
|
target = {
|
|
RESIN_HOST_CONFIG_initramfs: 'initramf.gz 0x00810000'
|
|
RESIN_HOST_CONFIG_dtparam: '"i2c=on","audio=on"'
|
|
RESIN_HOST_CONFIG_dtoverlay: '"ads7846","lirc-rpi,gpio_out_pin=17,gpio_in_pin=13"'
|
|
RESIN_HOST_CONFIG_foobar: 'baz'
|
|
}
|
|
promise = Promise.try =>
|
|
@deviceConfig.bootConfigChangeRequired('raspberry-pi', current, target)
|
|
expect(promise).to.be.rejected
|
|
promise.catch (err) =>
|
|
expect(@fakeLogger.logSystemMessage).to.be.calledOnce
|
|
expect(@fakeLogger.logSystemMessage).to.be.calledWith('Attempt to change blacklisted config value initramfs', {
|
|
error: 'Attempt to change blacklisted config value initramfs'
|
|
}, 'Apply boot config error')
|
|
@fakeLogger.logSystemMessage.reset()
|
|
|
|
it 'does not try to change config.txt if it should not change', ->
|
|
current = {
|
|
RESIN_HOST_CONFIG_initramfs: 'initramf.gz 0x00800000'
|
|
RESIN_HOST_CONFIG_dtparam: '"i2c=on","audio=on"'
|
|
RESIN_HOST_CONFIG_dtoverlay: '"ads7846","lirc-rpi,gpio_out_pin=17,gpio_in_pin=13"'
|
|
RESIN_HOST_CONFIG_foobar: 'baz'
|
|
}
|
|
target = {
|
|
RESIN_HOST_CONFIG_initramfs: 'initramf.gz 0x00800000'
|
|
RESIN_HOST_CONFIG_dtparam: '"i2c=on","audio=on"'
|
|
RESIN_HOST_CONFIG_dtoverlay: '"ads7846","lirc-rpi,gpio_out_pin=17,gpio_in_pin=13"'
|
|
RESIN_HOST_CONFIG_foobar: 'baz'
|
|
}
|
|
promise = Promise.try =>
|
|
@deviceConfig.bootConfigChangeRequired('raspberry-pi', current, target)
|
|
expect(promise).to.eventually.equal(false)
|
|
promise.then =>
|
|
expect(@fakeLogger.logSystemMessage).to.not.be.called
|
|
@fakeLogger.logSystemMessage.reset()
|
|
|
|
it 'writes the target config.txt', ->
|
|
stub(fsUtils, 'writeFileAtomic').resolves()
|
|
stub(childProcess, 'execAsync').resolves()
|
|
current = {
|
|
RESIN_HOST_CONFIG_initramfs: 'initramf.gz 0x00800000'
|
|
RESIN_HOST_CONFIG_dtparam: '"i2c=on","audio=on"'
|
|
RESIN_HOST_CONFIG_dtoverlay: '"ads7846","lirc-rpi,gpio_out_pin=17,gpio_in_pin=13"'
|
|
RESIN_HOST_CONFIG_foobar: 'baz'
|
|
}
|
|
target = {
|
|
RESIN_HOST_CONFIG_initramfs: 'initramf.gz 0x00800000'
|
|
RESIN_HOST_CONFIG_dtparam: '"i2c=on","audio=off"'
|
|
RESIN_HOST_CONFIG_dtoverlay: '"lirc-rpi,gpio_out_pin=17,gpio_in_pin=13"'
|
|
RESIN_HOST_CONFIG_foobar: 'bat'
|
|
RESIN_HOST_CONFIG_foobaz: 'bar'
|
|
}
|
|
promise = Promise.try =>
|
|
@deviceConfig.bootConfigChangeRequired('raspberry-pi', current, target)
|
|
expect(promise).to.eventually.equal(true)
|
|
promise.then =>
|
|
@deviceConfig.setBootConfig('raspberry-pi', target)
|
|
.then =>
|
|
expect(childProcess.execAsync).to.be.calledOnce
|
|
expect(@fakeLogger.logSystemMessage).to.be.calledTwice
|
|
expect(@fakeLogger.logSystemMessage.getCall(1).args[2]).to.equal('Apply boot config success')
|
|
expect(fsUtils.writeFileAtomic).to.be.calledWith('./test/data/mnt/boot/config.txt', '\
|
|
initramfs initramf.gz 0x00800000\n\
|
|
dtparam=i2c=on\n\
|
|
dtparam=audio=off\n\
|
|
dtoverlay=lirc-rpi,gpio_out_pin=17,gpio_in_pin=13\n\
|
|
foobar=bat\n\
|
|
foobaz=bar\n\
|
|
')
|
|
fsUtils.writeFileAtomic.restore()
|
|
childProcess.execAsync.restore()
|
|
@fakeLogger.logSystemMessage.reset()
|
|
|
|
# This will require stubbing device.reboot, gosuper.post, config.get/set
|
|
it 'applies the target state'
|