mirror of
				https://github.com/python/cpython.git
				synced 2025-11-02 22:51:25 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			78 lines
		
	
	
	
		
			1.8 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			78 lines
		
	
	
	
		
			1.8 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
"""Parser for future statements
 | 
						|
 | 
						|
"""
 | 
						|
 | 
						|
from compiler import ast, walk
 | 
						|
 | 
						|
def is_future(stmt):
 | 
						|
    """Return true if statement is a well-formed future statement"""
 | 
						|
    if not isinstance(stmt, ast.From):
 | 
						|
        return 0
 | 
						|
    if stmt.modname == "__future__":
 | 
						|
        return 1
 | 
						|
    else:
 | 
						|
        return 0
 | 
						|
 | 
						|
class FutureParser:
 | 
						|
 | 
						|
    features = ("nested_scopes",)
 | 
						|
    
 | 
						|
    def __init__(self):
 | 
						|
        self.found = {} # set
 | 
						|
 | 
						|
    def visitModule(self, node):
 | 
						|
        if node.doc is None:
 | 
						|
            off = 0
 | 
						|
        else:
 | 
						|
            off = 1
 | 
						|
 | 
						|
        stmt = node.node
 | 
						|
        for s in stmt.nodes[off:]:
 | 
						|
            if not self.check_stmt(s):
 | 
						|
                break
 | 
						|
 | 
						|
    def check_stmt(self, stmt):
 | 
						|
        if is_future(stmt):
 | 
						|
            for name, asname in stmt.names:
 | 
						|
                if name in self.features:
 | 
						|
                    self.found[name] = 1
 | 
						|
                else:
 | 
						|
                    raise SyntaxError, \
 | 
						|
                          "future feature %s is not defined" % name
 | 
						|
            stmt.valid_future = 1
 | 
						|
            return 1
 | 
						|
        return 0
 | 
						|
 | 
						|
    def get_features(self):
 | 
						|
        """Return list of features enabled by future statements"""
 | 
						|
        return self.found.keys()
 | 
						|
 | 
						|
class BadFutureParser:
 | 
						|
    """Check for invalid future statements"""
 | 
						|
 | 
						|
    def visitFrom(self, node):
 | 
						|
        if hasattr(node, 'valid_future'):
 | 
						|
            return
 | 
						|
        if node.modname != "__future__":
 | 
						|
            return
 | 
						|
        raise SyntaxError, "invalid future statement"
 | 
						|
 | 
						|
def find_futures(node):
 | 
						|
    p1 = FutureParser()
 | 
						|
    p2 = BadFutureParser()
 | 
						|
    walk(node, p1)
 | 
						|
    walk(node, p2)
 | 
						|
    return p1.get_features()
 | 
						|
 | 
						|
if __name__ == "__main__":
 | 
						|
    import sys
 | 
						|
    from compiler import parseFile, walk
 | 
						|
 | 
						|
    for file in sys.argv[1:]:
 | 
						|
        print file
 | 
						|
        tree = parseFile(file)
 | 
						|
        v = FutureParser()
 | 
						|
        walk(tree, v)
 | 
						|
        print v.found
 | 
						|
        print
 | 
						|
 |