2019-07-05 11:59:06 +02:00
|
|
|
#!/usr/bin/env python3
|
|
|
|
|
|
|
|
# Copyright (C) 2019 Inria
|
|
|
|
#
|
|
|
|
# This file is subject to the terms and conditions of the GNU Lesser
|
|
|
|
# General Public License v2.1. See the file LICENSE in the top level
|
|
|
|
# directory for more details.
|
|
|
|
|
|
|
|
import os
|
|
|
|
import subprocess
|
|
|
|
import sys
|
|
|
|
import tempfile
|
2019-10-18 13:21:21 +02:00
|
|
|
import time
|
2022-04-22 13:27:55 +02:00
|
|
|
import re
|
|
|
|
import random
|
|
|
|
from ipaddress import (
|
|
|
|
IPv6Address,
|
|
|
|
IPv6Network,
|
|
|
|
)
|
2019-07-05 11:59:06 +02:00
|
|
|
|
|
|
|
from testrunner import run
|
2020-11-06 14:03:02 +01:00
|
|
|
from testrunner import utils
|
2019-07-05 11:59:06 +02:00
|
|
|
|
2022-04-22 13:27:55 +02:00
|
|
|
COAP_HOST = os.getenv("SUIT_COAP_SERVER", "[fd00:dead:beef::1]")
|
|
|
|
BOARD = os.getenv("BOARD", "samr21-xpro")
|
2019-07-05 11:59:06 +02:00
|
|
|
|
|
|
|
UPDATING_TIMEOUT = 10
|
|
|
|
MANIFEST_TIMEOUT = 15
|
|
|
|
|
|
|
|
USE_ETHOS = int(os.getenv("USE_ETHOS", "1"))
|
2022-04-22 13:27:55 +02:00
|
|
|
IFACE = os.getenv("IFACE", "tapbr0")
|
2019-07-05 11:59:06 +02:00
|
|
|
TMPDIR = tempfile.TemporaryDirectory()
|
|
|
|
|
|
|
|
|
2022-04-22 13:27:55 +02:00
|
|
|
def get_iface_addr(iface):
|
|
|
|
out = subprocess.check_output(["ip", "a", "s", "dev", iface]).decode()
|
|
|
|
p = re.compile(
|
|
|
|
r"inet6\s+(?P<global>[0-9a-fA-F:]+:[A-Fa-f:0-9]+/\d+)\s+" r"scope\s+global"
|
|
|
|
)
|
|
|
|
for line in out.splitlines():
|
|
|
|
m = p.search(line)
|
|
|
|
if m is not None:
|
|
|
|
return m.group("global")
|
|
|
|
return None
|
|
|
|
|
|
|
|
|
2019-07-05 11:59:06 +02:00
|
|
|
def start_aiocoap_fileserver():
|
|
|
|
aiocoap_process = subprocess.Popen(
|
|
|
|
"exec aiocoap-fileserver %s" % TMPDIR.name, shell=True
|
|
|
|
)
|
|
|
|
|
|
|
|
return aiocoap_process
|
|
|
|
|
|
|
|
|
|
|
|
def cleanup(aiocoap_process):
|
|
|
|
aiocoap_process.kill()
|
|
|
|
TMPDIR.cleanup()
|
|
|
|
|
|
|
|
|
|
|
|
def notify(coap_server, client_url, version=None):
|
|
|
|
cmd = [
|
|
|
|
"make",
|
|
|
|
"suit/notify",
|
|
|
|
"SUIT_COAP_SERVER={}".format(coap_server),
|
|
|
|
"SUIT_CLIENT={}".format(client_url),
|
|
|
|
]
|
|
|
|
if version is not None:
|
|
|
|
cmd.append("SUIT_NOTIFY_VERSION={}".format(version))
|
|
|
|
assert not subprocess.call(cmd)
|
|
|
|
|
|
|
|
|
2022-04-22 13:24:01 +02:00
|
|
|
def publish(server_dir, server_url, app_ver, keys="default", latest_name=None):
|
2019-07-05 11:59:06 +02:00
|
|
|
cmd = [
|
|
|
|
"make",
|
|
|
|
"suit/publish",
|
|
|
|
"SUIT_COAP_FSROOT={}".format(server_dir),
|
|
|
|
"SUIT_COAP_SERVER={}".format(server_url),
|
|
|
|
"APP_VER={}".format(app_ver),
|
|
|
|
"RIOTBOOT_SKIP_COMPILE=1",
|
2019-08-07 13:22:59 +02:00
|
|
|
"SUIT_KEY={}".format(keys),
|
2019-07-05 11:59:06 +02:00
|
|
|
]
|
|
|
|
if latest_name is not None:
|
|
|
|
cmd.append("SUIT_MANIFEST_SIGNED_LATEST={}".format(latest_name))
|
|
|
|
|
|
|
|
assert not subprocess.call(cmd)
|
|
|
|
|
|
|
|
|
|
|
|
def wait_for_update(child):
|
2022-04-22 13:24:01 +02:00
|
|
|
return child.expect(
|
|
|
|
[r"Fetching firmware \|[█ ]+\|\s+\d+\%", "Finalizing payload store"],
|
|
|
|
timeout=UPDATING_TIMEOUT,
|
|
|
|
)
|
2019-07-05 11:59:06 +02:00
|
|
|
|
|
|
|
|
|
|
|
def get_ipv6_addr(child):
|
2022-04-22 13:24:01 +02:00
|
|
|
child.expect_exact(">")
|
|
|
|
child.sendline("ifconfig")
|
2019-07-05 11:59:06 +02:00
|
|
|
if USE_ETHOS == 0:
|
|
|
|
# Get device global address
|
|
|
|
child.expect(
|
|
|
|
r"inet6 addr: (?P<gladdr>[0-9a-fA-F:]+:[A-Fa-f:0-9]+)"
|
|
|
|
" scope: global VAL"
|
|
|
|
)
|
|
|
|
addr = child.match.group("gladdr").lower()
|
|
|
|
else:
|
|
|
|
# Get device local address
|
|
|
|
child.expect_exact("Link type: wired")
|
|
|
|
child.expect(
|
2022-04-22 13:24:01 +02:00
|
|
|
r"inet6 addr: (?P<lladdr>[0-9a-fA-F:]+:[A-Fa-f:0-9]+)" " scope: link VAL"
|
2019-07-05 11:59:06 +02:00
|
|
|
)
|
2022-04-22 13:27:55 +02:00
|
|
|
addr = "{}%{}".format(child.match.group("lladdr").lower(), IFACE)
|
2019-07-05 11:59:06 +02:00
|
|
|
return addr
|
|
|
|
|
|
|
|
|
|
|
|
def ping6(client):
|
|
|
|
print("pinging node...")
|
|
|
|
ping_ok = False
|
|
|
|
for _i in range(10):
|
|
|
|
try:
|
|
|
|
subprocess.check_call(["ping", "-q", "-c1", "-w1", client])
|
|
|
|
ping_ok = True
|
|
|
|
break
|
|
|
|
except subprocess.CalledProcessError:
|
|
|
|
pass
|
|
|
|
|
|
|
|
if not ping_ok:
|
|
|
|
print("pinging node failed. aborting test.")
|
|
|
|
sys.exit(1)
|
|
|
|
else:
|
|
|
|
print("pinging node succeeded.")
|
|
|
|
return ping_ok
|
|
|
|
|
|
|
|
|
2019-08-07 13:22:59 +02:00
|
|
|
def get_reachable_addr(child):
|
2020-11-06 14:03:02 +01:00
|
|
|
# Give some time for the network interface to be configured
|
2019-08-07 13:22:59 +02:00
|
|
|
time.sleep(1)
|
|
|
|
# Get address
|
2022-04-22 13:27:55 +02:00
|
|
|
if BOARD == "native":
|
|
|
|
iface_addr = get_iface_addr(IFACE)
|
|
|
|
network = IPv6Network(iface_addr, strict=False)
|
|
|
|
client_addr = iface_addr
|
|
|
|
while iface_addr == client_addr:
|
|
|
|
client_addr = IPv6Address(
|
|
|
|
random.randrange(
|
|
|
|
int(network.network_address) + 1, int(network.broadcast_address) - 1
|
|
|
|
)
|
|
|
|
)
|
|
|
|
child.sendline(f"ifconfig 5 add {client_addr}/{format(network.prefixlen)}")
|
|
|
|
client_addr = format(client_addr)
|
|
|
|
else:
|
|
|
|
client_addr = get_ipv6_addr(child)
|
2019-08-07 13:22:59 +02:00
|
|
|
# Verify address is reachable
|
|
|
|
ping6(client_addr)
|
|
|
|
return "[{}]".format(client_addr)
|
2019-07-05 11:59:06 +02:00
|
|
|
|
|
|
|
|
2022-04-21 10:56:00 +02:00
|
|
|
def seq_no(child):
|
2020-11-06 14:03:02 +01:00
|
|
|
utils.test_utils_interactive_sync_shell(child, 5, 1)
|
2019-07-05 11:59:06 +02:00
|
|
|
# get version of currently running image
|
2022-04-21 10:56:00 +02:00
|
|
|
# "seq_no: 0x00000000"
|
2022-04-22 13:24:01 +02:00
|
|
|
child.sendline("suit seq_no")
|
2022-04-21 10:56:00 +02:00
|
|
|
child.expect(r"seq_no: (?P<seq_no>0x[0-9a-fA-F:]+)\r\n")
|
|
|
|
app_ver = int(child.match.group("seq_no"), 16)
|
2019-08-07 13:22:59 +02:00
|
|
|
return app_ver
|
|
|
|
|
|
|
|
|
2020-11-06 14:03:02 +01:00
|
|
|
def running_slot(child):
|
|
|
|
utils.test_utils_interactive_sync_shell(child, 5, 1)
|
2022-04-21 10:56:00 +02:00
|
|
|
|
2022-04-22 13:24:01 +02:00
|
|
|
child.sendline("current-slot")
|
2020-11-06 14:03:02 +01:00
|
|
|
child.expect(r"Running from slot (\d+)\r\n")
|
|
|
|
slot = int(child.match.group(1))
|
|
|
|
return slot
|
|
|
|
|
|
|
|
|
2019-08-07 13:22:59 +02:00
|
|
|
def _test_invalid_version(child, client, app_ver):
|
2022-04-21 10:56:32 +02:00
|
|
|
publish(TMPDIR.name, COAP_HOST, app_ver)
|
|
|
|
notify(COAP_HOST, client, app_ver)
|
2019-08-07 13:22:59 +02:00
|
|
|
child.expect_exact("suit_coap: trigger received")
|
2020-02-26 14:52:39 +01:00
|
|
|
child.expect_exact("suit: verifying manifest signature")
|
2019-08-07 13:22:59 +02:00
|
|
|
child.expect_exact("seq_nr <= running image")
|
|
|
|
|
|
|
|
|
|
|
|
def _test_invalid_signature(child, client, app_ver):
|
2022-04-22 13:24:01 +02:00
|
|
|
publish(TMPDIR.name, COAP_HOST, app_ver + 1, "invalid_keys")
|
2019-08-07 13:22:59 +02:00
|
|
|
notify(COAP_HOST, client, app_ver + 1)
|
|
|
|
child.expect_exact("suit_coap: trigger received")
|
2020-02-26 14:52:39 +01:00
|
|
|
child.expect_exact("suit: verifying manifest signature")
|
2019-08-07 13:22:59 +02:00
|
|
|
child.expect_exact("Unable to validate signature")
|
|
|
|
|
|
|
|
|
|
|
|
def _test_successful_update(child, client, app_ver):
|
|
|
|
for version in [app_ver + 1, app_ver + 2]:
|
2019-07-05 11:59:06 +02:00
|
|
|
# Trigger update process, verify it validates manifest correctly
|
|
|
|
publish(TMPDIR.name, COAP_HOST, version)
|
|
|
|
notify(COAP_HOST, client, version)
|
|
|
|
child.expect_exact("suit_coap: trigger received")
|
2020-02-26 14:52:39 +01:00
|
|
|
child.expect_exact("suit: verifying manifest signature")
|
2019-07-05 11:59:06 +02:00
|
|
|
child.expect(
|
2022-04-22 13:27:55 +02:00
|
|
|
r"SUIT policy check OK.\r\n",
|
2019-07-05 11:59:06 +02:00
|
|
|
timeout=MANIFEST_TIMEOUT,
|
|
|
|
)
|
|
|
|
# Wait for update to complete
|
|
|
|
while wait_for_update(child) == 0:
|
|
|
|
pass
|
2022-04-22 13:27:55 +02:00
|
|
|
# Check successful install
|
|
|
|
child.expect_exact("Install correct payload")
|
|
|
|
child.expect_exact("Install correct payload")
|
|
|
|
|
|
|
|
# Wait for reboot on non-native BOARDs
|
|
|
|
if BOARD != "native":
|
|
|
|
child.expect_exact("suit_coap: rebooting...")
|
|
|
|
# Verify client is reachable and get address
|
|
|
|
client = get_reachable_addr(child)
|
|
|
|
assert seq_no(child) == version
|
2019-08-07 13:22:59 +02:00
|
|
|
|
|
|
|
|
2020-12-31 15:30:27 +01:00
|
|
|
def _test_suit_command_is_there(child):
|
2022-04-22 13:24:01 +02:00
|
|
|
child.sendline("suit")
|
2022-04-19 13:05:02 +02:00
|
|
|
child.expect_exact("Usage: suit fetch <manifest url>")
|
2020-12-31 15:30:27 +01:00
|
|
|
|
|
|
|
|
2019-08-07 13:22:59 +02:00
|
|
|
def testfunc(child):
|
|
|
|
# Get current app_ver
|
2022-04-21 10:56:00 +02:00
|
|
|
current_app_ver = seq_no(child)
|
2019-08-07 13:22:59 +02:00
|
|
|
# Verify client is reachable and get address
|
|
|
|
client = get_reachable_addr(child)
|
|
|
|
|
2020-12-31 15:30:27 +01:00
|
|
|
# Verify the suit shell command is there
|
|
|
|
_test_suit_command_is_there(child)
|
|
|
|
|
2019-08-07 13:22:59 +02:00
|
|
|
def run(func):
|
|
|
|
if child.logfile == sys.stdout:
|
|
|
|
func(child, client, current_app_ver)
|
|
|
|
else:
|
|
|
|
try:
|
|
|
|
func(child, client, current_app_ver)
|
|
|
|
print(".", end="", flush=True)
|
|
|
|
except Exception as e:
|
|
|
|
print("FAILED")
|
|
|
|
raise e
|
|
|
|
|
|
|
|
run(_test_invalid_signature)
|
|
|
|
run(_test_invalid_version)
|
|
|
|
run(_test_successful_update)
|
2019-07-05 11:59:06 +02:00
|
|
|
|
|
|
|
print("TEST PASSED")
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == "__main__":
|
|
|
|
try:
|
|
|
|
res = 1
|
|
|
|
aiocoap_process = start_aiocoap_fileserver()
|
|
|
|
# TODO: wait for coap port to be available
|
|
|
|
res = run(testfunc, echo=True)
|
|
|
|
|
|
|
|
except Exception as e:
|
|
|
|
print(e)
|
|
|
|
finally:
|
|
|
|
cleanup(aiocoap_process)
|
|
|
|
|
|
|
|
sys.exit(res)
|