Skip to main content
aboutsummaryrefslogtreecommitdiffstats
blob: 7a54ce0f1e84c4b82cc7594a09e694fd9078fd0c (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
# *******************************************************************************
# * Copyright (c) 2011 Wind River Systems, Inc. and others.
# * All rights reserved. This program and the accompanying materials
# * are made available under the terms of the Eclipse Public License v1.0
# * which accompanies this distribution, and is available at
# * http://www.eclipse.org/legal/epl-v10.html
# *
# * Contributors:
# *     Wind River Systems - initial API and implementation
# *******************************************************************************

import threading, exceptions
import protocol, channel
from tcf.services import locator

_channels = []
_listeners = []
_transports = {}
_lock = threading.RLock()

class TransportProvider(object):
    """
    TransportProvider represents communication protocol that can be used to open TCF communication channels.
    Examples of transports are: TCP/IP, RS-232, USB.
    
    Client can implement this interface if they want to provide support for a transport that is not
    supported directly by the framework.
    """

    def getName(self):
        """
        Return transport name. Same as used as peer attribute, @see IPeer.ATTR_TRANSPORT_NAME
        @return transport name.
        """
        raise exceptions.NotImplementedError("Abstract method")
    
    def openChannel(self, peer):
        """
        Open channel to communicate with this peer using this transport.
        Note: the channel can be not fully open yet when this method returns.
        It's state can be IChannel.STATE_OPENING.
        Protocol.Listener will be called when the channel will be opened or closed.
        @param peer - a IPeer object that describes remote end-point of the channel.
        @return TCF communication channel.
        """
        raise exceptions.NotImplementedError("Abstract method")

def addTransportProvider(transport):
    name = transport.getName()
    assert name
    with _lock:
        if _transports.get(name): raise exceptions.Exception("Already registered: " + name)
        _transports[name] = transport

def removeTransportProvider(transport):
    name = transport.getName()
    assert name
    with _lock:
        if _transports.get(name) == transport: del _transports[name]

def openChannel(peer):
    name = peer.getTransportName()
    if not name: raise Exception("No transport name")
    with _lock:
        provider = _transports.get(name)
        if not provider: raise Exception("Unknown transport name: " + name)
    return provider.openChannel(peer)

def channelOpened(channel):
    assert channel not in _channels
    _channels.append(channel)
    for l in _listeners:
        try:
            l.onChannelOpen(channel)
        except exceptions.Exception as x:
            protocol.log("Exception in channel listener", x)

def channelClosed(channel, error):
    assert channel in _channels
    _channels.remove(channel)

def getOpenChannels():
    return _channels[:]

def addChannelOpenListener(listener):
    assert listener
    _listeners.append(listener)

def removeChannelOpenListener(listener):
    _listeners.remove(listener)

class TCPTransportProvider(TransportProvider):
    def getName(self):
        return "TCP"
    def openChannel(self, p):
        assert self.getName() == p.getTransportName()
        from tcf import peer
        attrs = p.getAttributes()
        host = attrs.get(peer.ATTR_IP_HOST)
        port = attrs.get(peer.ATTR_IP_PORT)
        if not host: raise exceptions.RuntimeError("No host name")
        from channel.ChannelTCP import ChannelTCP
        return ChannelTCP(p, host, _parsePort(port))

def _parsePort(port):
    if not port: raise exceptions.Exception("No port number")
    try:
        return int(port)
    except exceptions.Exception:
        raise exceptions.RuntimeError(
                "Invalid value of \"Port\" attribute. Must be decimal number.")

def sendEvent(service_name, event_name, data):
    """
    Transmit TCF event message.
    The message is sent to all open communication channels - broadcasted.
    
    This is internal API, TCF clients should use protocol.sendEvent().
    """
    for c in _channels:
        # Skip channels that are executing "redirect" command - STATE_OPENING
        if c.getState() == channel.STATE_OPEN:
            s = c.getLocalService(service_name)
            if s: c.sendEvent(s, event_name, data)

def sync(done):
    """
    Call back after TCF messages sent by this host up to this moment are delivered
    to their intended targets. This method is intended for synchronization of messages
    across multiple channels.
    
    Note: Cross channel synchronization can reduce performance and throughput.
    Most clients don't need cross channel synchronization and should not call this method.
    
    @param done will be executed by dispatch thread after communication
    messages are delivered to corresponding targets.
    
    This is internal API, TCF clients should use protocol.sync().
    """
    set = set()
    class DoneSync(locator.DoneSync):
        def doneSync(self, token):
            assert set.contains(token)
            set.remove(token)
            if len(set) == 0: done()
    done_sync = DoneSync()
    for c in _channels:
        if c.getState() == channel.STATE_OPEN:
            s = c.getRemoteService(locator.NAME)
            if s: set.append(s.sync(done_sync))
    if len(set) == 0: protocol.invokeLater(done)

# initialize TCP transport
addTransportProvider(TCPTransportProvider())

Back to the top