1# Copyright (c) 2023 Nordic Semiconductor ASA
2#
3# SPDX-License-Identifier: Apache-2.0
4from __future__ import annotations
5
6import logging
7from pathlib import Path
8
9from test_upgrade import create_signed_image
10from twister_harness import DeviceAdapter, MCUmgr, Shell
11from twister_harness.helpers.utils import find_in_config, match_lines, match_no_lines
12from utils import check_with_mcumgr_command, check_with_shell_command
13
14logger = logging.getLogger(__name__)
15
16
17def test_downgrade_prevention(dut: DeviceAdapter, shell: Shell, mcumgr: MCUmgr):
18    """
19    Verify that the application is not downgraded
20    1) Device flashed with MCUboot and an application that contains SMP server.
21       Image version is 1.1.1+1
22    2) Prepare an update of an application containing the SMP server, where
23       image version is 0.0.0 (lower than version of the original app)
24    3) Upload the application update to slot 1 using mcumgr
25    4) Flag the application update in slot 1 as 'pending' by using mcumgr 'test'
26    5) Restart the device, verify that downgrade prevention mechanism
27       blocked the image swap
28    6) Verify that the original application is booted (version 1.1.1)
29    """
30    origin_version = find_in_config(
31        Path(dut.device_config.app_build_dir) / 'zephyr' / '.config',
32        'CONFIG_MCUBOOT_IMGTOOL_SIGN_VERSION'
33    )
34    check_with_shell_command(shell, origin_version)
35    assert origin_version != '0.0.0+0'
36
37    logger.info('Prepare upgrade image with lower version')
38    image_to_test = create_signed_image(dut.device_config.build_dir,
39                                        dut.device_config.app_build_dir, '0.0.0+0')
40
41    logger.info('Upload image with mcumgr')
42    dut.disconnect()
43    mcumgr.image_upload(image_to_test)
44
45    logger.info('Test uploaded APP image')
46    second_hash = mcumgr.get_hash_to_test()
47    mcumgr.image_test(second_hash)
48    mcumgr.reset_device()
49
50    dut.connect()
51    output = dut.readlines_until('Launching primary slot application')
52    match_no_lines(output, ['Starting swap using move algorithm'])
53    match_lines(output, ['erased due to downgrade prevention'])
54    logger.info('Verify that the original APP is booted')
55    check_with_shell_command(shell, origin_version)
56    dut.disconnect()
57    check_with_mcumgr_command(mcumgr, origin_version)
58