161 lines
5.7 KiB
Python
161 lines
5.7 KiB
Python
#!/usr/bin/env python3
|
|
#
|
|
# Copyright (c) 2019, The OpenThread Authors.
|
|
# All rights reserved.
|
|
#
|
|
# Redistribution and use in source and binary forms, with or without
|
|
# modification, are permitted provided that the following conditions are met:
|
|
# 1. Redistributions of source code must retain the above copyright
|
|
# notice, this list of conditions and the following disclaimer.
|
|
# 2. Redistributions in binary form must reproduce the above copyright
|
|
# notice, this list of conditions and the following disclaimer in the
|
|
# documentation and/or other materials provided with the distribution.
|
|
# 3. Neither the name of the copyright holder nor the
|
|
# names of its contributors may be used to endorse or promote products
|
|
# derived from this software without specific prior written permission.
|
|
#
|
|
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
|
# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
|
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
|
# ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
|
|
# LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
# CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
|
# SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
|
|
# INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
# CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
# ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
|
|
# POSSIBILITY OF SUCH DAMAGE.
|
|
#
|
|
import logging
|
|
import os
|
|
import sys
|
|
from typing import Callable, Union
|
|
|
|
from pktverify.addrs import EthAddr, ExtAddr, Ipv6Addr
|
|
from pktverify.bytes import Bytes
|
|
from pktverify.null_field import nullField
|
|
|
|
|
|
def make_filter_func(func: Union[str, Callable], **vars) -> Callable:
|
|
"""
|
|
Convert the filter to a callable function if it's a string.
|
|
|
|
:param func: The filter string or callable.
|
|
:param vars: The variables.
|
|
:return: The filter callable.
|
|
"""
|
|
if isinstance(func, str):
|
|
# if func is a string, compile it to a function
|
|
func = func.format_map({k: repr(v) for k, v in vars.items()}).strip()
|
|
print("\t%s" % func, file=sys.stderr)
|
|
code = compile('(\n' + func + '\n)', func, "eval")
|
|
|
|
def func(p):
|
|
return eval(
|
|
code, None, {
|
|
'p': p,
|
|
'coap': p.coap,
|
|
'wpan': p.wpan,
|
|
'mle': p.mle,
|
|
'ipv6': p.ipv6,
|
|
'lowpan': p.lowpan,
|
|
'eth': p.eth,
|
|
'icmpv6': p.icmpv6,
|
|
'udp': p.udp,
|
|
'thread_bl': p.thread_bl,
|
|
'thread_meshcop': p.thread_meshcop,
|
|
'Bytes': Bytes,
|
|
'ExtAddr': ExtAddr,
|
|
'Ipv6Addr': Ipv6Addr,
|
|
'EthAddr': EthAddr,
|
|
'thread_nm': p.thread_nm,
|
|
'thread_nwd': p.thread_nwd,
|
|
'thread_address': p.thread_address,
|
|
'thread_bcn': p.thread_bcn,
|
|
'dns': p.dns,
|
|
'null': nullField,
|
|
})
|
|
else:
|
|
assert not vars, 'can not provide vars for non-str filter: %r %r' % (func, vars)
|
|
|
|
assert callable(func)
|
|
return func
|
|
|
|
|
|
def _setup_wireshark_disabled_protos():
|
|
home = os.environ['HOME']
|
|
wireshark_config_dir = os.path.join(home, '.config', 'wireshark')
|
|
os.makedirs(wireshark_config_dir, exist_ok=True)
|
|
disabled_protos_path = os.path.join(wireshark_config_dir, 'disabled_protos')
|
|
# read current disabled protos
|
|
try:
|
|
with open(disabled_protos_path, 'rt') as fd:
|
|
disabled_protos = set(l.strip() for l in fd if l.strip() != '')
|
|
except FileNotFoundError:
|
|
disabled_protos = set()
|
|
|
|
old_disabled_protos_num = len(disabled_protos)
|
|
disabled_protos.add('lwm')
|
|
disabled_protos.add('prp')
|
|
disabled_protos.add('stcsig')
|
|
disabled_protos.add('transum')
|
|
disabled_protos.add('zbee_nwk')
|
|
disabled_protos.add('zbee_nwk_gp')
|
|
|
|
if len(disabled_protos) > old_disabled_protos_num:
|
|
logging.info(f"set disabled_protos = {' '.join(disabled_protos)}")
|
|
with open(disabled_protos_path, 'wt') as fd:
|
|
fd.write('\n'.join(sorted(disabled_protos)))
|
|
fd.write('\n')
|
|
|
|
|
|
def get_wireshark_dir() -> str:
|
|
"""
|
|
:return: The path to wireshark directory.
|
|
"""
|
|
dir = '/tmp/thread-wireshark'
|
|
_setup_wireshark_disabled_protos()
|
|
return dir
|
|
|
|
|
|
def which_tshark() -> str:
|
|
"""
|
|
:return: The path to `tshark` executable.
|
|
"""
|
|
return os.path.join(get_wireshark_dir(), 'tshark')
|
|
|
|
|
|
def which_dumpcap() -> str:
|
|
"""
|
|
:return: The path to `dumpcap` executable.
|
|
"""
|
|
return os.path.join(get_wireshark_dir(), 'dumpcap')
|
|
|
|
|
|
def which_mergecap() -> str:
|
|
"""
|
|
:return: The path to `mergecap` executable.
|
|
"""
|
|
return os.path.join(get_wireshark_dir(), 'mergecap')
|
|
|
|
|
|
def colon_hex(hexstr, interval) -> str:
|
|
""" Convert hexstr to colon seperated string every interval
|
|
|
|
:param hexstr: The hex string to convert.
|
|
:param interval: The interval number.
|
|
:return: The colon seperated string.
|
|
"""
|
|
assert len(hexstr) % interval == 0
|
|
return ':'.join(hexstr[i:i + interval] for i in range(0, len(hexstr), interval))
|
|
|
|
|
|
def is_sublist(lst1: list, lst2: list) -> bool:
|
|
""" Test whether lst1 is a slice of lst2
|
|
|
|
:param lst1: The list to judge if it is a sublist of lst2.
|
|
:param lst2: The list to judge if contains lst1.
|
|
:return: Whether lst1 is a slice of lst2.
|
|
"""
|
|
return lst1 in [lst2[i:len(lst1) + i] for i in range(len(lst1))]
|