forked from espressif/esp-idf
tools: Test IDF Monitor after it is ready to receive data
This commit is contained in:
@@ -558,7 +558,8 @@ test_build_system_cmake:
|
|||||||
test_idf_monitor:
|
test_idf_monitor:
|
||||||
<<: *host_test_template
|
<<: *host_test_template
|
||||||
artifacts:
|
artifacts:
|
||||||
when: on_failure
|
# save artifacts always in order to access results which were retried without consequent failure
|
||||||
|
when: always
|
||||||
paths:
|
paths:
|
||||||
- tools/test_idf_monitor/outputs/*
|
- tools/test_idf_monitor/outputs/*
|
||||||
expire_in: 1 week
|
expire_in: 1 week
|
||||||
|
66
tools/test_idf_monitor/idf_monitor_wrapper.py
Normal file
66
tools/test_idf_monitor/idf_monitor_wrapper.py
Normal file
@@ -0,0 +1,66 @@
|
|||||||
|
# Copyright 2019 Espressif Systems (Shanghai) PTE LTD
|
||||||
|
#
|
||||||
|
# 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.
|
||||||
|
|
||||||
|
from __future__ import unicode_literals
|
||||||
|
import sys
|
||||||
|
import argparse
|
||||||
|
import serial
|
||||||
|
import threading
|
||||||
|
import time
|
||||||
|
from io import open
|
||||||
|
|
||||||
|
try:
|
||||||
|
import idf_monitor
|
||||||
|
except ImportError:
|
||||||
|
sys.path.append('..')
|
||||||
|
import idf_monitor
|
||||||
|
|
||||||
|
|
||||||
|
ELF_FILE = './dummy.elf' # ELF file used for starting the monitor
|
||||||
|
|
||||||
|
|
||||||
|
def monitor_serial_reader_state(serial_reader, file_to_create):
|
||||||
|
"""
|
||||||
|
The pupose of this wrapper is to monitor the serial reader state of idf_monitor.py. file_to_create is created
|
||||||
|
after the serial reader thread has been started. The existence of this file will indicate to
|
||||||
|
run_test_idf_monitor.py that idf_monitor.py is ready to process inputs.
|
||||||
|
"""
|
||||||
|
while not serial_reader.serial.is_open or not serial_reader.alive:
|
||||||
|
time.sleep(1)
|
||||||
|
with open(file_to_create, "w", encoding='utf-8'):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
parser = argparse.ArgumentParser('Wrapper for idf_monitor used for testing')
|
||||||
|
parser.add_argument('--port')
|
||||||
|
parser.add_argument('--print_filter')
|
||||||
|
parser.add_argument('--serial_alive_file')
|
||||||
|
args = parser.parse_args()
|
||||||
|
|
||||||
|
serial_instance = serial.serial_for_url(args.port, 115200, do_not_open=True)
|
||||||
|
monitor = idf_monitor.Monitor(serial_instance, ELF_FILE, args.print_filter, 'make', 'xtensa-esp32-elf-', 'CR')
|
||||||
|
sys.stderr.write('Monitor instance has been created.\n')
|
||||||
|
monitor_thread = threading.Thread(target=monitor_serial_reader_state,
|
||||||
|
args=(monitor.serial_reader, args.serial_alive_file))
|
||||||
|
monitor_thread.start()
|
||||||
|
sys.stderr.write('Monitoring thread has been created.\n')
|
||||||
|
monitor.main_loop()
|
||||||
|
sys.stderr.write('Monitor loop exited.\n')
|
||||||
|
monitor_thread.join(60)
|
||||||
|
sys.stderr.write('Monitoring thread joined.\n')
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
main()
|
@@ -1,6 +1,6 @@
|
|||||||
#!/usr/bin/env python
|
#!/usr/bin/env python
|
||||||
#
|
#
|
||||||
# Copyright 2018 Espressif Systems (Shanghai) PTE LTD
|
# Copyright 2018-2019 Espressif Systems (Shanghai) PTE LTD
|
||||||
#
|
#
|
||||||
# Licensed under the Apache License, Version 2.0 (the "License");
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
# you may not use this file except in compliance with the License.
|
# you may not use this file except in compliance with the License.
|
||||||
@@ -27,6 +27,7 @@ import pty
|
|||||||
import filecmp
|
import filecmp
|
||||||
import threading
|
import threading
|
||||||
import errno
|
import errno
|
||||||
|
import tempfile
|
||||||
|
|
||||||
test_list = (
|
test_list = (
|
||||||
# Add new tests here. All files should be placed in IN_DIR. Columns are:
|
# Add new tests here. All files should be placed in IN_DIR. Columns are:
|
||||||
@@ -41,24 +42,24 @@ test_list = (
|
|||||||
|
|
||||||
IN_DIR = 'tests/' # tests are in this directory
|
IN_DIR = 'tests/' # tests are in this directory
|
||||||
OUT_DIR = 'outputs/' # test results are written to this directory (kept only for debugging purposes)
|
OUT_DIR = 'outputs/' # test results are written to this directory (kept only for debugging purposes)
|
||||||
ERR_OUT = OUT_DIR + 'monitor_error_output'
|
ERR_OUT = 'monitor_error_output.'
|
||||||
ELF_FILE = './dummy.elf' # ELF file used for starting the monitor
|
IDF_MONITOR_WAPPER = 'idf_monitor_wrapper.py'
|
||||||
IDF_MONITOR = '{}/tools/idf_monitor.py'.format(os.getenv("IDF_PATH"))
|
SERIAL_ALIVE_FILE = '/tmp/serial_alive' # the existence of this file signalize that idf_monitor is ready to receive
|
||||||
|
|
||||||
# connection related to communicating with idf_monitor through sockets
|
# connection related to communicating with idf_monitor through sockets
|
||||||
HOST = 'localhost'
|
HOST = 'localhost'
|
||||||
# blocking socket operations are used with timeout:
|
# blocking socket operations are used with timeout:
|
||||||
SOCKET_TIMEOUT = 30
|
SOCKET_TIMEOUT = 30
|
||||||
# the test is restarted after failure (idf_monitor has to be killed):
|
# the test is restarted after failure (idf_monitor has to be killed):
|
||||||
RETRIES_PER_TEST = 5
|
RETRIES_PER_TEST = 3
|
||||||
|
|
||||||
|
|
||||||
def monitor_timeout(process):
|
def monitor_timeout(process):
|
||||||
if process.poll() is None:
|
if process.poll() is None:
|
||||||
# idf_monitor is still running
|
# idf_monitor_wrapper is still running
|
||||||
try:
|
try:
|
||||||
process.kill()
|
process.kill()
|
||||||
print('\tidf_monitor was killed because it did not finish in time.')
|
print('\tidf_monitor_wrapper was killed because it did not finish in time.')
|
||||||
except OSError as e:
|
except OSError as e:
|
||||||
if e.errno == errno.ESRCH:
|
if e.errno == errno.ESRCH:
|
||||||
# ignores a possible race condition which can occur when the process exits between poll() and kill()
|
# ignores a possible race condition which can occur when the process exits between poll() and kill()
|
||||||
@@ -70,7 +71,6 @@ def monitor_timeout(process):
|
|||||||
class TestRunner(object):
|
class TestRunner(object):
|
||||||
def __enter__(self):
|
def __enter__(self):
|
||||||
self.serversocket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
|
self.serversocket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
|
||||||
self.serversocket.setsockopt(socket.SOL_TCP, socket.TCP_NODELAY, 1)
|
|
||||||
self.serversocket.bind((HOST, 0))
|
self.serversocket.bind((HOST, 0))
|
||||||
self.port = self.serversocket.getsockname()[1]
|
self.port = self.serversocket.getsockname()[1]
|
||||||
self.serversocket.listen(5)
|
self.serversocket.listen(5)
|
||||||
@@ -85,49 +85,68 @@ class TestRunner(object):
|
|||||||
""" returns a socket for sending the input for idf_monitor which must be closed before calling this again. """
|
""" returns a socket for sending the input for idf_monitor which must be closed before calling this again. """
|
||||||
(clientsocket, address) = self.serversocket.accept()
|
(clientsocket, address) = self.serversocket.accept()
|
||||||
# exception will be thrown here if the idf_monitor didn't connect in time
|
# exception will be thrown here if the idf_monitor didn't connect in time
|
||||||
clientsocket.setsockopt(socket.SOL_TCP, socket.TCP_NODELAY, 1)
|
|
||||||
return clientsocket
|
return clientsocket
|
||||||
|
|
||||||
|
|
||||||
def test_iteration(runner, test, startup_timeout):
|
def test_iteration(runner, test):
|
||||||
|
try:
|
||||||
|
# Make sure that the file doesn't exist. It will be recreated by idf_monitor_wrapper.py
|
||||||
|
os.remove(SERIAL_ALIVE_FILE)
|
||||||
|
except OSError:
|
||||||
|
pass
|
||||||
print('\nRunning test on {} with filter "{}" and expecting {}'.format(test[0], test[1], test[2]))
|
print('\nRunning test on {} with filter "{}" and expecting {}'.format(test[0], test[1], test[2]))
|
||||||
try:
|
try:
|
||||||
with open(OUT_DIR + test[2], "w", encoding='utf-8') as o_f, open(ERR_OUT, "w", encoding='utf-8') as e_f:
|
with open(OUT_DIR + test[2], "w", encoding='utf-8') as o_f, \
|
||||||
monitor_cmd = [sys.executable,
|
tempfile.NamedTemporaryFile(dir=OUT_DIR, prefix=ERR_OUT, mode="w", delete=False) as e_f:
|
||||||
IDF_MONITOR, '--port', 'socket://{}:{}'.format(HOST, runner.port), '--print_filter', test[1], ELF_FILE]
|
monitor_cmd = [sys.executable, IDF_MONITOR_WAPPER,
|
||||||
|
'--port', 'socket://{}:{}?logging=debug'.format(HOST, runner.port),
|
||||||
|
'--print_filter', test[1],
|
||||||
|
'--serial_alive_file', SERIAL_ALIVE_FILE]
|
||||||
(master_fd, slave_fd) = pty.openpty()
|
(master_fd, slave_fd) = pty.openpty()
|
||||||
print('\t', ' '.join(monitor_cmd), sep='')
|
print('\t', ' '.join(monitor_cmd), sep='')
|
||||||
print('\tstdout="{}" stderr="{}" stdin="{}"'.format(o_f.name, e_f.name, os.ttyname(slave_fd)))
|
print('\tstdout="{}" stderr="{}" stdin="{}"'.format(o_f.name, e_f.name, os.ttyname(slave_fd)))
|
||||||
print('\tMonitor timeout: {} seconds'.format(test[3]))
|
print('\tMonitor timeout: {} seconds'.format(test[3]))
|
||||||
start = time.time()
|
start = time.time()
|
||||||
# the server socket is alive so idf_monitor can start now
|
# the server socket is alive so idf_monitor can start now
|
||||||
proc = subprocess.Popen(monitor_cmd, stdin=slave_fd, stdout=o_f, stderr=e_f, close_fds=True, bufsize=-1)
|
proc = subprocess.Popen(monitor_cmd, stdin=slave_fd, stdout=o_f, stderr=e_f, close_fds=True, bufsize=0)
|
||||||
# - idf_monitor's stdin needs to be connected to some pseudo-tty in docker image even when it is not
|
# - idf_monitor's stdin needs to be connected to some pseudo-tty in docker image even when it is not
|
||||||
# used at all
|
# used at all
|
||||||
# - setting bufsize is needed because the default value is different on Python 2 and 3
|
# - setting bufsize is needed because the default value is different on Python 2 and 3
|
||||||
# - the default close_fds is also different on Python 2 and 3
|
# - the default close_fds is also different on Python 2 and 3
|
||||||
monitor_watchdog = threading.Timer(test[3], monitor_timeout, [proc])
|
monitor_watchdog = threading.Timer(test[3], monitor_timeout, [proc])
|
||||||
monitor_watchdog.start()
|
monitor_watchdog.start()
|
||||||
client = runner.accept_connection()
|
client = runner.accept_connection()
|
||||||
# The connection is ready but idf_monitor cannot yet receive data. This seems to happen on Ubuntu 16.04 LTS
|
# The connection is ready but idf_monitor cannot yet receive data (the serial reader thread is not running).
|
||||||
# and is not related to the version of Python or pyserial. There seems to be no reliable way other than to
|
# This seems to happen on Ubuntu 16.04 LTS and is not related to the version of Python or pyserial.
|
||||||
# do a sleep. The idf_monitor header in ERR_OUT could be checked on Python 2 but the file is not flushed
|
# Updating to Ubuntu 18.04 LTS also helps but here, a workaround is used: A wrapper is used for IDF monitor
|
||||||
# on Python 3.
|
# which checks the serial reader thread and creates a file when it is running.
|
||||||
print('\tSleeping for {:.2f} seconds'.format(startup_timeout))
|
while not os.path.isfile(SERIAL_ALIVE_FILE) and proc.poll() is None:
|
||||||
time.sleep(startup_timeout)
|
print('\tSerial reader is not ready. Do a sleep...')
|
||||||
with open(IN_DIR + test[0], "r", encoding='utf-8') as f:
|
time.sleep(1)
|
||||||
print('\tSending {} to the socket'.format(f.name))
|
# Only now can we send the inputs:
|
||||||
for line in f:
|
with open(IN_DIR + test[0], 'rb') as f:
|
||||||
client.sendall(line.encode())
|
print('\tSending {} to the socket'.format(f.name))
|
||||||
idf_exit_sequence = b'\x1d\n'
|
for chunk in iter(lambda: f.read(1024), b''):
|
||||||
print('\tSending <exit> to the socket')
|
client.sendall(chunk)
|
||||||
client.sendall(idf_exit_sequence)
|
idf_exit_sequence = b'\x1d\n'
|
||||||
ret = proc.wait()
|
print('\tSending <exit> to the socket')
|
||||||
end = time.time()
|
client.sendall(idf_exit_sequence)
|
||||||
print('\tidf_monitor exited after {:.2f} seconds'.format(end - start))
|
close_end_time = start + 0.75 * test[3] # time when the process is close to be killed
|
||||||
if ret < 0:
|
while True:
|
||||||
raise RuntimeError('idf_monitor was terminated by signal {}'.format(-ret))
|
ret = proc.poll()
|
||||||
# idf_monitor needs to end before the socket is closed in order to exit without an exception.
|
if ret is not None:
|
||||||
|
break
|
||||||
|
if time.time() > close_end_time:
|
||||||
|
# The process isn't finished yet so we are starting to send additional exit sequences because maybe
|
||||||
|
# the other end didn't received it.
|
||||||
|
print('\tSending additional <exit> to the socket')
|
||||||
|
client.sendall(idf_exit_sequence)
|
||||||
|
time.sleep(1)
|
||||||
|
end = time.time()
|
||||||
|
print('\tidf_monitor exited after {:.2f} seconds'.format(end - start))
|
||||||
|
if ret < 0:
|
||||||
|
raise RuntimeError('idf_monitor was terminated by signal {}'.format(-ret))
|
||||||
|
# idf_monitor needs to end before the socket is closed in order to exit without an exception.
|
||||||
finally:
|
finally:
|
||||||
if monitor_watchdog:
|
if monitor_watchdog:
|
||||||
monitor_watchdog.cancel()
|
monitor_watchdog.cancel()
|
||||||
@@ -153,21 +172,19 @@ def main():
|
|||||||
socket.setdefaulttimeout(SOCKET_TIMEOUT)
|
socket.setdefaulttimeout(SOCKET_TIMEOUT)
|
||||||
|
|
||||||
for test in test_list:
|
for test in test_list:
|
||||||
startup_timeout = 1.0
|
|
||||||
for i in range(RETRIES_PER_TEST):
|
for i in range(RETRIES_PER_TEST):
|
||||||
with TestRunner() as runner:
|
with TestRunner() as runner:
|
||||||
# Each test (and each retry) is run with a different port (and server socket). This is done for
|
# Each test (and each retry) is run with a different port (and server socket). This is done for
|
||||||
# the CI run where retry with a different socket is necessary to pass the test. According to the
|
# the CI run where retry with a different socket is necessary to pass the test. According to the
|
||||||
# experiments, retry with the same port (and server socket) is not sufficient.
|
# experiments, retry with the same port (and server socket) is not sufficient.
|
||||||
try:
|
try:
|
||||||
test_iteration(runner, test, startup_timeout)
|
test_iteration(runner, test)
|
||||||
# no more retries if test_iteration exited without an exception
|
# no more retries if test_iteration exited without an exception
|
||||||
break
|
break
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
if i < RETRIES_PER_TEST - 1:
|
if i < RETRIES_PER_TEST - 1:
|
||||||
print('Test has failed with exception:', e)
|
print('Test has failed with exception:', e)
|
||||||
print('Another attempt will be made.')
|
print('Another attempt will be made.')
|
||||||
startup_timeout += 1
|
|
||||||
else:
|
else:
|
||||||
raise
|
raise
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user