Skip to content
Projects
Groups
Snippets
Help
Loading...
Sign in
Toggle navigation
A
arpeggio-gm
Project
Project
Details
Activity
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Commits
Issue Boards
Open sidebar
backend
arpeggio-gm
Commits
6e82b6ca
Commit
6e82b6ca
authored
Jun 30, 2014
by
Igor Dejanovic
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Some PEP-8 fixes.
parent
35c767fc
Hide whitespace changes
Inline
Side-by-side
Showing
1 changed file
with
44 additions
and
33 deletions
+44
-33
__init__.py
arpeggio/__init__.py
+44
-33
No files found.
arpeggio/__init__.py
View file @
6e82b6ca
# -*- coding: utf-8 -*-
# -*- coding: utf-8 -*-
###############################################################################
#
###############################################################################
# Name: arpeggio.py
# Name: arpeggio.py
# Purpose: PEG parser interpreter
# Purpose: PEG parser interpreter
# Author: Igor R. Dejanović <igor DOT dejanovic AT gmail DOT com>
# Author: Igor R. Dejanović <igor DOT dejanovic AT gmail DOT com>
# Copyright: (c) 2009 Igor R. Dejanović <igor DOT dejanovic AT gmail DOT com>
# Copyright: (c) 2009 Igor R. Dejanović <igor DOT dejanovic AT gmail DOT com>
# License: MIT License
# License: MIT License
#
#
# This is an implementation of packrat parser interpreter based on PEG
grammars.
# This is an implementation of packrat parser interpreter based on PEG
#
Grammars are defined using Python language constructs or the PEG textual
#
grammars. Grammars are defined using Python language constructs or the PEG
# notation.
#
textual
notation.
###############################################################################
#
###############################################################################
from
__future__
import
print_function
,
unicode_literals
from
__future__
import
print_function
,
unicode_literals
import
re
import
re
...
@@ -51,7 +51,8 @@ class NoMatch(Exception):
...
@@ -51,7 +51,8 @@ class NoMatch(Exception):
Args:
Args:
value (str): A name of the parsing expression or rule.
value (str): A name of the parsing expression or rule.
position (int): A position in the input stream where exception occurred.
position (int): A position in the input stream where exception
occurred.
parser (Parser): An instance of a parser.
parser (Parser): An instance of a parser.
"""
"""
def
__init__
(
self
,
rule
,
position
,
parser
):
def
__init__
(
self
,
rule
,
position
,
parser
):
...
@@ -177,7 +178,8 @@ class ParsingExpression(object):
...
@@ -177,7 +178,8 @@ class ParsingExpression(object):
result
,
new_pos
=
self
.
result_cache
[
c_pos
]
result
,
new_pos
=
self
.
result_cache
[
c_pos
]
parser
.
position
=
new_pos
parser
.
position
=
new_pos
if
parser
.
debug
:
if
parser
.
debug
:
print
(
"** Cache hit for [{}, {}] = '{}'"
.
format
(
self
.
name
,
c_pos
,
str
(
result
)))
print
(
"** Cache hit for [{}, {}] = '{}'"
.
format
(
self
.
name
,
c_pos
,
str
(
result
)))
if
parser
.
debug
:
if
parser
.
debug
:
print
(
"<< Leaving rule {}"
.
format
(
self
.
name
))
print
(
"<< Leaving rule {}"
.
format
(
self
.
name
))
return
result
return
result
...
@@ -429,12 +431,12 @@ class Match(ParsingExpression):
...
@@ -429,12 +431,12 @@ class Match(ParsingExpression):
@property
@property
def
name
(
self
):
def
name
(
self
):
return
"
%
s(
%
s)"
%
(
self
.
__class__
.
__name__
,
self
.
to_match
)
return
"
%
s(
%
s)"
%
(
self
.
__class__
.
__name__
,
self
.
to_match
)
def
parse
(
self
,
parser
):
def
parse
(
self
,
parser
):
self
.
_parse_intro
(
parser
)
self
.
_parse_intro
(
parser
)
if
parser
.
_in_parse_comment
:
if
parser
.
_in_parse_comment
:
return
self
.
_parse
(
parser
)
return
self
.
_parse
(
parser
)
c_pos
=
parser
.
position
c_pos
=
parser
.
position
comments
=
[]
comments
=
[]
...
@@ -472,8 +474,8 @@ class RegExMatch(Match):
...
@@ -472,8 +474,8 @@ class RegExMatch(Match):
This Match class will perform input matching based on Regular Expressions.
This Match class will perform input matching based on Regular Expressions.
Args:
Args:
to_match (regex string): A regular expression string to match.
It will be
to_match (regex string): A regular expression string to match.
used to create regular expression using re.compile.
It will be
used to create regular expression using re.compile.
'''
'''
def
__init__
(
self
,
to_match
,
rule
=
None
,
flags
=
None
):
def
__init__
(
self
,
to_match
,
rule
=
None
,
flags
=
None
):
super
(
RegExMatch
,
self
)
.
__init__
(
rule
)
super
(
RegExMatch
,
self
)
.
__init__
(
rule
)
...
@@ -529,10 +531,10 @@ class StrMatch(Match):
...
@@ -529,10 +531,10 @@ class StrMatch(Match):
def
__eq__
(
self
,
other
):
def
__eq__
(
self
,
other
):
return
self
.
to_match
==
str
(
other
)
return
self
.
to_match
==
str
(
other
)
def
__hash__
(
self
):
def
__hash__
(
self
):
return
hash
(
self
.
to_match
)
return
hash
(
self
.
to_match
)
# HACK: Kwd class is a bit hackish. Need to find a better way to
# HACK: Kwd class is a bit hackish. Need to find a better way to
# introduce different classes of string tokens.
# introduce different classes of string tokens.
...
@@ -568,7 +570,8 @@ class EndOfFile(Match):
...
@@ -568,7 +570,8 @@ class EndOfFile(Match):
parser
.
_nm_raise
(
self
.
name
,
c_pos
,
parser
)
parser
.
_nm_raise
(
self
.
name
,
c_pos
,
parser
)
def
EOF
():
return
EndOfFile
()
def
EOF
():
return
EndOfFile
()
# ---------------------------------------------------------
# ---------------------------------------------------------
...
@@ -584,8 +587,10 @@ class ParseTreeNode(object):
...
@@ -584,8 +587,10 @@ class ParseTreeNode(object):
Attributes:
Attributes:
rule (str): The name of the rule that created this node or empty
rule (str): The name of the rule that created this node or empty
string in case this node is created by a non-root pexpression.
string in case this node is created by a non-root pexpression.
position (int): A position in the input stream where the match occurred.
position (int): A position in the input stream where the match
error (bool): Is this a false parse tree node created during error recovery.
occurred.
error (bool): Is this a false parse tree node created during error
recovery.
comments : A parse tree of comment(s) attached to this node.
comments : A parse tree of comment(s) attached to this node.
"""
"""
def
__init__
(
self
,
rule
,
position
,
error
):
def
__init__
(
self
,
rule
,
position
,
error
):
...
@@ -698,7 +703,8 @@ class SemanticAction(object):
...
@@ -698,7 +703,8 @@ class SemanticAction(object):
def
first_pass
(
self
,
parser
,
node
,
nodes
):
def
first_pass
(
self
,
parser
,
node
,
nodes
):
"""
"""
Called in the first pass of tree walk.
Called in the first pass of tree walk.
This is the default implementation used if no semantic action is defined.
This is the default implementation used if no semantic action is
defined.
"""
"""
# Special case. If only one child exist return it.
# Special case. If only one child exist return it.
if
len
(
nodes
)
==
1
:
if
len
(
nodes
)
==
1
:
...
@@ -720,7 +726,7 @@ class SemanticAction(object):
...
@@ -720,7 +726,7 @@ class SemanticAction(object):
else
:
else
:
# Return the only non-string child
# Return the only non-string child
retval
=
last_non_str
retval
=
last_non_str
# If there is only one non-string child return
return
retval
return
retval
...
@@ -740,7 +746,8 @@ class Parser(object):
...
@@ -740,7 +746,8 @@ class Parser(object):
debug (bool): If true debugging messages will be printed.
debug (bool): If true debugging messages will be printed.
comments_model: parser model for comments.
comments_model: parser model for comments.
"""
"""
def
__init__
(
self
,
skipws
=
True
,
ws
=
DEFAULT_WS
,
reduce_tree
=
True
,
debug
=
False
):
def
__init__
(
self
,
skipws
=
True
,
ws
=
DEFAULT_WS
,
reduce_tree
=
True
,
debug
=
False
):
self
.
skipws
=
skipws
self
.
skipws
=
skipws
self
.
ws
=
ws
self
.
ws
=
ws
self
.
reduce_tree
=
reduce_tree
self
.
reduce_tree
=
reduce_tree
...
@@ -766,9 +773,9 @@ class Parser(object):
...
@@ -766,9 +773,9 @@ class Parser(object):
Creates Abstract Semantic Graph (ASG) from the parse tree.
Creates Abstract Semantic Graph (ASG) from the parse tree.
Args:
Args:
sem_actions (dict): The semantic actions dictionary to use for
semantic
sem_actions (dict): The semantic actions dictionary to use for
analysis. Rule names are the keys and semantic action objects are
semantic analysis. Rule names are the keys and semantic action
values.
objects are
values.
"""
"""
if
not
self
.
parse_tree
:
if
not
self
.
parse_tree
:
raise
Exception
(
"Parse tree is empty. You did call parse(), didn't you?"
)
raise
Exception
(
"Parse tree is empty. You did call parse(), didn't you?"
)
...
@@ -791,20 +798,23 @@ class Parser(object):
...
@@ -791,20 +798,23 @@ class Parser(object):
called in the second pass.
called in the second pass.
"""
"""
if
self
.
debug
:
if
self
.
debug
:
print
(
"Walking down "
,
node
.
name
,
" type:"
,
type
(
node
),
"str:"
,
str
(
node
))
print
(
"Walking down "
,
node
.
name
,
" type:"
,
type
(
node
),
"str:"
,
str
(
node
))
children
=
[]
children
=
[]
if
isinstance
(
node
,
NonTerminal
):
if
isinstance
(
node
,
NonTerminal
):
for
n
in
node
:
for
n
in
node
:
children
.
append
(
tree_walk
(
n
))
children
.
append
(
tree_walk
(
n
))
if
self
.
debug
:
if
self
.
debug
:
print
(
"Visiting "
,
node
.
name
,
"= '"
,
str
(
node
),
"' type:"
,
type
(
node
),
\
print
(
"Visiting "
,
node
.
name
,
"= '"
,
str
(
node
),
"' type:"
,
type
(
node
),
\
"len:"
,
len
(
node
)
if
isinstance
(
node
,
list
)
else
""
)
"len:"
,
len
(
node
)
if
isinstance
(
node
,
list
)
else
""
)
for
i
,
a
in
enumerate
(
children
):
for
i
,
a
in
enumerate
(
children
):
print
(
"
\t
%
d:"
%
(
i
+
1
),
str
(
a
),
"type:"
,
type
(
a
))
print
(
"
\t
%
d:"
%
(
i
+
1
),
str
(
a
),
"type:"
,
type
(
a
))
if
node
.
rule
in
sem_actions
:
if
node
.
rule
in
sem_actions
:
retval
=
sem_actions
[
node
.
rule
]
.
first_pass
(
self
,
node
,
children
)
retval
=
sem_actions
[
node
.
rule
]
.
first_pass
(
self
,
node
,
children
)
if
hasattr
(
sem_actions
[
node
.
rule
],
"second_pass"
):
if
hasattr
(
sem_actions
[
node
.
rule
],
"second_pass"
):
for_second_pass
.
append
((
node
.
rule
,
retval
))
for_second_pass
.
append
((
node
.
rule
,
retval
))
else
:
else
:
...
@@ -857,14 +867,14 @@ class Parser(object):
...
@@ -857,14 +867,14 @@ class Parser(object):
if
self
.
input
[
self
.
line_ends
[
line
-
1
]]
in
'
\n\r
'
:
if
self
.
input
[
self
.
line_ends
[
line
-
1
]]
in
'
\n\r
'
:
col
-=
1
col
-=
1
return
line
+
1
,
col
+
1
return
line
+
1
,
col
+
1
def
context
(
self
,
length
=
None
,
position
=
None
):
def
context
(
self
,
length
=
None
,
position
=
None
):
"""
"""
Returns current context substring, i.e. the substring around current
Returns current context substring, i.e. the substring around current
position.
position.
Args:
Args:
length(int): If given used to mark with asterisk a length chars
from
length(int): If given used to mark with asterisk a length chars
current position.
from the
current position.
position(int): The position in the input stream.
position(int): The position in the input stream.
"""
"""
if
not
position
:
if
not
position
:
...
@@ -873,7 +883,7 @@ class Parser(object):
...
@@ -873,7 +883,7 @@ class Parser(object):
return
"{}*{}*{}"
.
format
(
return
"{}*{}*{}"
.
format
(
str
(
self
.
input
[
max
(
position
-
10
,
0
):
position
]),
str
(
self
.
input
[
max
(
position
-
10
,
0
):
position
]),
str
(
self
.
input
[
position
:
position
+
length
]),
str
(
self
.
input
[
position
:
position
+
length
]),
str
(
self
.
input
[
position
+
length
:
position
+
10
]))
str
(
self
.
input
[
position
+
length
:
position
+
10
]))
else
:
else
:
return
"{}*{}"
.
format
(
return
"{}*{}"
.
format
(
str
(
self
.
input
[
max
(
position
-
10
,
0
):
position
]),
str
(
self
.
input
[
max
(
position
-
10
,
0
):
position
]),
...
@@ -960,7 +970,8 @@ class ParserPython(Parser):
...
@@ -960,7 +970,8 @@ class ParserPython(Parser):
if
isinstance
(
c_rule
,
CrossRef
):
if
isinstance
(
c_rule
,
CrossRef
):
self
.
__cross_refs
+=
1
self
.
__cross_refs
+=
1
if
self
.
debug
:
if
self
.
debug
:
print
(
"CrossRef usage: {}"
.
format
(
c_rule
.
rule_name
))
print
(
"CrossRef usage: {}"
.
format
(
c_rule
.
rule_name
))
return
c_rule
return
c_rule
# Semantic action for the rule
# Semantic action for the rule
...
@@ -977,8 +988,8 @@ class ParserPython(Parser):
...
@@ -977,8 +988,8 @@ class ParserPython(Parser):
# Update cache
# Update cache
__rule_cache
[
rule
]
=
retval
__rule_cache
[
rule
]
=
retval
if
self
.
debug
:
if
self
.
debug
:
print
(
"New rule: {} -> {}"
.
format
(
rule
,
print
(
"New rule: {} -> {}"
retval
.
__class__
.
__name__
))
.
format
(
rule
,
retval
.
__class__
.
__name__
))
elif
isinstance
(
expression
,
Match
):
elif
isinstance
(
expression
,
Match
):
retval
=
expression
retval
=
expression
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment