Chromium Code Reviews| Index: tools/idl_parser/idl_node.py |
| diff --git a/tools/idl_parser/idl_node.py b/tools/idl_parser/idl_node.py |
| new file mode 100755 |
| index 0000000000000000000000000000000000000000..ae3849da114011e99857786d42975d7da41ff786 |
| --- /dev/null |
| +++ b/tools/idl_parser/idl_node.py |
| @@ -0,0 +1,208 @@ |
| +#!/usr/bin/env python |
| +# Copyright (c) 2012 The Chromium Authors. All rights reserved. |
| +# Use of this source code is governed by a BSD-style license that can be |
| +# found in the LICENSE file. |
| + |
| +import sys |
| + |
| +"""Nodes for PPAPI IDL AST""" |
| + |
| +# |
| +# IDL Node |
| +# |
| +# IDL Node defines the IDLAttribute and IDLNode objects which are constructed |
| +# by the parser as it processes the various 'productions'. The IDLAttribute |
| +# objects are assigned to the IDLNode's property dictionary instead of being |
| +# applied as children of The IDLNodes, so they do not exist in the final tree. |
| +# The AST of IDLNodes is the output from the parsing state and will be used |
| +# as the source data by the various generators. |
| +# |
| + |
| + |
| +# |
| +# CopyToList |
| +# |
| +# Takes an input item, list, or None, and returns a new list of that set. |
| +def CopyToList(item): |
| + # If the item is 'Empty' make it an empty list |
| + if not item: |
| + item = [] |
| + |
| + # If the item is not a list |
|
sehr
2013/04/08 18:37:07
add something like: "return a list containing one
|
| + if type(item) is not type([]): |
| + item = [item] |
| + |
| + # Make a copy we can modify |
|
sehr
2013/04/08 18:37:07
Please make sure that comments end with a period c
|
| + return list(item) |
| + |
| + |
| +# IDLSearch |
| +# |
| +# A temporary object used by the parsing process to hold an Extended Attribute |
| +# which will be passed as a child to a standard IDLNode. |
| +# |
| +class IDLSearch(object): |
| + def __init__(self): |
| + self.depth = 0 |
| + |
| + def Enter(self, node): |
| + pass |
| + |
| + def Exit(self, node): |
| + pass |
| + |
| + |
| +# IDLAttribute |
| +# |
| +# A temporary object used by the parsing process to hold an Extended Attribute |
| +# which will be passed as a child to a standard IDLNode. |
| +# |
| +class IDLAttribute(object): |
| + def __init__(self, name, value): |
| + self._cls = 'Property' |
| + self._name = name |
| + self._value = value |
| + |
| + def __str__(self): |
| + return '%s=%s' % (self._name, self._value) |
| + |
| +# |
| +# IDLNode |
| +# |
| +# This class implements the AST tree, providing the associations between |
| +# parents and children. It also contains a namepsace and propertynode to |
| +# allow for look-ups. IDLNode is derived from IDLRelease, so it is |
| +# version aware. |
| +# |
| +class IDLNode(object): |
| + def __init__(self, cls, filename, lineno, pos, children=None): |
| + self._cls = cls |
| + self._properties = { |
| + 'ERRORS' : [], |
| + 'WARNINGS': [], |
| + 'FILENAME': filename, |
| + 'LINENO' : lineno, |
| + 'POSSITION' : pos, |
| + } |
| + |
| + self._children = [] |
| + self._parent = None |
| + self.AddChildren(children) |
| + |
| +# |
| +# |
| +# |
| + # Return a string representation of this node |
| + def __str__(self): |
| + name = self.GetProperty('NAME','') |
| + return '%s(%s)' % (self._cls, name) |
| + |
| + |
| + # Log an error for this object |
| + def Error(self, msg): |
| + filename, lineno = self.GetFileAndLine() |
| + self._properties['ERRORS'].append(msg) |
| + ErrOut.LogLine(filename, lineno, 0, ' %s %s' % (str(self), msg)) |
| + |
| + # Log a warning for this object |
| + def Warning(self, msg): |
| + filename, lineno = self.GetFileAndLine() |
| + self._properties['WARNINGS'].append(msg) |
| + WarnOut.LogLine(self.filename, self.lineno, 0, ' %s %s' % |
| + (str(self), msg)) |
| + |
| + # Return file and line number for where node was defined |
| + def GetFileAndLine(self): |
| + return self._properties['FILENAME'], self._properties['LINENO'] |
|
sehr
2013/04/08 18:37:07
use self.GetProperty here.
|
| + |
| + def GetName(self): |
| + return self.GetProperty('NAME') |
| + |
| + def GetParent(self): |
| + return self._parent |
| + |
| + def Traverse(self, search, filter_nodes): |
| + if self._cls in filter_nodes: |
| + return '' |
| + |
| + search.Enter(self) |
| + search.depth += 1 |
| + for child in self._children: |
| + child.Traverse(search, filter_nodes) |
| + search.depth -= 1 |
| + search.Exit(self) |
| + |
| + |
| + def Tree(self, filter_nodes=None, accept_props=None): |
| + class DumpTreeSearch(IDLSearch): |
| + def __init__(self, props): |
| + IDLSearch.__init__(self) |
| + self.out = [] |
| + self.props = props |
| + |
| + def Enter(self, node): |
| + tab = ''.rjust(self.depth * 2) |
| + self.out.append(tab + str(node)) |
| + if self.props: |
| + for key, value in node._properties: |
| + proplist = [] |
| + if key in self.props: |
| + proplist.append(tab + ' %s: %s' % |
| + (prop, str(node.properties[prop]))) |
| + if proplist: |
| + self.out.append(tab + ' PROPERTIES') |
| + self.out.extend(proplist) |
| + |
| + if filter_nodes == None: |
| + filter_nodes = ['Comment', 'Copyright'] |
| + |
| + search = DumpTreeSearch(accept_props) |
| + self.Traverse(search, filter_nodes) |
| + return search.out |
| + |
| +# |
| +# Search related functions |
| +# |
| + # Check if node is of a given type |
| + def IsA(self, *typelist): |
| + if self._cls in typelist: return True |
|
sehr
2013/04/08 18:37:07
Does this work?
return self._cls in typelist
|
| + return False |
| + |
| + # Get a list of all children |
| + def GetChildren(self): |
| + return self._children |
| + |
| + def GetListOf(self, *keys): |
| + out = [] |
| + for child in self._children: |
| + if child._cls in keys: out.append(child) |
| + return out |
| + |
| + def GetOneOf(self, *keys): |
| + out = self.GetListOf(*keys) |
| + if out: return out[0] |
| + return None |
| + |
| + def AddChildren(self, children): |
| + children = CopyToList(children) |
| + for child in children: |
| + if not child: |
| + continue |
| + if type(child) == IDLAttribute: |
| + self.SetProperty(child._name, child._value) |
| + continue |
| + if type(child) == IDLNode: |
| + child._parent = self |
| + self._children.append(child) |
| + continue |
| + raise RuntimeError('Adding child of type .\n' % type(child).__name__) |
| + |
| + |
| +# |
| +# Property Functions |
| +# |
| + def SetProperty(self, name, val): |
| + self._properties[name] = val |
| + |
| + def GetProperty(self, name, default=None): |
| + return self._properties.get(name, default) |