blob: 24a5a5a7c9596e84b9908f26ff21bb0dc4d90790 [file] [log] [blame]
/*********************************************************************
* Copyright (c) 2010 Sony Ericsson/ST Ericsson and others.
* All rights reserved. This program and the accompanying materials
* are made available under the terms of the Eclipse Public License v1.0
* which accompanies this distribution, and is available at
* http://www.eclipse.org/legal/epl-v10.html
*
* Contributors:
* Sony Ericsson/ST Ericsson - initial API and implementation
* Tasktop Technologies - improvements
* GitHub, Inc. - fixes for bug 354753
* Sascha Scholz (SAP) - improvements
*********************************************************************/
package org.eclipse.mylyn.internal.gerrit.core;
import java.util.Date;
import java.util.List;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;
import java.util.logging.Level;
import java.util.logging.Logger;
import org.apache.commons.httpclient.HttpStatus;
import org.eclipse.core.runtime.CoreException;
import org.eclipse.core.runtime.IProgressMonitor;
import org.eclipse.core.runtime.IStatus;
import org.eclipse.core.runtime.Status;
import org.eclipse.mylyn.commons.core.StatusHandler;
import org.eclipse.mylyn.commons.net.Policy;
import org.eclipse.mylyn.internal.gerrit.core.client.GerritAuthenticationState;
import org.eclipse.mylyn.internal.gerrit.core.client.GerritClient;
import org.eclipse.mylyn.internal.gerrit.core.client.GerritConfiguration;
import org.eclipse.mylyn.internal.gerrit.core.client.GerritException;
import org.eclipse.mylyn.internal.gerrit.core.client.GerritHttpException;
import org.eclipse.mylyn.internal.gerrit.core.client.GerritLoginException;
import org.eclipse.mylyn.internal.gerrit.core.client.GerritSystemInfo;
import org.eclipse.mylyn.internal.gerrit.core.client.JSonSupport;
import org.eclipse.mylyn.tasks.core.AbstractRepositoryConnector;
import org.eclipse.mylyn.tasks.core.IRepositoryQuery;
import org.eclipse.mylyn.tasks.core.ITask;
import org.eclipse.mylyn.tasks.core.ITaskMapping;
import org.eclipse.mylyn.tasks.core.TaskRepository;
import org.eclipse.mylyn.tasks.core.TaskRepositoryLocationFactory;
import org.eclipse.mylyn.tasks.core.data.AbstractTaskDataHandler;
import org.eclipse.mylyn.tasks.core.data.TaskData;
import org.eclipse.mylyn.tasks.core.data.TaskDataCollector;
import org.eclipse.mylyn.tasks.core.data.TaskMapper;
import org.eclipse.mylyn.tasks.core.sync.ISynchronizationSession;
import org.eclipse.osgi.util.NLS;
import com.google.gerrit.common.data.ChangeInfo;
import com.google.gwtorm.client.KeyUtil;
import com.google.gwtorm.server.StandardKeyEncoder;
/**
* The Gerrit connector core.
*
* @author Mikael Kober
* @author Thomas Westling
* @author Sascha Scholz
*/
public class GerritConnector extends AbstractRepositoryConnector {
static Logger logger = Logger.getLogger("com.google.gson.ParameterizedTypeHandlerMap"); //$NON-NLS-1$
static {
KeyUtil.setEncoderImpl(new StandardKeyEncoder());
// disable logging of Overriding the existing type handler for class java.sql.Timestamp message
logger.setLevel(Level.OFF);
}
/**
* Prefix for task id in a task-url: http://[gerrit-repository]/#change,[task.id].
*/
public static final String CHANGE_PREFIX = "/#change,"; //$NON-NLS-1$
/**
* Connector kind
*/
public static final String CONNECTOR_KIND = "org.eclipse.mylyn.gerrit"; //$NON-NLS-1$
/**
* Label for the connector.
*/
public static final String CONNECTOR_LABEL = "Gerrit Code Review"; //$NON-NLS-1$
public static final String KEY_REPOSITORY_CONFIG = CONNECTOR_KIND + ".config"; //$NON-NLS-1$
public static final String KEY_REPOSITORY_AUTH = CONNECTOR_KIND + ".auth"; //$NON-NLS-1$
public static final String KEY_REPOSITORY_ACCOUNT_ID = CONNECTOR_KIND + ".accountId"; //$NON-NLS-1$
public static final String KEY_REPOSITORY_OPEN_ID_ENABLED = CONNECTOR_KIND + ".openId.enabled"; //$NON-NLS-1$
public static final String KEY_REPOSITORY_OPEN_ID_PROVIDER = CONNECTOR_KIND + ".openId.provider"; //$NON-NLS-1$
private final GerritTaskDataHandler taskDataHandler = new GerritTaskDataHandler(this);
private TaskRepositoryLocationFactory taskRepositoryLocationFactory = new TaskRepositoryLocationFactory();
private final ConcurrentMap<TaskRepository, GerritConfiguration> configurationCache = new ConcurrentHashMap<TaskRepository, GerritConfiguration>();
public GerritConnector() {
if (GerritCorePlugin.getDefault() != null) {
GerritCorePlugin.getDefault().setConnector(this);
}
}
/**
* Not supported, yet.
*/
@Override
public boolean canCreateNewTask(TaskRepository arg0) {
return false;
}
@Override
public boolean canCreateTaskFromKey(TaskRepository arg0) {
return true;
}
public GerritClient getClient(TaskRepository repository) {
return createClient(repository, true);
}
@Override
public String getConnectorKind() {
return CONNECTOR_KIND;
}
@Override
public String getLabel() {
return "Gerrit Code Review (supports 2.3 and later)";
}
@Override
public TaskData getTaskData(TaskRepository repository, String taskId, IProgressMonitor monitor)
throws CoreException {
return taskDataHandler.getTaskData(repository, taskId, monitor);
}
@Override
public AbstractTaskDataHandler getTaskDataHandler() {
return taskDataHandler;
}
@Override
public String getRepositoryUrlFromTaskUrl(String url) {
if (url == null) {
return null;
}
int i = url.indexOf(CHANGE_PREFIX);
if (i != -1) {
return url.substring(0, i);
}
return null;
}
@Override
public String getTaskIdFromTaskUrl(String url) {
if (url == null) {
return null;
}
// example: https://review.sonyericsson.net/#change,14175
int index = url.indexOf(CHANGE_PREFIX);
if (index > 0) {
return url.substring(index + CHANGE_PREFIX.length());
}
return null;
}
@Override
public ITaskMapping getTaskMapping(TaskData taskData) {
return new TaskMapper(taskData);
}
public synchronized TaskRepositoryLocationFactory getTaskRepositoryLocationFactory() {
return taskRepositoryLocationFactory;
}
@Override
public String getTaskUrl(String repositoryUrl, String taskId) {
return repositoryUrl + CHANGE_PREFIX + taskId;
}
@Override
public boolean hasTaskChanged(TaskRepository repository, ITask task, TaskData taskData) {
ITaskMapping taskMapping = getTaskMapping(taskData);
Date repositoryDate = taskMapping.getModificationDate();
Date localDate = task.getModificationDate();
if (repositoryDate != null && repositoryDate.equals(localDate)) {
return false;
}
return true;
}
@Override
public IStatus performQuery(TaskRepository repository, IRepositoryQuery query, TaskDataCollector resultCollector,
ISynchronizationSession session, IProgressMonitor monitor) {
try {
monitor.beginTask("Executing query", IProgressMonitor.UNKNOWN);
GerritClient client = getClient(repository);
client.refreshConfigOnce(monitor);
List<ChangeInfo> result = null;
if (GerritQuery.ALL_OPEN_CHANGES.equals(query.getAttribute(GerritQuery.TYPE))) {
result = client.queryAllReviews(monitor);
} else if (GerritQuery.CUSTOM.equals(query.getAttribute(GerritQuery.TYPE))) {
String queryString = query.getAttribute(GerritQuery.QUERY_STRING);
result = client.executeQuery(monitor, queryString);
} else if (GerritQuery.MY_CHANGES.equals(query.getAttribute(GerritQuery.TYPE))) {
result = client.queryMyReviews(monitor);
} else if (GerritQuery.MY_WATCHED_CHANGES.equals(query.getAttribute(GerritQuery.TYPE))) {
result = client.queryWatchedReviews(monitor);
} else if (GerritQuery.OPEN_CHANGES_BY_PROJECT.equals(query.getAttribute(GerritQuery.TYPE))) {
String project = query.getAttribute(GerritQuery.PROJECT);
result = client.queryByProject(monitor, project);
}
if (result != null) {
for (ChangeInfo changeInfo : result) {
TaskData taskData = taskDataHandler.createPartialTaskData(repository,
changeInfo.getId() + "", monitor); //$NON-NLS-1$
taskData.setPartial(true);
taskDataHandler.updateTaskData(repository, taskData, changeInfo);
resultCollector.accept(taskData);
}
return Status.OK_STATUS;
} else {
return new Status(IStatus.ERROR, GerritCorePlugin.PLUGIN_ID, NLS.bind("Unknows query type: {0}",
query.getAttribute(GerritQuery.PROJECT)));
}
} catch (UnsupportedClassVersionError e) {
return toStatus(repository, e);
} catch (GerritException e) {
return toStatus(repository, e);
} finally {
monitor.done();
}
}
public synchronized void setTaskRepositoryLocationFactory(
TaskRepositoryLocationFactory taskRepositoryLocationFactory) {
this.taskRepositoryLocationFactory = taskRepositoryLocationFactory;
}
@Override
public void updateRepositoryConfiguration(TaskRepository repository, IProgressMonitor monitor) throws CoreException {
try {
getClient(repository).refreshConfig(monitor);
} catch (GerritException e) {
throw toCoreException(repository, e);
}
}
@Override
public void updateTaskFromTaskData(TaskRepository taskRepository, ITask task, TaskData taskData) {
Date oldModificationDate = task.getModificationDate();
TaskMapper mapper = (TaskMapper) getTaskMapping(taskData);
mapper.applyTo(task);
String key = task.getTaskKey();
if (key != null) {
task.setSummary(NLS.bind("{0} [{1}]", mapper.getSummary(), key));
task.setTaskKey(task.getTaskId());
}
// retain modification date to force an update when full task data is received
if (taskData.isPartial()) {
task.setModificationDate(oldModificationDate);
}
}
public GerritSystemInfo validate(TaskRepository repository, IProgressMonitor monitor) throws CoreException {
try {
// only allow user prompting in case of Open ID authentication
if (!Boolean.parseBoolean(repository.getProperty(GerritConnector.KEY_REPOSITORY_OPEN_ID_ENABLED))) {
monitor = Policy.backgroundMonitorFor(monitor);
}
return createClient(repository, false).getInfo(monitor);
} catch (UnsupportedClassVersionError e) {
throw toCoreException(repository, e);
} catch (GerritException e) {
throw toCoreException(repository, e);
}
}
private GerritClient createClient(final TaskRepository repository, boolean cachedConfig) {
GerritConfiguration config = (cachedConfig) ? loadConfiguration(repository) : null;
GerritAuthenticationState authState = (cachedConfig)
? GerritClient.authStateFromString(repository.getProperty(KEY_REPOSITORY_AUTH))
: null;
return new GerritClient(taskRepositoryLocationFactory.createWebLocation(repository), config, authState) {
@Override
protected void configurationChanged(GerritConfiguration config) {
saveConfiguration(repository, config);
}
@Override
protected void authStateChanged(GerritAuthenticationState authState) {
repository.setProperty(KEY_REPOSITORY_AUTH, GerritClient.authStateToString(authState));
}
};
}
protected GerritConfiguration loadConfiguration(TaskRepository repository) {
GerritConfiguration configuration = configurationCache.get(repository);
if (configuration == null) {
configuration = configurationFromString(repository.getProperty(KEY_REPOSITORY_CONFIG));
if (configuration != null) {
configurationCache.put(repository, configuration);
}
}
return configuration;
}
protected void saveConfiguration(TaskRepository repository, GerritConfiguration configuration) {
configurationCache.put(repository, configuration);
repository.setProperty(KEY_REPOSITORY_CONFIG, configurationToString(configuration));
}
public GerritConfiguration getConfiguration(TaskRepository repository) {
GerritConfiguration configuration = configurationCache.get(repository);
if (configuration == null) {
configuration = loadConfiguration(repository);
}
return configuration;
}
private static GerritConfiguration configurationFromString(String token) {
try {
JSonSupport support = new JSonSupport();
return support.getGson().fromJson(token, GerritConfiguration.class);
} catch (Exception e) {
StatusHandler.log(new Status(IStatus.ERROR, GerritCorePlugin.PLUGIN_ID,
"Failed to deserialize configuration: '" + token + "'", e));
return null;
}
}
private static String configurationToString(GerritConfiguration config) {
try {
JSonSupport support = new JSonSupport();
return support.getGson().toJson(config);
} catch (Exception e) {
StatusHandler.log(new Status(IStatus.ERROR, GerritCorePlugin.PLUGIN_ID,
"Failed to serialize configuration", e));
return null;
}
}
CoreException toCoreException(TaskRepository repository, GerritException e) {
return new CoreException(toStatus(repository, e));
}
CoreException toCoreException(TaskRepository repository, UnsupportedClassVersionError e) {
return new CoreException(toStatus(repository, e));
}
Status toStatus(TaskRepository repository, GerritException e) {
String message;
if (e instanceof GerritHttpException) {
int code = ((GerritHttpException) e).getResponseCode();
message = NLS.bind("Unexpected error: {1} ({0})", code, HttpStatus.getStatusText(code));
} else if (e instanceof GerritLoginException) {
message = "Login failed";
} else if (e.getMessage() != null) {
message = NLS.bind("Unexpected error: {0}", e.getMessage());
} else {
message = "Unexpected error while communicating with Gerrit";
}
return new Status(IStatus.ERROR, GerritCorePlugin.PLUGIN_ID, message, e);
}
Status toStatus(TaskRepository repository, UnsupportedClassVersionError e) {
String message = NLS.bind("The Gerrit Connector requires at Java 1.6 or higer (installed version: {0})",
System.getProperty("java.version"));
return new Status(IStatus.ERROR, GerritCorePlugin.PLUGIN_ID, message, e);
}
}