com.google.javascript.jscomp.disambiguate.UseSiteRenamer Maven / Gradle / Ivy
Show all versions of closure-compiler-unshaded Show documentation
/*
* Copyright 2019 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.collect.ImmutableMap.toImmutableMap;
import static java.util.function.Function.identity;
import com.google.common.collect.ImmutableMap;
import com.google.common.collect.ImmutableSet;
import com.google.common.collect.ImmutableSetMultimap;
import com.google.javascript.jscomp.JSError;
import com.google.javascript.rhino.Node;
import java.util.Map;
import java.util.Objects;
import java.util.function.Consumer;
/** Applies renaming to property use sites following cluster computation. */
final class UseSiteRenamer {
private static final String INVALIDATED_NAME_VALUE = "";
private final ImmutableSet propertiesThatMustDisambiguate;
private final Consumer errorCb;
private final Consumer mutationCb;
private final ImmutableSetMultimap.Builder renamingIndex =
ImmutableSetMultimap.builder();
UseSiteRenamer(
ImmutableSet propertiesThatMustDisambiguate,
Consumer errorCb,
Consumer mutationCb) {
this.propertiesThatMustDisambiguate = propertiesThatMustDisambiguate;
this.errorCb = errorCb;
this.mutationCb = mutationCb;
}
/**
* Renames all references to {@code prop}.
*
* If {@code prop} is invalid or should otherwise not be renamed, the AST will not be changed.
*/
void renameUses(PropertyClustering prop) {
if (prop.isInvalidated()) {
this.renamingIndex.put(prop.getName(), INVALIDATED_NAME_VALUE);
if (this.propertiesThatMustDisambiguate.contains(prop.getName())) {
this.errorCb.accept(createInvalidationError(prop.getName()));
}
return;
}
ImmutableMap clusterNames = createAllClusterNames(prop);
if (clusterNames.size() <= 1) {
/**
* Don't bother renaming clusters with a single element. Renaming won't actaully disambiguate
* anything in this case, so skip the work.
*/
this.renamingIndex.put(prop.getName(), prop.getName());
return;
}
this.renamingIndex.putAll(prop.getName(), clusterNames.values());
for (Map.Entry usage : prop.getUseSites().entrySet()) {
Node site = usage.getKey();
ColorGraphNode flatRep = prop.getClusters().find(usage.getValue());
String newName = clusterNames.get(flatRep);
if (!Objects.equals(newName, site.getString())) {
site.setString(newName);
this.mutationCb.accept(site);
}
}
}
ImmutableSetMultimap getRenamingIndex() {
return this.renamingIndex.build();
}
/**
* Creates a unique name for each cluster in {@code prop} and maps it to the cluster
* representative.
*/
private static ImmutableMap createAllClusterNames(
PropertyClustering prop) {
return prop.getClusters().allRepresentatives().stream()
.collect(toImmutableMap(identity(), (r) -> createClusterName(prop, r)));
}
private static String createClusterName(PropertyClustering prop, ColorGraphNode rep) {
if (Objects.equals(prop.getOriginalNameClusterRep(), rep)) {
return prop.getName();
}
return "JSC$" + rep.getIndex() + "_" + prop.getName();
}
private static JSError createInvalidationError(String name) {
return JSError.make(null, -1, -1, DisambiguateProperties.PROPERTY_INVALIDATION, name);
}
}