1#! python
2#
3# Python Serial Port Extension for Win32, Linux, BSD, Jython
4# see __init__.py
5#
6# This module implements a simple socket based client.
7# It does not support changing any port parameters and will silently ignore any
8# requests to do so.
9#
10# The purpose of this module is that applications using pySerial can connect to
11# TCP/IP to serial port converters that do not support RFC 2217.
12#
13# (C) 2001-2011 Chris Liechti <cliechti@gmx.net>
14# this is distributed under a free software license, see license.txt
15#
16# URL format:    socket://<host>:<port>[/option[/option...]]
17# options:
18# - "debug" print diagnostic messages
19
20from serial.serialutil import *
21import time
22import socket
23import logging
24
25# map log level names to constants. used in fromURL()
26LOGGER_LEVELS = {
27    'debug': logging.DEBUG,
28    'info': logging.INFO,
29    'warning': logging.WARNING,
30    'error': logging.ERROR,
31    }
32
33POLL_TIMEOUT = 2
34
35class SocketSerial(SerialBase):
36    """Serial port implementation for plain sockets."""
37
38    BAUDRATES = (50, 75, 110, 134, 150, 200, 300, 600, 1200, 1800, 2400, 4800,
39                 9600, 19200, 38400, 57600, 115200)
40
41    def open(self):
42        """Open port with current settings. This may throw a SerialException
43           if the port cannot be opened."""
44        self.logger = None
45        if self._port is None:
46            raise SerialException("Port must be configured before it can be used.")
47        if self._isOpen:
48            raise SerialException("Port is already open.")
49        try:
50            # XXX in future replace with create_connection (py >=2.6)
51            self._socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
52            self._socket.connect(self.fromURL(self.portstr))
53        except Exception, msg:
54            self._socket = None
55            raise SerialException("Could not open port %s: %s" % (self.portstr, msg))
56
57        self._socket.settimeout(POLL_TIMEOUT) # used for write timeout support :/
58
59        # not that there anything to configure...
60        self._reconfigurePort()
61        # all things set up get, now a clean start
62        self._isOpen = True
63        if not self._rtscts:
64            self.setRTS(True)
65            self.setDTR(True)
66        self.flushInput()
67        self.flushOutput()
68
69    def _reconfigurePort(self):
70        """Set communication parameters on opened port. for the socket://
71        protocol all settings are ignored!"""
72        if self._socket is None:
73            raise SerialException("Can only operate on open ports")
74        if self.logger:
75            self.logger.info('ignored port configuration change')
76
77    def close(self):
78        """Close port"""
79        if self._isOpen:
80            if self._socket:
81                try:
82                    self._socket.shutdown(socket.SHUT_RDWR)
83                    self._socket.close()
84                except:
85                    # ignore errors.
86                    pass
87                self._socket = None
88            self._isOpen = False
89            # in case of quick reconnects, give the server some time
90            time.sleep(0.3)
91
92    def makeDeviceName(self, port):
93        raise SerialException("there is no sensible way to turn numbers into URLs")
94
95    def fromURL(self, url):
96        """extract host and port from an URL string"""
97        if url.lower().startswith("socket://"): url = url[9:]
98        try:
99            # is there a "path" (our options)?
100            if '/' in url:
101                # cut away options
102                url, options = url.split('/', 1)
103                # process options now, directly altering self
104                for option in options.split('/'):
105                    if '=' in option:
106                        option, value = option.split('=', 1)
107                    else:
108                        value = None
109                    if option == 'logging':
110                        logging.basicConfig()   # XXX is that good to call it here?
111                        self.logger = logging.getLogger('pySerial.socket')
112                        self.logger.setLevel(LOGGER_LEVELS[value])
113                        self.logger.debug('enabled logging')
114                    else:
115                        raise ValueError('unknown option: %r' % (option,))
116            # get host and port
117            host, port = url.split(':', 1) # may raise ValueError because of unpacking
118            port = int(port)               # and this if it's not a number
119            if not 0 <= port < 65536: raise ValueError("port not in range 0...65535")
120        except ValueError, e:
121            raise SerialException('expected a string in the form "[rfc2217://]<host>:<port>[/option[/option...]]": %s' % e)
122        return (host, port)
123
124    #  -  -  -  -  -  -  -  -  -  -  -  -  -  -  -  -  -  -  -  -  -  -  -  -
125
126    def inWaiting(self):
127        """Return the number of characters currently in the input buffer."""
128        if not self._isOpen: raise portNotOpenError
129        if self.logger:
130            # set this one to debug as the function could be called often...
131            self.logger.debug('WARNING: inWaiting returns dummy value')
132        return 0 # hmmm, see comment in read()
133
134    def read(self, size=1):
135        """Read size bytes from the serial port. If a timeout is set it may
136        return less characters as requested. With no timeout it will block
137        until the requested number of bytes is read."""
138        if not self._isOpen: raise portNotOpenError
139        data = bytearray()
140        if self._timeout is not None:
141            timeout = time.time() + self._timeout
142        else:
143            timeout = None
144        while len(data) < size and (timeout is None or time.time() < timeout):
145            try:
146                # an implementation with internal buffer would be better
147                # performing...
148                t = time.time()
149                block = self._socket.recv(size - len(data))
150                duration = time.time() - t
151                if block:
152                    data.extend(block)
153                else:
154                    # no data -> EOF (connection probably closed)
155                    break
156            except socket.timeout:
157                # just need to get out of recv from time to time to check if
158                # still alive
159                continue
160            except socket.error, e:
161                # connection fails -> terminate loop
162                raise SerialException('connection failed (%s)' % e)
163        return bytes(data)
164
165    def write(self, data):
166        """Output the given string over the serial port. Can block if the
167        connection is blocked. May raise SerialException if the connection is
168        closed."""
169        if not self._isOpen: raise portNotOpenError
170        try:
171            self._socket.sendall(to_bytes(data))
172        except socket.error, e:
173            # XXX what exception if socket connection fails
174            raise SerialException("socket connection failed: %s" % e)
175        return len(data)
176
177    def flushInput(self):
178        """Clear input buffer, discarding all that is in the buffer."""
179        if not self._isOpen: raise portNotOpenError
180        if self.logger:
181            self.logger.info('ignored flushInput')
182
183    def flushOutput(self):
184        """Clear output buffer, aborting the current output and
185        discarding all that is in the buffer."""
186        if not self._isOpen: raise portNotOpenError
187        if self.logger:
188            self.logger.info('ignored flushOutput')
189
190    def sendBreak(self, duration=0.25):
191        """Send break condition. Timed, returns to idle state after given
192        duration."""
193        if not self._isOpen: raise portNotOpenError
194        if self.logger:
195            self.logger.info('ignored sendBreak(%r)' % (duration,))
196
197    def setBreak(self, level=True):
198        """Set break: Controls TXD. When active, to transmitting is
199        possible."""
200        if not self._isOpen: raise portNotOpenError
201        if self.logger:
202            self.logger.info('ignored setBreak(%r)' % (level,))
203
204    def setRTS(self, level=True):
205        """Set terminal status line: Request To Send"""
206        if not self._isOpen: raise portNotOpenError
207        if self.logger:
208            self.logger.info('ignored setRTS(%r)' % (level,))
209
210    def setDTR(self, level=True):
211        """Set terminal status line: Data Terminal Ready"""
212        if not self._isOpen: raise portNotOpenError
213        if self.logger:
214            self.logger.info('ignored setDTR(%r)' % (level,))
215
216    def getCTS(self):
217        """Read terminal status line: Clear To Send"""
218        if not self._isOpen: raise portNotOpenError
219        if self.logger:
220            self.logger.info('returning dummy for getCTS()')
221        return True
222
223    def getDSR(self):
224        """Read terminal status line: Data Set Ready"""
225        if not self._isOpen: raise portNotOpenError
226        if self.logger:
227            self.logger.info('returning dummy for getDSR()')
228        return True
229
230    def getRI(self):
231        """Read terminal status line: Ring Indicator"""
232        if not self._isOpen: raise portNotOpenError
233        if self.logger:
234            self.logger.info('returning dummy for getRI()')
235        return False
236
237    def getCD(self):
238        """Read terminal status line: Carrier Detect"""
239        if not self._isOpen: raise portNotOpenError
240        if self.logger:
241            self.logger.info('returning dummy for getCD()')
242        return True
243
244    # - - - platform specific - - -
245    # None so far
246
247
248# assemble Serial class with the platform specific implementation and the base
249# for file-like behavior. for Python 2.6 and newer, that provide the new I/O
250# library, derive from io.RawIOBase
251try:
252    import io
253except ImportError:
254    # classic version with our own file-like emulation
255    class Serial(SocketSerial, FileLike):
256        pass
257else:
258    # io library present
259    class Serial(SocketSerial, io.RawIOBase):
260        pass
261
262
263# simple client test
264if __name__ == '__main__':
265    import sys
266    s = Serial('socket://localhost:7000')
267    sys.stdout.write('%s\n' % s)
268
269    sys.stdout.write("write...\n")
270    s.write("hello\n")
271    s.flush()
272    sys.stdout.write("read: %s\n" % s.read(5))
273
274    s.close()
275