Source code for pcapkit.vendor.reg.ethertype

# -*- coding: utf-8 -*-
"""Ethertype IEEE 802 Numbers
================================

This module contains the vendor crawler for **Ethertype IEEE 802 Numbers**,
which is automatically generating :class:`pcapkit.const.reg.ethertype.EtherType`.

"""

import collections
import csv
import re
import sys
from typing import TYPE_CHECKING

from pcapkit.vendor.default import Vendor

if TYPE_CHECKING:
    from collections import Counter
    from typing import Optional

__all__ = ['EtherType']


[docs]class EtherType(Vendor): """Ethertype IEEE 802 Numbers""" #: Value limit checker. FLAG = 'isinstance(value, int) and 0x0000 <= value <= 0xFFFF' #: Link to registry. LINK = 'https://www.iana.org/assignments/ieee-802-numbers/ieee-802-numbers-1.csv' def count(self, data: 'list[str]') -> 'Counter[str]': """Count field records. Args: data: CSV data. Returns: Field recordings. """ reader = csv.reader(data) next(reader) # header return collections.Counter(map(lambda item: self.safe_name(item[4]), filter(lambda item: len(item[1].split('-')) != 2, reader))) def rename(self, name: 'str', code: 'str', *, original: 'Optional[str]' = None) -> 'str': # pylint: disable=redefined-outer-name """Rename duplicated fields. Args: name: Field name. code: Field code. Keyword Args: original: Original field name (extracted from CSV records). Returns: Revised field name. """ if self.record[self.safe_name(name)] > 1: name = f'{name}_0x{code}' return self.safe_name(name) def process(self, data: 'list[str]') -> 'tuple[list[str], list[str]]': """Process CSV data. Args: data: CSV data. Returns: Enumeration fields and missing fields. """ reader = csv.reader(data) next(reader) # header enum = [] # type: list[str] miss = [] # type: list[str] for item in reader: name = item[4] rfcs = item[5] temp = [] # type: list[str] for rfc in filter(None, re.split(r'\[|\]', rfcs)): if 'RFC' in rfc and re.match(r'\d+', rfc[3:]): #temp.append(f'[{rfc[:3]} {rfc[3:]}]') temp.append(f'[:rfc:`{rfc[3:]}`]') else: temp.append(f'[{rfc}]'.replace('_', ' ')) tmp1 = re.sub(r'( )( )*', ' ', f"{''.join(temp)}".replace('\n', ' ')) if rfcs else '' tmp2 = re.sub(r'\r*\n', ' ', tmp1, re.MULTILINE) tmp3 = name.replace('\n', ' ') desc = self.wrap_comment(f"{tmp3} {tmp2}") try: code, _ = item[1], int(item[1], base=16) renm = re.sub(r'( )( )*', ' ', self.rename(name, code).replace('\n', ' ')) pres = f'{renm} = 0x{code}' sufs = f'#: {desc}' # if len(pres) > 74: # sufs = f"\n{' '*80}{sufs}" # enum.append(f'{pres.ljust(76)}{sufs}') enum.append(f'{sufs}\n {pres}') except ValueError: start, stop = item[1].split('-') miss.append(f'if 0x{start} <= value <= 0x{stop}:') miss.append(f' #: {desc}') miss.append(f" extend_enum(cls, '{self.safe_name(name)}_0x%s' % hex(value)[2:].upper().zfill(4), value)") # pylint: disable=line-too-long miss.append(' return cls(value)') return enum, miss
if __name__ == '__main__': sys.exit(EtherType())