CollectFunctionNames.java

/*
 * Copyright 2008 The Closure Compiler Authors.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package com.google.javascript.jscomp;

import com.google.javascript.jscomp.NodeTraversal.AbstractPostOrderCallback;
import com.google.javascript.rhino.Node;

import com.google.javascript.rhino.Token;
import java.io.Serializable;
import java.util.LinkedHashMap;
import java.util.Map;

/**
 * Extract a list of all function nodes defined in a JavaScript
 * program, assigns them globally unique ids and computes their fully
 * qualified names.  Function names are derived from the property they
 * are assigned to and the scope they are defined in, and are meant to be
 * human-readable rather than valid Javascript identifiers.  For instance,
 * the following code
 *
 * goog.widget = function(str) {
 *   this.member_fn = function() {}
 *   local_fn = function() {}
 *   goog.array.map(arr, function(){});
 * }
 *
 * defines the following functions
 *
 *  goog.widget
 *  goog.widget.member_fn
 *  goog.widget::local_fn
 *  goog.widget::<anonymous>
 *
 */
class CollectFunctionNames extends AbstractPostOrderCallback implements CompilerPass {

  private static class FunctionNamesMap implements FunctionNames {
    private int nextId = 0;

    private final Map<Node, FunctionRecord> functionMap = new LinkedHashMap<>();

    @Override
    public Iterable<Node> getFunctionNodeList() {
      return functionMap.keySet();
    }

    @Override
    public int getFunctionId(Node f) {
      FunctionRecord record = functionMap.get(f);
      if (record != null) {
        return record.id;
      } else {
        return -1;
      }
    }

    @Override
    public String getFunctionName(Node f) {
      FunctionRecord record = functionMap.get(f);
      if (record == null) {
        // Function node was added during compilation and has no name.
        return null;
      }

      String str = record.name;
      if (str.isEmpty()) {
        str = "<anonymous>";
      }

      Node parent = record.parent;
      if (parent != null) {
        str = getFunctionName(parent) + "::" + str;
      }

      // this.foo -> foo
      str = str.replace("::this.", ".");
      // foo.prototype.bar -> foo.bar
      // AnonymousFunctionNamingCallback already replaces ".prototype."
      // with "..", just remove the extra dot.
      str = str.replace("..", ".");
      // remove toplevel anonymous blocks, if they exists.
      str = str.replaceFirst("^(<anonymous>::)*", "");
      return str;
    }

    private void put(Node n, Node enclosingFunction, String functionName) {
      functionMap.put(n, new FunctionRecord(nextId++, enclosingFunction, functionName));
    }

    private void setFunctionName(String name, Node fnNode) {
      FunctionRecord record = functionMap.get(fnNode);
      assert(record != null);
      assert(record.name.isEmpty());
      record.name = name;
    }
  }

  private static class FunctionRecord implements Serializable {
    public final int id;
    public final Node parent;
    public String name;

    FunctionRecord(int id, Node parent, String name) {
      this.id = id;
      this.parent = parent;
      this.name = name;
    }
  }

  private final transient AbstractCompiler compiler;
  private final FunctionNamesMap functionNames = new FunctionNamesMap();
  private static final char DELIMITER = '.';
  private final NodeNameExtractor extractor = new NodeNameExtractor(DELIMITER);

  CollectFunctionNames(AbstractCompiler compiler) {
    this.compiler = compiler;
  }

  @Override
  public void process(Node externs, Node root) {
    NodeTraversal.traverseEs6(compiler, root, this);
  }

  public FunctionNames getFunctionNames() {
    return functionNames;
  }

  @Override
  public void visit(NodeTraversal t, Node n, Node parent) {
    switch (n.getToken()) {
      case FUNCTION:
        Node functionNameNode = n.getFirstChild();
        String functionName = functionNameNode.getString();
        if (functionName.isEmpty()) {
          if (parent.isAssign()) {
            Node lhs = parent.getFirstChild();
            functionName = extractor.getName(lhs);
          } else if (parent.isName()) {
            functionName = extractor.getName(parent);
          }
        }

        Node enclosingFunction = t.getEnclosingFunction();

        // Here functionName may still be empty. We handle certain cases of empty function names in
        // collectObjectLiteralnames and collectClassMethodsNames. Other functions remain unnamed.
        // (See unit tests for examples).
        functionNames.put(n, enclosingFunction, functionName);
        break;
      case ASSIGN:
        Node lhs = n.getFirstChild();
        Node rhs = lhs.getNext();
        // We handle object literal methods starting from ASSIGN nodes instead of OBJECT_LIT
        // to avoid revisiting nested object literals.
        if (rhs.isObjectLit()) {
          collectObjectLiteralMethodsNames(rhs, extractor.getName(lhs));
        }
        break;
      case CLASS:
        collectClassMethodsNames(n, extractor.getName(n));
        break;
      default:
        break;
    }
  }

  /**
   * Sets names in the functionNames map for unnamed functions inside object literals,
   * and recursively visits nested object literals.
   * @param objectLiteral The object literal node to visit.
   * @param context Represents the qualified name "so far"
   */
  private void collectObjectLiteralMethodsNames(Node objectLiteral, String context) {
    for (Node keyNode = objectLiteral.getFirstChild();
        keyNode != null;
        keyNode = keyNode.getNext()) {
      Node valueNode = keyNode.getFirstChild();

      // Object literal keys may be STRING_KEY, GETTER_DEF, SETTER_DEF,
      // MEMBER_FUNCTION_DEF (Shorthand function definition) or COMPUTED_PROP.
      // We currently skip Get, Set and CompProp keys.
      // TODO(lharker): Find a way to name Get, Set, and CompProp keys.
      if (keyNode.isStringKey() || keyNode.isMemberFunctionDef()) {
        // concatenate the context and key name to get a new qualified name.
        String name = combineNames(context, extractor.getName(keyNode));

        Token type = valueNode.getToken();
        if (type == Token.FUNCTION) {
          Node functionNameNode = valueNode.getFirstChild();
          String functionName = functionNameNode.getString();
          if (functionName.isEmpty()) {
            functionNames.setFunctionName(name, valueNode);
          }
        } else if (type == Token.OBJECTLIT) {
          // process nested object literal
          collectObjectLiteralMethodsNames(valueNode, name);
        }
      }
    }
  }

  /**
   * Collects the names of class methods, which require special handling because
   * method names are stored differently in the AST than regular function expression/declaration
   * names. For example,
   * class A {
   *   method() {}
   * }
   * will set the name of "method" to be "A.method".
   */
  private void collectClassMethodsNames(Node classNode, String className) {
    Node classMembersNode = classNode.getLastChild();

    for (Node methodNode : classMembersNode.children()) {
      if (methodNode.isMemberFunctionDef()) {
        Node functionNode = methodNode.getFirstChild();
        String name = combineNames(className, extractor.getName(methodNode));
        functionNames.setFunctionName(name, functionNode);
      }
    }
  }

  private String combineNames(String lhs, String rhs) {
    return lhs + DELIMITER + rhs;
  }
}