Source code for udsoncan.services.RequestDownload

import struct
from udsoncan import DataFormatIdentifier, MemoryLocation
from udsoncan.Request import Request
from udsoncan.Response import Response
from udsoncan.exceptions import *
from udsoncan.BaseService import BaseService, BaseResponseData
from udsoncan.ResponseCode import ResponseCode

from typing import Optional, cast


class RequestDownload(BaseService):
    _sid = 0x34
    _use_subfunction = False

    supported_negative_response = [ResponseCode.IncorrectMessageLengthOrInvalidFormat,
                                   ResponseCode.ConditionsNotCorrect,
                                   ResponseCode.RequestOutOfRange,
                                   ResponseCode.SecurityAccessDenied,
                                   ResponseCode.UploadDownloadNotAccepted
                                   ]

[docs] class ResponseData(BaseResponseData): """ .. data:: max_length (int) Maximum number of data blocks to write """ max_length: int def __init__(self, max_length): super().__init__(RequestDownload) self.max_length = max_length
class InterpretedResponse(Response): service_data: "RequestDownload.ResponseData" @classmethod def normalize_data_format_identifier(cls, dfi: Optional[DataFormatIdentifier]) -> DataFormatIdentifier: if dfi is None: dfi = DataFormatIdentifier() if not isinstance(dfi, DataFormatIdentifier): raise ValueError('dfi must be an instance of DataFormatIdentifier') return dfi
[docs] @classmethod def make_request(cls, memory_location: MemoryLocation, dfi: Optional[DataFormatIdentifier] = None) -> Request: """ Generates a request for RequestDownload :param memory_location: The address and the size of the memory block to be written. :type memory_location: :ref:`MemoryLocation <MemoryLocation>` :param dfi: Optional :ref:`DataFormatIdentifier <DataFormatIdentifier>` defining the compression and encryption scheme of the data. If not specified, the default value of 00 will be used, specifying no encryption and no compression :type dfi: :ref:`DataFormatIdentifier <DataFormatIdentifier>` :raises ValueError: If parameters are out of range, missing or wrong type """ dfi = cls.normalize_data_format_identifier(dfi) if not isinstance(memory_location, MemoryLocation): raise ValueError('memory_location must be an instance of MemoryLocation') request = Request(service=cls) request.data = bytes() request.data += dfi.get_byte() # Data Format Identifier request.data += memory_location.alfid.get_byte() # AddressAndLengthFormatIdentifier request.data += memory_location.get_address_bytes() request.data += memory_location.get_memorysize_bytes() return request
[docs] @classmethod def interpret_response(cls, response: Response) -> InterpretedResponse: """ Populates the response ``service_data`` property with an instance of :class:`RequestDownload.ResponseData<udsoncan.services.RequestDownload.ResponseData>` :param response: The received response to interpret :type response: :ref:`Response<Response>` :raises InvalidResponseException: If length of ``response.data`` is too short :raises NotImplementedError: If the ``maxNumberOfBlockLength`` value is encoded over more than 8 bytes. """ if response.data is None: raise InvalidResponseException(response, "No data in response") if len(response.data) < 1: raise InvalidResponseException(response, "Response data must be at least 1 bytes") lfid = int(response.data[0]) >> 4 if lfid > 8: raise NotImplementedError('This client does not support number bigger than %d bits' % (8 * 8)) if len(response.data) < lfid + 1: raise InvalidResponseException( response, "Length of data (%d) is too short to contains the number of block of given length (%d)" % (len(response.data), lfid)) todecode = bytearray(b'\x00\x00\x00\x00\x00\x00\x00\x00') for i in range(1, lfid + 1): todecode[-i] = response.data[lfid + 1 - i] response.service_data = cls.ResponseData( max_length=struct.unpack('>q', todecode)[0] ) return cast(RequestDownload.InterpretedResponse, response)