// 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.lib.skyframe;

import com.google.common.collect.ImmutableSet;
import com.google.devtools.build.lib.cmdline.PackageIdentifier;
import com.google.devtools.build.lib.packages.Globber;
import com.google.devtools.build.lib.packages.Globber.Operation;
import com.google.devtools.build.lib.skyframe.serialization.VisibleForSerialization;
import com.google.devtools.build.lib.skyframe.serialization.autocodec.AutoCodec;
import com.google.devtools.build.lib.vfs.PathFragment;
import com.google.devtools.build.lib.vfs.Root;
import com.google.devtools.build.lib.vfs.UnixGlob;
import com.google.devtools.build.skyframe.SkyFunctionName;
import com.google.devtools.build.skyframe.SkyKey;
import com.google.devtools.build.skyframe.SkyValue;
import java.util.Objects;

/** {@link SkyValue} corresponding to the computation result of the {@link GlobsFunction}. */
public class GlobsValue implements SkyValue {

  private final ImmutableSet<PathFragment> matches;

  public GlobsValue(ImmutableSet<PathFragment> matches) {
    this.matches = matches;
  }

  public ImmutableSet<PathFragment> getMatches() {
    return matches;
  }

  @Override
  public boolean equals(Object other) {
    if (other == this) {
      return true;
    }
    if (!(other instanceof GlobsValue)) {
      return false;
    }

    return getMatches().equals(((GlobsValue) other).getMatches());
  }

  @Override
  public int hashCode() {
    return matches.hashCode();
  }

  /**
   * Representation of individual glob inside a package, including its expression and Globber
   * operation type.
   */
  public static class GlobRequest {

    private final String pattern;
    private final Globber.Operation globOperation;

    public String getPattern() {
      return pattern;
    }

    public Operation getGlobOeration() {
      return globOperation;
    }

    private GlobRequest(String pattern, Globber.Operation globOperation) {
      this.pattern = pattern;
      this.globOperation = globOperation;
    }

    @Override
    public String toString() {
      return String.format("GlobRequest: %s %s", pattern, globOperation);
    }

    @Override
    public boolean equals(Object obj) {
      if (this == obj) {
        return true;
      }
      if (!(obj instanceof GlobRequest)) {
        return false;
      }

      GlobRequest other = (GlobRequest) obj;
      return pattern.equals(other.pattern) && globOperation.equals(other.globOperation);
    }

    @Override
    public int hashCode() {
      return Objects.hash(pattern, globOperation);
    }

    /**
     * Creates {@link GlobRequest} object iff pattern is a valid glob expression.
     *
     * <p>@throws InvalidGlobPatternException if the pattern is not valid.
     */
    public static GlobRequest create(String pattern, Globber.Operation globOeration)
        throws InvalidGlobPatternException {
      if (pattern.indexOf('?') != -1) {
        throw new InvalidGlobPatternException(pattern, "wildcard ? forbidden");
      }

      String error = UnixGlob.checkPatternForError(pattern);
      if (error != null) {
        throw new InvalidGlobPatternException(pattern, error);
      }
      return new GlobRequest(pattern, globOeration);
    }
  }

  /**
   * Returns the interned {@link GlobsValue.Key} object which contains all glob deps of a package.
   *
   * @param packageIdentifier packageId the name of the owner package (must be an existing package)
   * @param packageRoot the package root of {@code packageId}
   * @param globRequests container of all glob expressions and types of Globber operations, all
   *     input glob expressions are expected to be valid.
   */
  public static Key key(
      PackageIdentifier packageIdentifier,
      Root packageRoot,
      ImmutableSet<GlobRequest> globRequests) {
    return Key.create(packageIdentifier, packageRoot, globRequests);
  }

  /**
   * {@link SkyKey} type for {@link GlobsValue}, serving as the input to {@link GlobsFunction}.
   *
   * <p>Expects all glob expressions inside {@link Key#globRequests} are valid, as indicated by
   * {@code UnixGlob#checkPatternForError}.
   */
  @VisibleForSerialization
  @AutoCodec
  public static class Key implements SkyKey {
    private static final SkyKeyInterner<Key> interner = SkyKey.newInterner();

    private final PackageIdentifier packageIdentifier;
    private final Root packageRoot;
    private final ImmutableSet<GlobRequest> globRequests;

    private static Key create(
        PackageIdentifier packageIdentifier,
        Root packageRoot,
        ImmutableSet<GlobRequest> globRequests) {
      return interner.intern(new Key(packageIdentifier, packageRoot, globRequests));
    }

    @VisibleForSerialization
    @AutoCodec.Interner
    static Key intern(Key key) {
      return interner.intern(key);
    }

    private Key(
        PackageIdentifier packageIdentifier,
        Root packageRoot,
        ImmutableSet<GlobRequest> globRequests) {
      this.packageIdentifier = packageIdentifier;
      this.packageRoot = packageRoot;
      this.globRequests = globRequests;
    }

    /**
     * Returns the package that "owns" all globs.
     *
     * <p>The globs evaluation code ensures that the boundaries of this package are not crossed.
     */
    public PackageIdentifier getPackageIdentifier() {
      return packageIdentifier;
    }

    /** Returns the package root of {@link #packageIdentifier}. */
    public Root getPackageRoot() {
      return packageRoot;
    }

    /**
     * Returns an {@link ImmutableSet} containing all globs inside the package, including each glob
     * expression and operation.
     */
    public ImmutableSet<GlobRequest> getGlobRequests() {
      return globRequests;
    }

    @Override
    public SkyFunctionName functionName() {
      return SkyFunctions.GLOBS;
    }

    @Override
    public boolean equals(Object obj) {
      if (this == obj) {
        return true;
      }
      if (!(obj instanceof Key)) {
        return false;
      }
      Key other = (Key) obj;
      return packageIdentifier.equals(other.packageIdentifier)
          && packageRoot.equals(other.packageRoot)
          && globRequests.equals(other.globRequests);
    }

    @Override
    public int hashCode() {
      return Objects.hash(packageIdentifier, packageRoot, globRequests);
    }

    @Override
    public String toString() {
      StringBuilder stringBuilder = new StringBuilder();
      stringBuilder.append(
          String.format(
              "<GlobsKey packageRoot = %s, packageIdentifier = %s, globRequests = %s",
              packageRoot, packageIdentifier, globRequests));
      return stringBuilder.toString();
    }

    @Override
    public SkyKeyInterner<Key> getSkyKeyInterner() {
      return interner;
    }
  }
}
