/*
* Copyright 2000-2010 JetBrains s.r.o.
* 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 org.community.intellij.plugins.communitycase.history.wholeTree;
import com.intellij.openapi.diagnostic.Logger;
import com.intellij.openapi.progress.BackgroundTaskQueue;
import com.intellij.openapi.progress.ProgressIndicator;
import com.intellij.openapi.progress.Task;
import com.intellij.openapi.project.Project;
import com.intellij.openapi.vcs.VcsException;
import com.intellij.openapi.vcs.changes.BackgroundFromStartOption;
import com.intellij.openapi.vfs.VirtualFile;
import com.intellij.util.containers.HashMap;
import com.intellij.util.containers.MultiMap;
import org.community.intellij.plugins.communitycase.history.browser.LowLevelAccess;
import org.community.intellij.plugins.communitycase.history.browser.LowLevelAccessImpl;
import org.community.intellij.plugins.communitycase.history.browser.SymbolicRefs;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;
import java.util.ArrayList;
import java.util.Collection;
import java.util.List;
import java.util.Map;
/**
* @author irengrig
*/
public class DetailsLoaderImpl implements DetailsLoader {
private final static Logger LOG = Logger.getInstance("#"+DetailsLoaderImpl.class.getName());
private final static int ourLoadSize = 20;
private final static long ourRefsReload = 120000;
private final BackgroundTaskQueue myQueue;
private final Map<VirtualFile, CommitIdsHolder<AbstractHash>> myLoadIdsGatherer;
private final Map<VirtualFile, LowLevelAccess> myAccesses;
private DetailsCache myDetailsCache;
private final Project myProject;
private final Map<VirtualFile, SymbolicRefs> myRefs;
private long myRefsReloadTick;
private RefsLoader myRefsLoader;
public DetailsLoaderImpl(Project project) {
myQueue = new BackgroundTaskQueue(project, " log details");
myProject = project;
myLoadIdsGatherer = new HashMap<VirtualFile, CommitIdsHolder<AbstractHash>>();
myAccesses = new HashMap<VirtualFile, LowLevelAccess>();
myRefs = new HashMap<VirtualFile, SymbolicRefs>();
myRefsReloadTick = 0;
scheduleRefs();
}
private void scheduleRefs() {
myRefsLoader = new RefsLoader(myProject);
myQueue.run(myRefsLoader);
}
public void setDetailsCache(DetailsCache detailsCache) {
myDetailsCache = detailsCache;
}
public void setRoots(final Collection<VirtualFile> roots) {
myLoadIdsGatherer.clear();
myAccesses.clear();
for (VirtualFile root : roots) {
myLoadIdsGatherer.put(root, new CommitIdsHolder<AbstractHash>());
myAccesses.put(root, new LowLevelAccessImpl(myProject, root));
}
myRefsReloadTick = 0;
}
@Override
public void load(MultiMap<VirtualFile,AbstractHash> hashes) {
final long now = System.currentTimeMillis();
if (now - myRefsReloadTick >= ourRefsReload) {
scheduleRefs();
}
myRefsReloadTick = now;
for (VirtualFile root : hashes.keySet()) {
final CommitIdsHolder<AbstractHash> holder = myLoadIdsGatherer.get(root);
holder.add(hashes.get(root));
myQueue.run(new Worker(myProject, root, myAccesses.get(root), myDetailsCache, myQueue));
}
}
private class RefsLoader extends Task.Backgroundable {
private RefsLoader(@Nullable Project project) {
super(project, "Reload symbolic references", false, BackgroundFromStartOption.getInstance());
}
@Override
public void run(@NotNull ProgressIndicator indicator) {
for (VirtualFile file : myAccesses.keySet()) {
final LowLevelAccess access = myAccesses.get(file);
try {
myRefs.put(file, access.getRefs());
}
catch (VcsException e) {
LOG.info(e);
}
}
}
}
private class Worker extends Task.Backgroundable {
private final LowLevelAccess myAccess;
private final DetailsCache myDetailsCache;
private final BackgroundTaskQueue myQueue;
private final VirtualFile myVirtualFile;
private Worker(@Nullable final Project project,
final VirtualFile virtualFile,
final LowLevelAccess access,
final DetailsCache detailsCache, final BackgroundTaskQueue queue) {
super(project, "Load commits details", false, BackgroundFromStartOption.getInstance());
myVirtualFile = virtualFile;
myAccess = access;
myDetailsCache = detailsCache;
myQueue = queue;
}
@Override
public void run(@NotNull ProgressIndicator indicator) {
final CommitIdsHolder<AbstractHash> holder = myLoadIdsGatherer.get(myVirtualFile);
if (holder == null) return;
final Collection<AbstractHash> hashes = holder.get(ourLoadSize);
final List<String> converted = new ArrayList<String>();
for (final AbstractHash hash : hashes) {
if (myDetailsCache.convert(myVirtualFile, hash) == null) {
converted.add(hash.getString());
}
}
if (! hashes.isEmpty()) {
try {
final Collection<org.community.intellij.plugins.communitycase.history.browser.Commit> result = myAccess.getCommitDetails(converted, myRefs.get(myAccess.getRoot()));
if (result != null && (! result.isEmpty())) {
myDetailsCache.acceptAnswer(result, myAccess.getRoot());
}
}
catch (VcsException e) {
LOG.info(e);
// suppress here. further : todo put fictive details!!!!!!!!!
// todo to dont repeat load infinitely
}
}
if (holder.haveData()) {
myQueue.run(this);
}
}
}
}