forked from mirrors/gecko-dev
		
	 ab55fb68d1
			
		
	
	
		ab55fb68d1
		
	
	
	
	
		
			
			The license used to be LGPL so the code lived in other-licenses, but it was changed to BSD eleven years ago. Let's move it over to third_party/python/ply where it belongs.
    ./mach vendor python ply==3.10
`diff -r` between the original `ply` directory and the new one only comes up with the new file `third_party/python/ply/CHANGES` which isn't relevant to the functionality of the code, so this should be a no-op all told.
Differential Revision: https://phabricator.services.mozilla.com/D73341
		
	
			
		
			
				
	
	
		
			165 lines
		
	
	
	
		
			4 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable file
		
	
	
	
	
			
		
		
	
	
			165 lines
		
	
	
	
		
			4 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable file
		
	
	
	
	
| #!/usr/bin/env python
 | |
| 
 | |
| # -----------------------------------------------------------------------------
 | |
| # calc.py
 | |
| #
 | |
| # A simple calculator with variables.   This is from O'Reilly's
 | |
| # "Lex and Yacc", p. 63.
 | |
| #
 | |
| # Class-based example contributed to PLY by David McNab
 | |
| # -----------------------------------------------------------------------------
 | |
| 
 | |
| import sys
 | |
| sys.path.insert(0, "../..")
 | |
| 
 | |
| if sys.version_info[0] >= 3:
 | |
|     raw_input = input
 | |
| 
 | |
| import ply.lex as lex
 | |
| import ply.yacc as yacc
 | |
| import os
 | |
| 
 | |
| 
 | |
| class Parser:
 | |
|     """
 | |
|     Base class for a lexer/parser that has the rules defined as methods
 | |
|     """
 | |
|     tokens = ()
 | |
|     precedence = ()
 | |
| 
 | |
|     def __init__(self, **kw):
 | |
|         self.debug = kw.get('debug', 0)
 | |
|         self.names = {}
 | |
|         try:
 | |
|             modname = os.path.split(os.path.splitext(__file__)[0])[
 | |
|                 1] + "_" + self.__class__.__name__
 | |
|         except:
 | |
|             modname = "parser" + "_" + self.__class__.__name__
 | |
|         self.debugfile = modname + ".dbg"
 | |
|         self.tabmodule = modname + "_" + "parsetab"
 | |
|         # print self.debugfile, self.tabmodule
 | |
| 
 | |
|         # Build the lexer and parser
 | |
|         lex.lex(module=self, debug=self.debug)
 | |
|         yacc.yacc(module=self,
 | |
|                   debug=self.debug,
 | |
|                   debugfile=self.debugfile,
 | |
|                   tabmodule=self.tabmodule)
 | |
| 
 | |
|     def run(self):
 | |
|         while 1:
 | |
|             try:
 | |
|                 s = raw_input('calc > ')
 | |
|             except EOFError:
 | |
|                 break
 | |
|             if not s:
 | |
|                 continue
 | |
|             yacc.parse(s)
 | |
| 
 | |
| 
 | |
| class Calc(Parser):
 | |
| 
 | |
|     tokens = (
 | |
|         'NAME', 'NUMBER',
 | |
|         'PLUS', 'MINUS', 'EXP', 'TIMES', 'DIVIDE', 'EQUALS',
 | |
|         'LPAREN', 'RPAREN',
 | |
|     )
 | |
| 
 | |
|     # Tokens
 | |
| 
 | |
|     t_PLUS = r'\+'
 | |
|     t_MINUS = r'-'
 | |
|     t_EXP = r'\*\*'
 | |
|     t_TIMES = r'\*'
 | |
|     t_DIVIDE = r'/'
 | |
|     t_EQUALS = r'='
 | |
|     t_LPAREN = r'\('
 | |
|     t_RPAREN = r'\)'
 | |
|     t_NAME = r'[a-zA-Z_][a-zA-Z0-9_]*'
 | |
| 
 | |
|     def t_NUMBER(self, t):
 | |
|         r'\d+'
 | |
|         try:
 | |
|             t.value = int(t.value)
 | |
|         except ValueError:
 | |
|             print("Integer value too large %s" % t.value)
 | |
|             t.value = 0
 | |
|         # print "parsed number %s" % repr(t.value)
 | |
|         return t
 | |
| 
 | |
|     t_ignore = " \t"
 | |
| 
 | |
|     def t_newline(self, t):
 | |
|         r'\n+'
 | |
|         t.lexer.lineno += t.value.count("\n")
 | |
| 
 | |
|     def t_error(self, t):
 | |
|         print("Illegal character '%s'" % t.value[0])
 | |
|         t.lexer.skip(1)
 | |
| 
 | |
|     # Parsing rules
 | |
| 
 | |
|     precedence = (
 | |
|         ('left', 'PLUS', 'MINUS'),
 | |
|         ('left', 'TIMES', 'DIVIDE'),
 | |
|         ('left', 'EXP'),
 | |
|         ('right', 'UMINUS'),
 | |
|     )
 | |
| 
 | |
|     def p_statement_assign(self, p):
 | |
|         'statement : NAME EQUALS expression'
 | |
|         self.names[p[1]] = p[3]
 | |
| 
 | |
|     def p_statement_expr(self, p):
 | |
|         'statement : expression'
 | |
|         print(p[1])
 | |
| 
 | |
|     def p_expression_binop(self, p):
 | |
|         """
 | |
|         expression : expression PLUS expression
 | |
|                   | expression MINUS expression
 | |
|                   | expression TIMES expression
 | |
|                   | expression DIVIDE expression
 | |
|                   | expression EXP expression
 | |
|         """
 | |
|         # print [repr(p[i]) for i in range(0,4)]
 | |
|         if p[2] == '+':
 | |
|             p[0] = p[1] + p[3]
 | |
|         elif p[2] == '-':
 | |
|             p[0] = p[1] - p[3]
 | |
|         elif p[2] == '*':
 | |
|             p[0] = p[1] * p[3]
 | |
|         elif p[2] == '/':
 | |
|             p[0] = p[1] / p[3]
 | |
|         elif p[2] == '**':
 | |
|             p[0] = p[1] ** p[3]
 | |
| 
 | |
|     def p_expression_uminus(self, p):
 | |
|         'expression : MINUS expression %prec UMINUS'
 | |
|         p[0] = -p[2]
 | |
| 
 | |
|     def p_expression_group(self, p):
 | |
|         'expression : LPAREN expression RPAREN'
 | |
|         p[0] = p[2]
 | |
| 
 | |
|     def p_expression_number(self, p):
 | |
|         'expression : NUMBER'
 | |
|         p[0] = p[1]
 | |
| 
 | |
|     def p_expression_name(self, p):
 | |
|         'expression : NAME'
 | |
|         try:
 | |
|             p[0] = self.names[p[1]]
 | |
|         except LookupError:
 | |
|             print("Undefined name '%s'" % p[1])
 | |
|             p[0] = 0
 | |
| 
 | |
|     def p_error(self, p):
 | |
|         if p:
 | |
|             print("Syntax error at '%s'" % p.value)
 | |
|         else:
 | |
|             print("Syntax error at EOF")
 | |
| 
 | |
| if __name__ == '__main__':
 | |
|     calc = Calc()
 | |
|     calc.run()
 |