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

com.google.javascript.jscomp.disambiguate.ColorFindPropertyReferences Maven / Gradle / Ivy

Go to download

Closure Compiler is a JavaScript optimizing compiler. It parses your JavaScript, analyzes it, removes dead code and rewrites and minimizes what's left. It also checks syntax, variable references, and types, and warns about common JavaScript pitfalls. It is used in many of Google's JavaScript apps, including Gmail, Google Web Search, Google Maps, and Google Docs.

There is a newer version: v20240317
Show newest version
/*
 * 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.disambiguate;

import static com.google.common.base.Preconditions.checkState;

import com.google.common.collect.ImmutableSet;
import com.google.javascript.jscomp.NodeTraversal;
import com.google.javascript.jscomp.NodeTraversal.AbstractPostOrderCallback;
import com.google.javascript.jscomp.NodeUtil;
import com.google.javascript.jscomp.colors.Color;
import com.google.javascript.jscomp.colors.StandardColors;
import com.google.javascript.rhino.Node;
import com.google.javascript.rhino.jstype.JSType;
import java.util.LinkedHashMap;
import java.util.function.Function;
import org.jspecify.nullness.Nullable;

/**
 * Traverses the AST, collecting connections between {@link JSType}s, property accesses, and their
 * accociated {@link Node}s.
 *
 * 

Also collects declarations of constructors even if no property accesses are visible. * *

This callback is intended for both source and externs. */ final class ColorFindPropertyReferences extends AbstractPostOrderCallback { /** * Tests whether the named JS function is a "property reflector"; a function that treats a string * literal as a property name. * *

Such a function is assumed to have the following signature: * *

    *
  1. A string literal that is the name being reflected. *
  2. An object on which the property is referenced. (Optional) *
  3. Var args. *
*/ @FunctionalInterface interface IsPropertyReflector { boolean test(Node nameNode); } private @Nullable LinkedHashMap propIndex = new LinkedHashMap<>(); private final ColorGraphNodeFactory colorGraphNodeFactory; private final IsPropertyReflector isPropertyReflector; ColorFindPropertyReferences( ColorGraphNodeFactory colorGraphNodeFactory, IsPropertyReflector isPropertyReflector) { this.colorGraphNodeFactory = colorGraphNodeFactory; this.isPropertyReflector = isPropertyReflector; } LinkedHashMap getPropertyIndex() { LinkedHashMap tmp = this.propIndex; this.propIndex = null; return tmp; } @Override public void visit(NodeTraversal t, Node n, Node parent) { switch (n.getToken()) { case GETPROP: case OPTCHAIN_GETPROP: this.registerPropertyUse(n, n.getFirstChild().getColor()); break; case OBJECTLIT: this.handleObjectLit(n); break; case CALL: this.handleCall(n); break; case CLASS: this.handleClass(n); break; case OBJECT_PATTERN: this.handleObjectPattern(n); break; case FUNCTION: this.handleFunction(n); break; default: break; } } private void handleObjectLit(Node n) { // Object.defineProperties literals are handled at the CALL node. if (n.getParent().isCall() && NodeUtil.isObjectDefinePropertiesDefinition(n.getParent())) { return; } Color owner = n.getColor(); this.traverseObjectlitLike(n, (m) -> owner); } /** Examines calls in case they are Object.defineProperties calls */ private void handleCall(Node call) { Node target = call.getFirstChild(); if (!target.isQualifiedName()) { return; } if (this.isPropertyReflector.test(target)) { this.handlePropertyReflectorCall(call); } else if (NodeUtil.isObjectDefinePropertiesDefinition(call)) { this.handleObjectDefineProperties(call); } } private void handleClass(Node classNode) { Color classType = classNode.getColor(); this.traverseObjectlitLike( NodeUtil.getClassMembers(classNode), (m) -> { if (m.isStaticMember()) { return classType; } else if (m.isMemberFieldDef()) { ImmutableSet classInstanceType = classType.getInstanceColors(); return classInstanceType.isEmpty() ? StandardColors.UNKNOWN : Color.createUnion(classInstanceType); } else { checkState(m.isMemberFunctionDef() || m.isGetterDef() || m.isSetterDef(), m); ImmutableSet classPrototypeType = classType.getPrototypes(); return classPrototypeType.isEmpty() ? StandardColors.UNKNOWN : Color.createUnion(classPrototypeType); } }); colorGraphNodeFactory.createNode(classType); } private void handleFunction(Node fnNode) { // Ensure the flattener knows about the class constructor and instance. Even if we don't see any // direct property references off either type, it's possible that a property is referenced // via a superclass or implemented interface. // e.g. `const /** !FooInterface */ x = new Foo(); x.method();` Color fnType = fnNode.getColor(); if (fnType != null && !fnType.getInstanceColors().isEmpty()) { this.colorGraphNodeFactory.createNode(fnType); } } private void handleObjectPattern(Node pattern) { Color owner = pattern.getColor(); this.traverseObjectlitLike(pattern, (m) -> owner); } private void handlePropertyReflectorCall(Node call) { Node name = call.getSecondChild(); if (name == null || !name.isStringLit()) { return; } Node obj = name.getNext(); Color objColor = (obj == null) ? null : obj.getColor(); this.registerPropertyUse(name, objColor); } private void handleObjectDefineProperties(Node call) { Node typeObj = call.getSecondChild(); Node objectLiteral = typeObj.getNext(); if (!objectLiteral.isObjectLit()) { return; } Color type = typeObj.getColor(); this.traverseObjectlitLike(objectLiteral, (m) -> type); } private void traverseObjectlitLike(Node n, Function memberOwnerFn) { // The keys in an object pattern are r-values, not l-values, but they are still accesses. checkState(n.isObjectLit() || n.isObjectPattern() || n.isClassMembers()); for (Node child = n.getFirstChild(); child != null; child = child.getNext()) { switch (child.getToken()) { case BLOCK: case COMPUTED_PROP: case COMPUTED_FIELD_DEF: case OBJECT_REST: case OBJECT_SPREAD: continue; case STRING_KEY: case MEMBER_FUNCTION_DEF: case MEMBER_FIELD_DEF: case GETTER_DEF: case SETTER_DEF: if (child.isQuotedStringKey()) { continue; // These won't be renamed due to our assumptions. Ignore them. } this.registerPropertyUse(child, memberOwnerFn.apply(child)); break; default: throw new IllegalStateException( "Unexpected child of " + n.getToken() + ": " + child.toStringTree()); } } } /** * Update all datastructures as necessary to consider property use {@code site} from type {@code * owner}. */ private void registerPropertyUse(Node site, Color owner) { PropertyClustering prop = this.propIndex.computeIfAbsent(site.getString(), PropertyClustering::new); ColorGraphNode flatOwner = this.colorGraphNodeFactory.createNode(owner); // Set the initial condition for flowing this property along the graph. flatOwner.getAssociatedProps().put(prop, ColorGraphNode.PropAssociation.AST); // Make sure there's a cluster for this name/type combination. prop.getClusters().add(flatOwner); // Record the site to rename once clusters are found. If it's an extern, we won't rename anyway. prop.getUseSites().put(site, flatOwner); // Track the cluster of types whose properties must keep their original name after // disambiguation. Note: an "enum type" is the type of an enum object like "{STOP: 0, GO: 1}". if (site.isFromExterns() || (owner != null && owner.getPropertiesKeepOriginalName())) { prop.registerOriginalNameType(flatOwner); } } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy