123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373 |
- # Based from "GDBus - GLib D-Bus Library":
- #
- # Copyright (C) 2008-2011 Red Hat, Inc.
- #
- # This library is free software; you can redistribute it and/or
- # modify it under the terms of the GNU Lesser General Public
- # License as published by the Free Software Foundation; either
- # version 2.1 of the License, or (at your option) any later version.
- #
- # This library is distributed in the hope that it will be useful,
- # but WITHOUT ANY WARRANTY; without even the implied warranty of
- # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
- # Lesser General Public License for more details.
- #
- # You should have received a copy of the GNU Lesser General
- # Public License along with this library; if not, see <http://www.gnu.org/licenses/>.
- #
- # Author: David Zeuthen <davidz@redhat.com>
- import xml.parsers.expat
- class Annotation:
- def __init__(self, key, value):
- self.key = key
- self.value = value
- self.annotations = []
- self.since = ""
- class Arg:
- def __init__(self, name, signature):
- self.name = name
- self.signature = signature
- self.annotations = []
- self.doc_string = ""
- self.since = ""
- class Method:
- def __init__(self, name, h_type_implies_unix_fd=True):
- self.name = name
- self.h_type_implies_unix_fd = h_type_implies_unix_fd
- self.in_args = []
- self.out_args = []
- self.annotations = []
- self.doc_string = ""
- self.since = ""
- self.deprecated = False
- self.unix_fd = False
- class Signal:
- def __init__(self, name):
- self.name = name
- self.args = []
- self.annotations = []
- self.doc_string = ""
- self.since = ""
- self.deprecated = False
- class Property:
- def __init__(self, name, signature, access):
- self.name = name
- self.signature = signature
- self.access = access
- self.annotations = []
- self.arg = Arg("value", self.signature)
- self.arg.annotations = self.annotations
- self.readable = False
- self.writable = False
- if self.access == "readwrite":
- self.readable = True
- self.writable = True
- elif self.access == "read":
- self.readable = True
- elif self.access == "write":
- self.writable = True
- else:
- raise ValueError('Invalid access type "{}"'.format(self.access))
- self.doc_string = ""
- self.since = ""
- self.deprecated = False
- self.emits_changed_signal = True
- class Interface:
- def __init__(self, name):
- self.name = name
- self.methods = []
- self.signals = []
- self.properties = []
- self.annotations = []
- self.doc_string = ""
- self.doc_string_brief = ""
- self.since = ""
- self.deprecated = False
- class DBusXMLParser:
- STATE_TOP = "top"
- STATE_NODE = "node"
- STATE_INTERFACE = "interface"
- STATE_METHOD = "method"
- STATE_SIGNAL = "signal"
- STATE_PROPERTY = "property"
- STATE_ARG = "arg"
- STATE_ANNOTATION = "annotation"
- STATE_IGNORED = "ignored"
- def __init__(self, xml_data, h_type_implies_unix_fd=True):
- self._parser = xml.parsers.expat.ParserCreate()
- self._parser.CommentHandler = self.handle_comment
- self._parser.CharacterDataHandler = self.handle_char_data
- self._parser.StartElementHandler = self.handle_start_element
- self._parser.EndElementHandler = self.handle_end_element
- self.parsed_interfaces = []
- self._cur_object = None
- self.state = DBusXMLParser.STATE_TOP
- self.state_stack = []
- self._cur_object = None
- self._cur_object_stack = []
- self.doc_comment_last_symbol = ""
- self._h_type_implies_unix_fd = h_type_implies_unix_fd
- self._parser.Parse(xml_data)
- COMMENT_STATE_BEGIN = "begin"
- COMMENT_STATE_PARAMS = "params"
- COMMENT_STATE_BODY = "body"
- COMMENT_STATE_SKIP = "skip"
- def handle_comment(self, data):
- comment_state = DBusXMLParser.COMMENT_STATE_BEGIN
- lines = data.split("\n")
- symbol = ""
- body = ""
- in_para = False
- params = {}
- for line in lines:
- orig_line = line
- line = line.lstrip()
- if comment_state == DBusXMLParser.COMMENT_STATE_BEGIN:
- if len(line) > 0:
- colon_index = line.find(": ")
- if colon_index == -1:
- if line.endswith(":"):
- symbol = line[0 : len(line) - 1]
- comment_state = DBusXMLParser.COMMENT_STATE_PARAMS
- else:
- comment_state = DBusXMLParser.COMMENT_STATE_SKIP
- else:
- symbol = line[0:colon_index]
- rest_of_line = line[colon_index + 2 :].strip()
- if len(rest_of_line) > 0:
- body += rest_of_line + "\n"
- comment_state = DBusXMLParser.COMMENT_STATE_PARAMS
- elif comment_state == DBusXMLParser.COMMENT_STATE_PARAMS:
- if line.startswith("@"):
- colon_index = line.find(": ")
- if colon_index == -1:
- comment_state = DBusXMLParser.COMMENT_STATE_BODY
- if not in_para:
- in_para = True
- body += orig_line + "\n"
- else:
- param = line[1:colon_index]
- docs = line[colon_index + 2 :]
- params[param] = docs
- else:
- comment_state = DBusXMLParser.COMMENT_STATE_BODY
- if len(line) > 0:
- if not in_para:
- in_para = True
- body += orig_line + "\n"
- elif comment_state == DBusXMLParser.COMMENT_STATE_BODY:
- if len(line) > 0:
- if not in_para:
- in_para = True
- body += orig_line + "\n"
- else:
- if in_para:
- body += "\n"
- in_para = False
- if in_para:
- body += "\n"
- if symbol != "":
- self.doc_comment_last_symbol = symbol
- self.doc_comment_params = params
- self.doc_comment_body = body
- def handle_char_data(self, data):
- # print 'char_data=%s'%data
- pass
- def handle_start_element(self, name, attrs):
- old_state = self.state
- old_cur_object = self._cur_object
- if self.state == DBusXMLParser.STATE_IGNORED:
- self.state = DBusXMLParser.STATE_IGNORED
- elif self.state == DBusXMLParser.STATE_TOP:
- if name == DBusXMLParser.STATE_NODE:
- self.state = DBusXMLParser.STATE_NODE
- else:
- self.state = DBusXMLParser.STATE_IGNORED
- elif self.state == DBusXMLParser.STATE_NODE:
- if name == DBusXMLParser.STATE_INTERFACE:
- self.state = DBusXMLParser.STATE_INTERFACE
- iface = Interface(attrs["name"])
- self._cur_object = iface
- self.parsed_interfaces.append(iface)
- elif name == DBusXMLParser.STATE_ANNOTATION:
- self.state = DBusXMLParser.STATE_ANNOTATION
- anno = Annotation(attrs["name"], attrs["value"])
- self._cur_object.annotations.append(anno)
- self._cur_object = anno
- else:
- self.state = DBusXMLParser.STATE_IGNORED
- # assign docs, if any
- if "name" in attrs and self.doc_comment_last_symbol == attrs["name"]:
- self._cur_object.doc_string = self.doc_comment_body
- if "short_description" in self.doc_comment_params:
- short_description = self.doc_comment_params["short_description"]
- self._cur_object.doc_string_brief = short_description
- if "since" in self.doc_comment_params:
- self._cur_object.since = self.doc_comment_params["since"].strip()
- elif self.state == DBusXMLParser.STATE_INTERFACE:
- if name == DBusXMLParser.STATE_METHOD:
- self.state = DBusXMLParser.STATE_METHOD
- method = Method(
- attrs["name"], h_type_implies_unix_fd=self._h_type_implies_unix_fd
- )
- self._cur_object.methods.append(method)
- self._cur_object = method
- elif name == DBusXMLParser.STATE_SIGNAL:
- self.state = DBusXMLParser.STATE_SIGNAL
- signal = Signal(attrs["name"])
- self._cur_object.signals.append(signal)
- self._cur_object = signal
- elif name == DBusXMLParser.STATE_PROPERTY:
- self.state = DBusXMLParser.STATE_PROPERTY
- prop = Property(attrs["name"], attrs["type"], attrs["access"])
- self._cur_object.properties.append(prop)
- self._cur_object = prop
- elif name == DBusXMLParser.STATE_ANNOTATION:
- self.state = DBusXMLParser.STATE_ANNOTATION
- anno = Annotation(attrs["name"], attrs["value"])
- self._cur_object.annotations.append(anno)
- self._cur_object = anno
- else:
- self.state = DBusXMLParser.STATE_IGNORED
- # assign docs, if any
- if "name" in attrs and self.doc_comment_last_symbol == attrs["name"]:
- self._cur_object.doc_string = self.doc_comment_body
- if "since" in self.doc_comment_params:
- self._cur_object.since = self.doc_comment_params["since"].strip()
- elif self.state == DBusXMLParser.STATE_METHOD:
- if name == DBusXMLParser.STATE_ARG:
- self.state = DBusXMLParser.STATE_ARG
- arg_name = None
- if "name" in attrs:
- arg_name = attrs["name"]
- arg = Arg(arg_name, attrs["type"])
- direction = attrs.get("direction", "in")
- if direction == "in":
- self._cur_object.in_args.append(arg)
- elif direction == "out":
- self._cur_object.out_args.append(arg)
- else:
- raise ValueError('Invalid direction "{}"'.format(direction))
- self._cur_object = arg
- elif name == DBusXMLParser.STATE_ANNOTATION:
- self.state = DBusXMLParser.STATE_ANNOTATION
- anno = Annotation(attrs["name"], attrs["value"])
- self._cur_object.annotations.append(anno)
- self._cur_object = anno
- else:
- self.state = DBusXMLParser.STATE_IGNORED
- # assign docs, if any
- if self.doc_comment_last_symbol == old_cur_object.name:
- if "name" in attrs and attrs["name"] in self.doc_comment_params:
- doc_string = self.doc_comment_params[attrs["name"]]
- if doc_string is not None:
- self._cur_object.doc_string = doc_string
- if "since" in self.doc_comment_params:
- self._cur_object.since = self.doc_comment_params[
- "since"
- ].strip()
- elif self.state == DBusXMLParser.STATE_SIGNAL:
- if name == DBusXMLParser.STATE_ARG:
- self.state = DBusXMLParser.STATE_ARG
- arg_name = None
- if "name" in attrs:
- arg_name = attrs["name"]
- arg = Arg(arg_name, attrs["type"])
- self._cur_object.args.append(arg)
- self._cur_object = arg
- elif name == DBusXMLParser.STATE_ANNOTATION:
- self.state = DBusXMLParser.STATE_ANNOTATION
- anno = Annotation(attrs["name"], attrs["value"])
- self._cur_object.annotations.append(anno)
- self._cur_object = anno
- else:
- self.state = DBusXMLParser.STATE_IGNORED
- # assign docs, if any
- if self.doc_comment_last_symbol == old_cur_object.name:
- if "name" in attrs and attrs["name"] in self.doc_comment_params:
- doc_string = self.doc_comment_params[attrs["name"]]
- if doc_string is not None:
- self._cur_object.doc_string = doc_string
- if "since" in self.doc_comment_params:
- self._cur_object.since = self.doc_comment_params[
- "since"
- ].strip()
- elif self.state == DBusXMLParser.STATE_PROPERTY:
- if name == DBusXMLParser.STATE_ANNOTATION:
- self.state = DBusXMLParser.STATE_ANNOTATION
- anno = Annotation(attrs["name"], attrs["value"])
- self._cur_object.annotations.append(anno)
- self._cur_object = anno
- else:
- self.state = DBusXMLParser.STATE_IGNORED
- elif self.state == DBusXMLParser.STATE_ARG:
- if name == DBusXMLParser.STATE_ANNOTATION:
- self.state = DBusXMLParser.STATE_ANNOTATION
- anno = Annotation(attrs["name"], attrs["value"])
- self._cur_object.annotations.append(anno)
- self._cur_object = anno
- else:
- self.state = DBusXMLParser.STATE_IGNORED
- elif self.state == DBusXMLParser.STATE_ANNOTATION:
- if name == DBusXMLParser.STATE_ANNOTATION:
- self.state = DBusXMLParser.STATE_ANNOTATION
- anno = Annotation(attrs["name"], attrs["value"])
- self._cur_object.annotations.append(anno)
- self._cur_object = anno
- else:
- self.state = DBusXMLParser.STATE_IGNORED
- else:
- raise ValueError(
- 'Unhandled state "{}" while entering element with name "{}"'.format(
- self.state, name
- )
- )
- self.state_stack.append(old_state)
- self._cur_object_stack.append(old_cur_object)
- def handle_end_element(self, name):
- self.state = self.state_stack.pop()
- self._cur_object = self._cur_object_stack.pop()
- def parse_dbus_xml(xml_data):
- parser = DBusXMLParser(xml_data, True)
- return parser.parsed_interfaces
|