// Copyright (C) 2014 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.googlesource.gerrit.plugins.uploadvalidator;

import com.google.gerrit.extensions.annotations.PluginName;
import com.google.gerrit.server.config.PluginConfig;
import com.google.gerrit.server.config.PluginConfigFactory;
import com.google.gerrit.server.events.CommitReceivedEvent;
import com.google.gerrit.server.git.GitRepositoryManager;
import com.google.gerrit.server.git.validators.CommitValidationException;
import com.google.gerrit.server.git.validators.CommitValidationMessage;
import com.google.gerrit.server.project.NoSuchProjectException;
import com.google.inject.Inject;

import org.eclipse.jgit.api.errors.GitAPIException;
import org.eclipse.jgit.lib.Repository;

import java.io.IOException;
import java.util.Collections;
import java.util.LinkedList;
import java.util.List;

public class MaxPathLengthValidator extends PathValidator {
  public static String KEY_MAX_PATH_LENGTH = "maxPathLength";

  private final String pluginName;
  private final PluginConfigFactory cfgFactory;
  private final GitRepositoryManager repoManager;

  @Inject
  MaxPathLengthValidator(@PluginName String pluginName,
      PluginConfigFactory cfgFactory, GitRepositoryManager repoManager) {
    this.pluginName = pluginName;
    this.cfgFactory = cfgFactory;
    this.repoManager = repoManager;
  }

  @Override
  public List<CommitValidationMessage> onCommitReceived(
      CommitReceivedEvent receiveEvent) throws CommitValidationException {
    try {
      PluginConfig cfg =
          cfgFactory.getFromProjectConfig(
              receiveEvent.project.getNameKey(), pluginName);
      int maxPathLength = cfg.getInt(KEY_MAX_PATH_LENGTH, 0);
      if (maxPathLength > 0) {
        Repository repo = repoManager.openRepository(receiveEvent.project.getNameKey());
        try {
          List<CommitValidationMessage> messages = new LinkedList<>();
          List<String> files = getFiles(repo, receiveEvent.commit);
          for (String file : files) {
            if (file.length() > maxPathLength) {
              messages.add(new CommitValidationMessage("path too long: " + file, true));
            }
          }
          if (!messages.isEmpty()) {
            throw new CommitValidationException(
                "contains files with too long paths (max path length: "
                    + maxPathLength + ")", messages);
          }
        } finally {
          repo.close();
        }
      }
    } catch (NoSuchProjectException | IOException | GitAPIException e) {
      throw new CommitValidationException("failed to check for max file path length", e);
    }

    return Collections.emptyList();
  }
}
