blob: bb4a8757b94c3056a0ac3a503e74a6ff0276ad2f [file] [log] [blame]
// Copyright 2023 The Bazel Authors. All rights reserved.
//
// 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.devtools.build.docgen.starlark;
import com.google.common.base.Preconditions;
import com.google.common.collect.HashMultimap;
import com.google.common.collect.ImmutableCollection;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableSortedSet;
import com.google.common.collect.Iterables;
import com.google.common.collect.Multimap;
import java.text.Collator;
import java.util.Collection;
import java.util.Comparator;
import java.util.Locale;
import java.util.TreeMap;
import javax.annotation.Nullable;
/**
* A typical Starlark documentation page, containing a bunch of field/method documentation entries.
*/
public abstract class StarlarkDocPage extends StarlarkDoc {
private final Multimap<String, StarlarkJavaMethodDoc> javaMethods;
private final TreeMap<String, StarlarkMethodDoc> methodMap;
@Nullable private StarlarkConstructorMethodDoc javaConstructor;
protected StarlarkDocPage(StarlarkDocExpander expander) {
super(expander);
this.methodMap = new TreeMap<>(Collator.getInstance(Locale.US));
this.javaMethods = HashMultimap.create();
}
public abstract String getTitle();
public void setConstructor(StarlarkConstructorMethodDoc method) {
Preconditions.checkState(
javaConstructor == null,
"Constructor method doc already set for %s:\n existing: %s\n attempted: %s",
getName(),
javaConstructor,
method);
javaConstructor = method;
}
public void addMethod(StarlarkJavaMethodDoc method) {
if (!method.documented()) {
return;
}
String shortName = method.getName();
Collection<StarlarkJavaMethodDoc> overloads = javaMethods.get(shortName);
if (!overloads.isEmpty()) {
method.setOverloaded(true);
// Overload information only needs to be updated if we're discovering the first overload
// (= the second method of the same name).
if (overloads.size() == 1) {
Iterables.getOnlyElement(overloads).setOverloaded(true);
}
}
javaMethods.put(shortName, method);
// If the method is overloaded, getName() now returns a longer,
// unique name including the names of the parameters.
StarlarkMethodDoc prev = methodMap.put(method.getName(), method);
if (prev != null && !prev.getMethod().equals(method.getMethod())) {
throw new IllegalStateException(
String.format(
"Starlark type '%s' has distinct overloads of %s: %s, %s",
getName(), method.getName(), method.getMethod(), prev.getMethod()));
}
}
public Collection<StarlarkMethodDoc> getJavaMethods() {
ImmutableSortedSet.Builder<StarlarkMethodDoc> methods =
new ImmutableSortedSet.Builder<>(Comparator.comparing(StarlarkMethodDoc::getName));
if (javaConstructor != null) {
methods.add(javaConstructor);
}
methods.addAll(javaMethods.values());
return methods.build();
}
public ImmutableCollection<? extends StarlarkMethodDoc> getMethods() {
ImmutableList.Builder<StarlarkMethodDoc> methods = ImmutableList.builder();
if (javaConstructor != null) {
methods.add(javaConstructor);
}
return methods.addAll(methodMap.values()).build();
}
@Nullable
public StarlarkConstructorMethodDoc getConstructor() {
return javaConstructor;
}
/** Returns the path to the source file backing this doc page. */
// This method may seem unused, but it's actually used in the template file (starlark-library.vm).
public abstract String getSourceFile();
}