// Copyright (C) 2017 The Android Open Source Project
//
// 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.gerrit.server.schema;

import static java.util.stream.Collectors.toList;

import com.google.common.primitives.Ints;
import com.google.gerrit.reviewdb.client.RefNames;
import com.google.gerrit.reviewdb.server.ReviewDb;
import com.google.gerrit.server.StarredChangesUtil;
import com.google.gerrit.server.StarredChangesUtil.IllegalLabelException;
import com.google.gerrit.server.StarredChangesUtil.StarRef;
import com.google.gerrit.server.config.AllUsersName;
import com.google.gerrit.server.git.GitRepositoryManager;
import com.google.gwtorm.server.OrmException;
import com.google.inject.Inject;
import com.google.inject.Provider;
import java.io.IOException;
import java.util.List;
import java.util.Objects;
import java.util.Set;
import org.eclipse.jgit.lib.BatchRefUpdate;
import org.eclipse.jgit.lib.ObjectId;
import org.eclipse.jgit.lib.Ref;
import org.eclipse.jgit.lib.Repository;
import org.eclipse.jgit.lib.TextProgressMonitor;
import org.eclipse.jgit.revwalk.RevWalk;
import org.eclipse.jgit.transport.ReceiveCommand;

public class Schema_161 extends ReviewDbSchemaVersion {
  private static final String MUTE_LABEL = "mute";

  private final GitRepositoryManager repoManager;
  private final AllUsersName allUsersName;

  @Inject
  Schema_161(
      Provider<Schema_160> prior, GitRepositoryManager repoManager, AllUsersName allUsersName) {
    super(prior);
    this.repoManager = repoManager;
    this.allUsersName = allUsersName;
  }

  @Override
  protected void migrateData(ReviewDb db, UpdateUI ui) throws OrmException {
    try (Repository git = repoManager.openRepository(allUsersName);
        RevWalk rw = new RevWalk(git)) {
      BatchRefUpdate bru = git.getRefDatabase().newBatchUpdate();
      bru.setAllowNonFastForwards(true);

      for (Ref ref : git.getRefDatabase().getRefsByPrefix(RefNames.REFS_STARRED_CHANGES)) {
        StarRef starRef = StarredChangesUtil.readLabels(git, ref.getName());

        Set<Integer> mutedPatchSets =
            StarredChangesUtil.getStarredPatchSets(starRef.labels(), MUTE_LABEL);
        if (mutedPatchSets.isEmpty()) {
          continue;
        }

        Set<Integer> reviewedPatchSets =
            StarredChangesUtil.getStarredPatchSets(
                starRef.labels(), StarredChangesUtil.REVIEWED_LABEL);
        Set<Integer> unreviewedPatchSets =
            StarredChangesUtil.getStarredPatchSets(
                starRef.labels(), StarredChangesUtil.UNREVIEWED_LABEL);

        List<String> newLabels =
            starRef
                .labels()
                .stream()
                .map(
                    l -> {
                      if (l.startsWith(MUTE_LABEL)) {
                        Integer mutedPatchSet = Ints.tryParse(l.substring(MUTE_LABEL.length() + 1));
                        if (mutedPatchSet == null) {
                          // unexpected format of mute label, must be a label that was manually
                          // set, just leave it alone
                          return l;
                        }
                        if (!reviewedPatchSets.contains(mutedPatchSet)
                            && !unreviewedPatchSets.contains(mutedPatchSet)) {
                          // convert mute label to reviewed label
                          return StarredChangesUtil.REVIEWED_LABEL + "/" + mutedPatchSet;
                        }
                        // else patch set is muted but has either reviewed or unreviewed label
                        // -> just drop the mute label
                        return null;
                      }
                      return l;
                    })
                .filter(Objects::nonNull)
                .collect(toList());

        ObjectId id = StarredChangesUtil.writeLabels(git, newLabels);
        bru.addCommand(new ReceiveCommand(ref.getTarget().getObjectId(), id, ref.getName()));
      }
      bru.execute(rw, new TextProgressMonitor());
    } catch (IOException | IllegalLabelException ex) {
      throw new OrmException(ex);
    }
  }
}
