blob: c14a87e4e97f2f75e0638654d0a3ede4f471b9c1 [file] [log] [blame]
Chris Liechtiad11d172015-10-18 01:02:17 +02001#! python
2#
Chris Liechti3e02f702015-12-16 23:06:04 +01003# This module implements a special URL handler that allows selecting an
Chris Liechtiad11d172015-10-18 01:02:17 +02004# alternate implementation provided by some backends.
5#
Chris Liechti3e02f702015-12-16 23:06:04 +01006# This file is part of pySerial. https://github.com/pyserial/pyserial
Chris Liechtiad11d172015-10-18 01:02:17 +02007# (C) 2015 Chris Liechti <cliechti@gmx.net>
8#
9# SPDX-License-Identifier: BSD-3-Clause
10#
11# URL format: alt://port[?option[=value][&option[=value]]]
12# options:
13# - class=X used class named X instead of Serial
14#
15# example:
16# use poll based implementation on Posix (Linux):
17# python -m serial.tools.miniterm alt:///dev/ttyUSB0?class=PosixPollSerial
18
Chris Liechtiad11d172015-10-18 01:02:17 +020019try:
20 import urlparse
21except ImportError:
22 import urllib.parse as urlparse
23
Chris Liechti9eaa40c2016-02-12 23:32:59 +010024import serial
25
Chris Liechtiad11d172015-10-18 01:02:17 +020026
27def serial_class_for_url(url):
28 """extract host and port from an URL string"""
29 parts = urlparse.urlsplit(url)
30 if parts.scheme != 'alt':
Chris Liechti9eaa40c2016-02-12 23:32:59 +010031 raise serial.SerialException(
32 'expected a string in the form "alt://port[?option[=value][&option[=value]]]": '
Chris Liechtic8f3f822016-06-08 03:35:28 +020033 'not starting with alt:// ({!r})'.format(parts.scheme))
Chris Liechtiad11d172015-10-18 01:02:17 +020034 class_name = 'Serial'
35 try:
36 for option, values in urlparse.parse_qs(parts.query, True).items():
37 if option == 'class':
38 class_name = values[0]
39 else:
Chris Liechtic8f3f822016-06-08 03:35:28 +020040 raise ValueError('unknown option: {!r}'.format(option))
Chris Liechtiad11d172015-10-18 01:02:17 +020041 except ValueError as e:
Chris Liechti9eaa40c2016-02-12 23:32:59 +010042 raise serial.SerialException(
43 'expected a string in the form '
Chris Liechtic8f3f822016-06-08 03:35:28 +020044 '"alt://port[?option[=value][&option[=value]]]": {!r}'.format(e))
Chris Liechti7f4b4ee2016-01-16 23:21:42 +010045 if not hasattr(serial, class_name):
Chris Liechtic8f3f822016-06-08 03:35:28 +020046 raise ValueError('unknown class: {!r}'.format(class_name))
Chris Liechti7f4b4ee2016-01-16 23:21:42 +010047 cls = getattr(serial, class_name)
48 if not issubclass(cls, serial.Serial):
Chris Liechtic8f3f822016-06-08 03:35:28 +020049 raise ValueError('class {!r} is not an instance of Serial'.format(class_name))
Chris Liechti7f4b4ee2016-01-16 23:21:42 +010050 return (''.join([parts.netloc, parts.path]), cls)
Chris Liechtiad11d172015-10-18 01:02:17 +020051
52# - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
53if __name__ == '__main__':
Chris Liechtib10daf42016-01-26 00:27:10 +010054 s = serial.serial_for_url('alt:///dev/ttyS0?class=PosixPollSerial')
Chris Liechtiad11d172015-10-18 01:02:17 +020055 print(s)