blob: b39b37111ccf79fbfaacc94810be33800caf79d5 [file] [log] [blame]
// Copyright (C) 2015 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.deleteproject.fs;
import com.google.gerrit.extensions.events.LifecycleListener;
import com.google.gerrit.server.config.GerritServerConfig;
import com.google.gerrit.server.config.SitePaths;
import com.google.inject.Inject;
import org.eclipse.jgit.lib.Config;
import org.eclipse.jgit.lib.RepositoryCache.FileKey;
import org.eclipse.jgit.util.FS;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.io.File;
import java.io.IOException;
import java.nio.file.FileVisitResult;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.SimpleFileVisitor;
import java.nio.file.attribute.BasicFileAttributes;
public class DeleteTrashFolders implements LifecycleListener {
private static final Logger log = LoggerFactory.getLogger(DeleteTrashFolders.class);
private File gitDir;
@Inject
public DeleteTrashFolders(SitePaths site, @GerritServerConfig Config cfg) {
gitDir = site.resolve(cfg.getString("gerrit", null, "basePath"));
}
class TrashFolderRemover extends SimpleFileVisitor<Path> {
@Override
public FileVisitResult preVisitDirectory(Path dir, BasicFileAttributes attrs)
throws IOException {
String fName = dir.getFileName().toString();
// Search for directories which end with a dot, 13 digits and the string
// ".deleted". A folder 'f' is renamed to 'f.<currentTimeMillis>.deleted'.
// <currentTimeMillis> happens to be exactly 13 digits for commits created
// between 2002 (before git was born) and 2285.
if (fName.endsWith(".deleted")
&& fName.matches(".*\\d{13}.deleted$")) {
log.warn("Will delete this folder: {}", dir);
recursiveDelete(dir);
return FileVisitResult.SKIP_SUBTREE;
} else if (FileKey.isGitRepository(dir.toFile(), FS.DETECTED)) {
// We are in a GITDIR and don't expect trash folders inside GITDIR's.
return FileVisitResult.SKIP_SUBTREE;
}
return super.preVisitDirectory(dir, attrs);
}
/**
* Recursively delete the specified file and all of its contents.
*
* @throws IOException
*/
private void recursiveDelete(Path file) throws IOException {
Files.walkFileTree(file, new SimpleFileVisitor<Path>() {
@Override
public FileVisitResult visitFile(Path file, BasicFileAttributes attrs)
throws IOException {
Files.delete(file);
return FileVisitResult.CONTINUE;
}
@Override
public FileVisitResult postVisitDirectory(Path dir, IOException e)
throws IOException {
if (e != null) {
throw e;
}
Files.delete(dir);
return FileVisitResult.CONTINUE;
}
});
}
}
@Override
public void start() {
new Thread(new Runnable() {
@Override
public void run() {
try {
Files.walkFileTree(gitDir.toPath(), new TrashFolderRemover());
} catch (IOException e) {
log.warn("Exception occured while trying to delete trash folders", e);
}
}
}, "DeleteTrashFolders").start();
}
@Override
public void stop() {
}
}