All Downloads are FREE. Search and download functionalities are using the official Maven repository.

src.org.python.indexer.ast.NQname Maven / Gradle / Ivy

There is a newer version: 2.7.1.1
Show newest version
/**
 * Copyright 2009, Google Inc.  All rights reserved.
 * Licensed to PSF under a Contributor Agreement.
 */
package org.python.indexer.ast;

import org.python.indexer.Indexer;
import org.python.indexer.NBinding;
import org.python.indexer.Scope;
import org.python.indexer.Util;
import org.python.indexer.types.NType;
import org.python.indexer.types.NModuleType;
import org.python.indexer.types.NUnknownType;

import java.io.File;

/**
 * Recursive doubly-linked list representation of a qualified module name,
 * either absolute or relative.  Similar to {@link NAttribute}, but handles
 * leading dots and other import-specific special cases.

* * Qualified names with leading dots are given {@code NQname} elements for each * leading dot. Dots separating simple names are not given {@code NQname} * elements.

*/ public class NQname extends NNode { static final long serialVersionUID = -5892553606852895686L; private NQname next; private NName name; public NQname(NQname next, NName name) { this(next, name, 0, 1); } public NQname(NQname next, NName name, int start, int end) { super(start, end); if (name == null) throw new IllegalArgumentException("null name"); this.name = name; this.next = next; addChildren(name, next); } /** * Returns this component of the qname chain. */ public NName getName() { return name; } /** * Returns the previous component of this qname chain, or {@code null} if * this is the first component. */ public NQname getPrevious() { NNode parent = getParent(); if (parent instanceof NQname) { return (NQname)parent; } return null; } /** * Returns the next component of the chain, or {@code null} if this is * the last component. */ public NQname getNext() { return next; } /** * Returns the last/bottom component of the chain. */ public NQname getBottom() { return next == null ? this : next.getBottom(); } /** * Returns {@code true} if this is the first/top component of the chain, * or if the name is unqualified (i.e. has only one component, no dots). */ public boolean isTop() { return getPrevious() == null; } /** * Returns {@code true} if this is the last/bottom component of the chain, * or if the name is unqualified (i.e. has only one component, no dots). */ public boolean isBottom() { return next == null; } /** * Returns {@code true} if this qname represents a simple, non-dotted module * name such as "os", "random" or "foo". */ public boolean isUnqualified() { return isTop() && isBottom(); } /** * Joins all components in this qname chain, beginning with the * current component. */ public String toQname() { return isBottom() ? name.id : name.id + "." + next.toQname(); } /** * Returns the qname down to (and including) this component, ending * with this component's name. For instance, if this {@code NQname} * instance represents the {@code foo} in {@code org.foo.bar}, this * method will return {@code org.foo}. */ public String thisQname() { NQname n = getTop(); StringBuilder sb = new StringBuilder(); sb.append(n.name.id); while (n != this) { sb.append("."); n = n.next; sb.append(n.name.id); } return sb.toString(); } /** * Returns the top (first) component in the chain. */ public NQname getTop() { return isTop() ? this : getPrevious().getTop(); } /** * Returns {@code true} if this qname component is a leading dot. */ public boolean isDot() { return ".".equals(name.id); } /** * Resolves and loads the module named by this qname. * @return the module represented by the qname up to this point. */ @Override public NType resolve(Scope s) throws Exception { setType(name.setType(new NUnknownType())); // Check for top-level native or standard module. if (isUnqualified()) { NModuleType mt = Indexer.idx.loadModule(name.id); if (mt != null) { return setType(name.setType(mt)); } } else { // Check for second-level builtin such as "os.path". NModuleType mt = Indexer.idx.getBuiltinModule(thisQname()); if (mt != null) { setType(name.setType(mt)); resolveExpr(next, s); return mt; } } return resolveInFilesystem(s); } private NType resolveInFilesystem(Scope s) throws Exception { NModuleType start = getStartModule(s); if (start == null) { reportUnresolvedModule(); return getType(); } String qname = start.getTable().getPath(); String relQname; if (isDot()) { relQname = Util.getQnameParent(qname); } else if (!isTop()) { relQname = qname + "." + name.id; } else { // top name: first look in current dir, then sys.path String dirQname = isInitPy() ? qname : Util.getQnameParent(qname); relQname = dirQname + "." + name.id; if (Indexer.idx.loadModule(relQname) == null) { relQname = name.id; } } NModuleType mod = Indexer.idx.loadModule(relQname); if (mod == null) { reportUnresolvedModule(); return getType(); } setType(name.setType(mod)); if (!isTop() && mod.getFile() != null) { Scope parentPkg = getPrevious().getTable(); NBinding mb = Indexer.idx.moduleTable.lookup(mod.getFile()); parentPkg.put(name.id, mb); } resolveExpr(next, s); return getType(); } private boolean isInitPy() { String path = getFile(); if (path == null) { return false; } return new File(path).getName().equals("__init__.py"); } private NModuleType getStartModule(Scope s) throws Exception { if (!isTop()) { return getPrevious().getType().asModuleType(); } // Start with module for current file (i.e. containing directory). NModuleType start = null; Scope mtable = s.getSymtabOfType(Scope.Type.MODULE); if (mtable != null) { start = Indexer.idx.loadModule(mtable.getPath()); if (start != null) { return start; } } String dir = new File(getFile()).getParent(); if (dir == null) { Indexer.idx.warn("Unable to find parent dir for " + getFile()); return null; } return Indexer.idx.loadModule(dir); } private void reportUnresolvedModule() { addError("module not found: " + name.id); Indexer.idx.recordUnresolvedModule(thisQname(), getFile()); } @Override public String toString() { return ""; } @Override public void visit(NNodeVisitor v) { if (v.visit(this)) { visitNode(next, v); visitNode(name, v); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy