| // 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(); | 
 | } |