connectors/ERPC/__init__.py
author Edouard Tisserant <edouard.tisserant@gmail.com>
Sun, 08 Dec 2024 11:41:31 +0100
changeset 4063 6668dcbca915
parent 4043 b217fb367574
permissions -rw-r--r--
IDE: Fix wrong red wire coloring in FBD, this time for good.
#!/usr/bin/env python
# -*- coding: utf-8 -*-

# Written by Edouard TISSERANT (C) 2024
# This file is part of Beremiz IDE
# See COPYING file for copyrights details.


import os.path
import re
import traceback
from inspect import getmembers, isfunction


import erpc

# eRPC service code
from erpc_interface.erpc_PLCObject.interface import IBeremizPLCObjectService
from erpc_interface.erpc_PLCObject.client import BeremizPLCObjectServiceClient
from erpc_interface.erpc_PLCObject.common import trace_order, extra_file, PLCstatus_enum

import PSKManagement as PSK
from connectors.ERPC.PSK_Adapter import SSLPSKClientTransport
from connectors.ConnectorBase import ConnectorBase
from connectors.ERPC_URI import per_scheme_model

enum_to_PLCstatus = dict(map(lambda t:(t[1],t[0]),getmembers(PLCstatus_enum, lambda x:type(x)==int)))

class MissingCallException(Exception):
    pass

def ExceptionFromERPCReturn(ret):
    return {1:Exception,
            2:MissingCallException}.get(ret,ValueError)

def ReturnAsLastOutput(client_method, obj, args_wrapper, *args):
    retval = erpc.Reference()
    ret = client_method(obj, *args_wrapper(*args), retval)
    if ret != 0:
        raise ExceptionFromERPCReturn(ret)(client_method.__name__)
    return retval.value

def TranslatedReturnAsLastOutput(translator):
    def wrapper(client_method, obj, args_wrapper, *args):
        res = ReturnAsLastOutput(client_method, obj, args_wrapper, *args)
        return translator(res)
    return wrapper

ReturnWrappers = {
    "AppendChunkToBlob":ReturnAsLastOutput,
    "GetLogMessage":TranslatedReturnAsLastOutput(
        lambda res:(res.msg, res.tick, res.sec, res.nsec)),
    "GetPLCID":TranslatedReturnAsLastOutput(
        lambda res:(res.ID, res.PSK)),
    "GetPLCstatus":TranslatedReturnAsLastOutput(
        lambda res:(enum_to_PLCstatus[res.PLCstatus], res.logcounts)),
    "GetTraceVariables":TranslatedReturnAsLastOutput(
        lambda res:(enum_to_PLCstatus[res.PLCstatus],
                    [(sample.tick, bytes(sample.TraceBuffer)) for sample in res.traces])),
    "MatchMD5":ReturnAsLastOutput,
    "NewPLC":ReturnAsLastOutput,
    "SeedBlob":ReturnAsLastOutput,
    "SetTraceVariablesList": ReturnAsLastOutput,
    "StopPLC":ReturnAsLastOutput,
    "ExtendedCall":ReturnAsLastOutput,
}

ArgsWrappers = {
    "NewPLC":
        lambda md5sum, plcObjectBlobID, extrafiles: (
            md5sum, plcObjectBlobID, [extra_file(*f) for f in extrafiles]),
    "SetTraceVariablesList":
        lambda orders : ([
            trace_order(idx, b"" if force is None else force) 
            for idx, force in orders],)
}

def rpc_wrapper(method_name, confnodesroot):
    client_method = getattr(BeremizPLCObjectServiceClient, method_name)
    return_wrapper = ReturnWrappers.get(
        method_name, 
        lambda client_method, obj, args_wrapper, *args: client_method(obj, *args_wrapper(*args)))
    args_wrapper = ArgsWrappers.get(method_name, lambda *x:x)

    def exception_wrapper(self, *args):
        try:
            return return_wrapper(client_method, self, args_wrapper, *args)
        except erpc.transport.ConnectionClosed as e:
            confnodesroot._SetConnector(None)
            confnodesroot.logger.write_error(_("Connection lost!\n"))
        except erpc.codec.CodecError as e:
            confnodesroot.logger.write_warning(_("ERPC codec error: %s\n") % e)
        except erpc.client.RequestError as e:
            confnodesroot.logger.write_error(_("ERPC request error: %s\n") % e)                
        except MissingCallException as e:
            confnodesroot.logger.write_warning(_("Remote call not supported: %s\n") % e)
        except Exception as e:
            errmess = _("Exception calling remote PLC object fucntion %s:\n") % method_name \
                        + traceback.format_exc()
            confnodesroot.logger.write_error(errmess + "\n")
            confnodesroot._SetConnector(None)

        return self.PLCObjDefaults.get(method_name)
    return exception_wrapper



def ERPC_connector_factory(uri, confnodesroot):
    """
    returns the ERPC connector
    """
    confnodesroot.logger.write(_("ERPC connecting to URI : %s\n") % uri)

    # TODO add parsing for serial URI
    # ERPC:///dev/ttyXX:baudrate or ERPC://:COM4:baudrate

    try:
        scheme, location = uri.split("://",1)
        _model, _useID, parser, _builder = per_scheme_model[scheme]
        location_data = parser(location)
    except Exception as e:
        confnodesroot.logger.write_error(
            'Malformed URI "%s": %s\n' % (uri, str(e)))
        return None

    PLCObjectERPCProxy = type(
        "PLCObjectERPCProxy",
        (ConnectorBase, BeremizPLCObjectServiceClient),
        {name: rpc_wrapper(name, confnodesroot)
            for name,_func in getmembers(IBeremizPLCObjectService, isfunction)})

    if scheme in ["ERPCS", "ERPC"]:
        if scheme=="ERPCS":
            ID = location_data["ID"]
            if not ID:
                confnodesroot.logger.write_error(
                    f'Invalid URI "{uri}": ERPCS requires PLC ID after "#"\n')
                return None
            default_port = 4000
        else:
            ID = None
            if "#" in location:
                confnodesroot.logger.write_error(
                    f'URI "{uri}": Non-encrypted ERPC does not take a PLC ID after "#"\n')
                return None
            default_port = 3000

        host = location_data["host"]
        port = location_data["port"]
        port = int(port) if port else default_port

        try:
            if ID:
                # load PSK from project
                secpath = os.path.join(confnodesroot.ProjectPath, 'psk', ID + '.secret')
                if not os.path.exists(secpath):
                    confnodesroot.logger.write_error(
                        'Error: Pre-Shared-Key Secret in %s is missing!\n' % secpath)
                    return None
                secret = open(secpath).read().partition(':')[2].rstrip('\n\r').encode()
                transport = SSLPSKClientTransport(host, port, (secret, ID.encode())) # type: ignore
            else:

                transport = erpc.transport.TCPTransport(host, port, False)

            clientManager = erpc.client.ClientManager(transport, erpc.basic_codec.BasicCodec)
            client = PLCObjectERPCProxy(clientManager)

        except Exception as e:
            confnodesroot.logger.write_error(
                _("Connection to {loc} failed with exception {ex}\n").format(
                    loc=uri, ex=str(e)))
            return None

    else:
        # TODO if serial URI then 
        # transport = erpc.transport.SerialTransport(device, baudrate)

        confnodesroot.logger.write_error(
            _("Unknown scheme {scheme} in URI {uri}\n").format(
                scheme=scheme, uri=uri))
        return None
    # Check connection is effective.
    IDPSK = client.GetPLCID()
    if IDPSK:
        ID, secret = IDPSK
        PSK.UpdateID(confnodesroot.ProjectPath, ID, secret, uri)
    else:
        confnodesroot.logger.write_warning(_("PLC did not provide identity and security infomation.\n"))

    return client