PLY Internals¶
1. Introduction¶
This document describes classes and functions that make up the internal operation of PLY. Using this programming interface, it is possible to manually build an parser using a different interface specification than what PLY normally uses. For example, you could build a gramar from information parsed in a completely different input format. Some of these objects may be useful for building more advanced parsing engines such as GLR.
It should be stressed that using PLY at this level is not for the faint of heart. Generally, it’s assumed that you know a bit of the underlying compiler theory and how an LR parser is put together.
2. Grammar Class¶
The file ply.yacc defines a class Grammar that
is used to hold and manipulate information about a grammar
specification. It encapsulates the same basic information
about a grammar that is put into a YACC file including
the list of tokens, precedence rules, and grammar rules.
Various operations are provided to perform different validations
on the grammar. In addition, there are operations to compute
the first and follow sets that are needed by the various table
generation algorithms.
Grammar(terminals)- Creates a new grammar object.
terminalsis a list of strings specifying the terminals for the grammar. An instancegofGrammarhas the following methods: g.set_precedence(term,assoc,level)Sets the precedence level and associativity for a given terminal
term.associs one of'right','left', or'nonassoc'andlevelis a positive integer. The higher the value oflevel, the higher the precedence. Here is an example of typical precedence settings:g.set_precedence('PLUS', 'left',1) g.set_precedence('MINUS', 'left',1) g.set_precedence('TIMES', 'left',2) g.set_precedence('DIVIDE','left',2) g.set_precedence('UMINUS','left',3)
This method must be called prior to adding any productions to the grammar with
g.add_production(). The precedence of individual grammar rules is determined by the precedence of the right-most terminal.g.add_production(name,syms,func=None,file='',line=0)Adds a new grammar rule.
nameis the name of the rule,symsis a list of symbols making up the right hand side of the rule,funcis the function to call when reducing the rule.fileandlinespecify the filename and line number of the rule and are used for generating error messages.The list of symbols in
symsmay include character literals and%precspecifiers. Here are some examples:g.add_production('expr',['expr','PLUS','term'],func,file,line) g.add_production('expr',['expr','"+"','term'],func,file,line) g.add_production('expr',['MINUS','expr','%prec','UMINUS'],func,file,line)
If any kind of error is detected, a
GrammarErrorexception is raised with a message indicating the reason for the failure.g.set_start(start=None)- Sets the starting rule for the grammar.
startis a string specifying the name of the start rule. Ifstartis omitted, the first grammar rule added withadd_production()is taken to be the starting rule. This method must always be called after all productions have been added. g.find_unreachable()- Diagnostic function. Returns a list of all unreachable non-terminals defined in the grammar. This is used to identify inactive parts of the grammar specification.
g.infinite_cycle()- Diagnostic function. Returns a list of all non-terminals in the grammar that result in an infinite cycle. This condition occurs if there is no way for a grammar rule to expand to a string containing only terminal symbols.
g.undefined_symbols()- Diagnostic function. Returns a list of tuples
(name, prod)corresponding to undefined symbols in the grammar.nameis the name of the undefined symbol andprodis an instance ofProductionwhich has information about the production rule where the undefined symbol was used. g.unused_terminals()- Diagnostic function. Returns a list of terminals that were defined, but never used in the grammar.
g.unused_rules()- Diagnostic function. Returns a list of
Productioninstances corresponding to production rules that were defined in the grammar, but never used anywhere. This is slightly different thanfind_unreachable(). g.unused_precedence()- Diagnostic function. Returns a list of tuples
(term, assoc)corresponding to precedence rules that were set, but never used the grammar.termis the terminal name andassocis the precedence associativity (e.g.,'left','right', or'nonassoc'. g.compute_first()- Compute all of the first sets for all symbols in the grammar. Returns a dictionary mapping symbol names to a list of all first symbols.
g.compute_follow()- Compute all of the follow sets for all non-terminals in the grammar. The follow set is the set of all possible symbols that might follow a given non-terminal. Returns a dictionary mapping non-terminal names to a list of symbols.
g.build_lritems()- Calculates all of the LR items for all productions in the grammar. This step is required before using the grammar for any kind of table generation. See the section on LR items below.
The following attributes are set by the above methods and may be useful in code that works with the grammar. All of these attributes should be assumed to be read-only. Changing their values directly will likely break the grammar.
g.Productions- A list of all productions added. The first entry is reserved for
a production representing the starting rule. The objects in this list
are instances of the
Productionclass, described shortly. g.Prodnames- A dictionary mapping the names of nonterminals to a list of all productions of that nonterminal.
g.Terminals- A dictionary mapping the names of terminals to a list of the production numbers where they are used.
g.Nonterminals- A dictionary mapping the names of nonterminals to a list of the production numbers where they are used.
g.First- A dictionary representing the first sets for all grammar symbols. This is
computed and returned by the
compute_first()method. g.Follow- A dictionary representing the follow sets for all grammar rules. This is
computed and returned by the
compute_follow()method. g.Start- Starting symbol for the grammar. Set by the
set_start()method.
For the purposes of debugging, a Grammar object supports the __len__() and
__getitem__() special methods. Accessing g[n] returns the nth production
from the grammar.
3. Productions¶
Grammar objects store grammar rules as instances of a Production class. This
class has no public constructor–you should only create productions by calling Grammar.add_production().
The following attributes are available on a Production instance p.
p.name- The name of the production. For a grammar rule such as
A : B C D, this is'A'. p.prod- A tuple of symbols making up the right-hand side of the production. For a grammar rule such as
A : B C D, this is('B','C','D'). p.number- Production number. An integer containing the index of the production in the grammar’s
Productionslist. p.func- The name of the reduction function associated with the production. This is the function that will execute when reducing the entire grammar rule during parsing.
p.callable- The callable object associated with the name in
p.func. This isNoneunless the production has been bound usingbind(). p.file- Filename associated with the production. Typically this is the file where the production was defined. Used for error messages.
p.lineno- Line number associated with the production. Typically this is the line number in
p.filewhere the production was defined. Used for error messages. p.prec- Precedence and associativity associated with the production. This is a tuple
(assoc,level)whereassocis one of'left',``’right’, or ``'nonassoc'andlevelis an integer. This value is determined by the precedence of the right-most terminal symbol in the production or by use of the%precspecifier when adding the production. p.usyms- A list of all unique symbols found in the production.
p.lr_items- A list of all LR items for this production. This attribute only has a meaningful value if the
Grammar.build_lritems()method has been called. The items in this list are instances ofLRItemdescribed below. p.lr_next- The head of a linked-list representation of the LR items in
p.lr_items. This attribute only has a meaningful value if theGrammar.build_lritems()method has been called. EachLRIteminstance has alr_nextattribute to move to the next item. The list is terminated byNone. p.bind(dict)- Binds the production function name in
p.functo a callable object indict. This operation is typically carried out in the last step prior to running the parsing engine and is needed since parsing tables are typically read from files which only include the function names, not the functions themselves.
Production objects support
the __len__(), __getitem__(), and __str__()
special methods.
len(p) returns the number of symbols in p.prod
and p[n] is the same as p.prod[n].
4. LRItems¶
The construction of parsing tables in an LR-based parser generator is primarily
done over a set of “LR Items”. An LR item represents a stage of parsing one
of the grammar rules. To compute the LR items, it is first necessary to
call Grammar.build_lritems(). Once this step, all of the productions
in the grammar will have their LR items attached to them.
Here is an interactive example that shows what LR items look like if you
interactively experiment. In this example, g is a Grammar
object:
>>> g.build_lritems()
>>> p = g[1]
>>> p
Production(statement -> ID = expr)
>>>
In the above code, p represents the first grammar rule. In
this case, a rule 'statement -> ID = expr'.
Now, let’s look at the LR items for p:
>>> p.lr_items
[LRItem(statement -> . ID = expr),
LRItem(statement -> ID . = expr),
LRItem(statement -> ID = . expr),
LRItem(statement -> ID = expr .)]
>>>
In each LR item, the dot (.) represents a specific stage of parsing. In each LR item, the dot is advanced by one symbol. It is only when the dot reaches the very end that a production is successfully parsed.
An instance lr of LRItem has the following
attributes that hold information related to that specific stage of
parsing.
lr.name- The name of the grammar rule. For example,
'statement'in the above example. lr.prod- A tuple of symbols representing the right-hand side of the production, including the
special
'.'character. For example,('ID','.','=','expr'). lr.number- An integer representing the production number in the grammar.
lr.usyms- A set of unique symbols in the production. Inherited from the original
Productioninstance. lr.lr_index- An integer representing the position of the dot (.). You should never use
lr.prod.index()to search for it–the result will be wrong if the grammar happens to also use (.) as a character literal. lr.lr_afterA list of all productions that can legally appear immediately to the right of the dot (.). This list contains
Productioninstances. This attribute represents all of the possible branches a parse can take from the current position. For example, suppose thatlrrepresents a stage immediately before an expression like this:>>> lr LRItem(statement -> ID = . expr) >>>
Then, the value of
lr.lr_aftermight look like this, showing all productions that can legally appear next:>>> lr.lr_after [Production(expr -> expr PLUS expr), Production(expr -> expr MINUS expr), Production(expr -> expr TIMES expr), Production(expr -> expr DIVIDE expr), Production(expr -> MINUS expr), Production(expr -> LPAREN expr RPAREN), Production(expr -> NUMBER), Production(expr -> ID)] >>>
lr.lr_before- The grammar symbol that appears immediately before the dot (.) or
Noneif at the beginning of the parse. lr.lr_next- A link to the next LR item, representing the next stage of the parse.
Noneiflris the last LR item.
LRItem instances also support the __len__() and __getitem__() special methods.
len(lr) returns the number of items in lr.prod including the dot (.). lr[n]
returns lr.prod[n].
It goes without saying that all of the attributes associated with LR items should be assumed to be read-only. Modifications will very likely create a small black-hole that will consume you and your code.
5. LRTable¶
The LRTable class represents constructed LR parsing tables on a
grammar.
LRTable(grammar, log=None)- Create the LR parsing tables on a grammar.
grammaris an instance ofGrammarandlogis a logger object used to write debugging information. The debugging information written tologis the same as what appears in theparser.outfile created by yacc. By supplying a custom logger with a different message format, it is possible to get more information (e.g., the line number inyacc.pyused for issuing each line of output in the log).
An instance lr of LRTable has the following attributes.
lr.grammar- A link to the Grammar object used to construct the parsing tables.
lr.lr_method- The LR parsing method used (e.g.,
'LALR') lr.lr_productions- A reference to
grammar.Productions. This, together withlr_actionandlr_gotocontain all of the information needed by the LR parsing engine. lr.lr_action- The LR action dictionary that implements the underlying state machine. The keys of this dictionary are the LR states.
lr.lr_goto- The LR goto table that contains information about grammar rule reductions.
lr.sr_conflicts- A list of tuples
(state,token,resolution)identifying all shift/reduce conflicts.stateis the LR state number where the conflict occurred,tokenis the token causing the conflict, andresolutionis a string describing the resolution taken.resolutionis either'shift'or'reduce'. lr.rr_conflicts- A list of tuples
(state,rule,rejected)identifying all reduce/reduce conflicts.stateis the LR state number where the conflict occurred,ruleis the production rule that was selected andrejectedis the production rule that was rejected. Bothruleandrejectedare instances ofProduction. They can be inspected to provide the user with more information. lrtab.bind_callables(dict)- This binds all of the function names used in productions to callable objects
found in the dictionary
dict. During table generation and when reading LR tables from files, PLY only uses the names of action functions such as'p_expr','p_statement', etc. In order to actually run the parser, these names have to be bound to callable objects. This method is always called prior to running a parser.
6. LRParser¶
The LRParser class implements the low-level LR parsing engine.
LRParser(lrtab, error_func)- Create an LRParser.
lrtabis an instance ofLRTablecontaining the LR production and state tables.error_funcis the error function to invoke in the event of a parsing error.
An instance p of LRParser has the following methods:
p.parse(input=None,lexer=None,debug=0,tracking=0)- Run the parser.
inputis a string, which if supplied is fed into the lexer using itsinput()method.lexeris an instance of theLexerclass to use for tokenizing. If not supplied, the last lexer created with thelexmodule is used.debugis a boolean flag that enables debugging.trackingis a boolean flag that tells the parser to perform additional line number tracking. p.restart()- Resets the parser state for a parse already in progress.
7. ParserReflect¶
The ParserReflect class is used to collect parser specification data
from a Python module or object. This class is what collects all of the
p_rule() functions in a PLY file, performs basic error checking,
and collects all of the needed information to build a grammar. Most of the
high-level PLY interface as used by the yacc() function is actually
implemented by this class.
ParserReflect(pdict, log=None)- Creates a
ParserReflectinstance.pdictis a dictionary containing parser specification data. This dictionary typically corresponds to the module or class dictionary of code that implements a PLY parser.logis a logger instance that will be used to report error messages.
An instance p of ParserReflect has the following methods:
p.get_all()- Collect and store all required parsing information.
p.validate_all()- Validate all of the collected parsing information. This is a seprate step
from
p.get_all()as a performance optimization. In order to increase parser start-up time, a parser can elect to only validate the parsing data when regenerating the parsing tables. The validation step tries to collect as much information as possible rather than raising an exception at the first sign of trouble. The attributep.erroris set if there are any validation errors. The value of this attribute is also returned. p.signature()- Compute a signature representing the contents of the collected parsing
data. The signature value should change if anything in the parser
specification has changed in a way that would justify parser table
regeneration. This method can be called after
p.get_all(), but beforep.validate_all().
The following attributes are set in the process of collecting data:
p.start- The grammar start symbol, if any. Taken from
pdict['start']. p.error_func- The error handling function or
None. Taken frompdict['p_error']. p.tokens- The token list. Taken from
pdict['tokens']. p.prec- The precedence specifier. Taken from
pdict['precedence']. p.preclist- A parsed version of the precedence specified. A list of tuples of the form
(token,assoc,level)wheretokenis the terminal symbol,associs the associativity (e.g.,'left') andlevelis a numeric precedence level. p.grammarA list of tuples
(name, rules)representing the grammar rules.nameis the name of a Python function or method inpdictthat starts with"p_".rulesis a list of tuples(filename,line,prodname,syms)representing the grammar rules found in the documentation string of that function.filenameandlinecontain location information that can be used for debugging.prodnameis the name of the production.symsis the right-hand side of the production. If you have a function like this:def p_expr(p): '''expr : expr PLUS expr | expr MINUS expr | expr TIMES expr | expr DIVIDE expr'''
then the corresponding entry in
p.grammarmight look like this:('p_expr', [ ('calc.py',10,'expr', ['expr','PLUS','expr']), ('calc.py',11,'expr', ['expr','MINUS','expr']), ('calc.py',12,'expr', ['expr','TIMES','expr']), ('calc.py',13,'expr', ['expr','DIVIDE','expr']) ])
p.pfuncs- A sorted list of tuples
(line, file, name, doc)representing all of thep_functions found.lineandfilegive location information.nameis the name of the function.docis the documentation string. This list is sorted in ascending order by line number. p.files- A dictionary holding all of the source filenames that were encountered while collecting parser information. Only the keys of this dictionary have any meaning.
p.error- An attribute that indicates whether or not any critical errors occurred in validation. If this is set, it means that that some kind of problem was detected and that no further processing should be performed.
8. High-level operation¶
Using all of the above classes requires some attention to detail. The yacc()
function carries out a very specific sequence of operations to create a grammar.
This same sequence should be emulated if you build an alternative PLY interface.
1. A ParserReflect object is created and raw grammar specification data is
collected.
2. A Grammar object is created and populated with information
from the specification data.
3. A LRTable object is created to run the LALR algorithm over
the Grammar object.
4. Productions in the LRTable and bound to callables using the bind_callables()
method.
5. A LRParser object is created from from the information in the
LRTable object.