org.netbeans.modules.bugtracking.RepositoryRegistry Maven / Gradle / Ivy
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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.netbeans.modules.bugtracking;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.beans.PropertyChangeListener;
import java.beans.PropertyChangeSupport;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.HashMap;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;
import java.util.concurrent.Semaphore;
import java.util.logging.Level;
import java.util.prefs.BackingStoreException;
import java.util.prefs.Preferences;
import javax.swing.Timer;
import org.netbeans.api.keyring.Keyring;
import org.netbeans.api.progress.ProgressHandle;
import org.netbeans.api.progress.ProgressHandleFactory;
import org.netbeans.modules.bugtracking.api.Repository;
import org.netbeans.modules.bugtracking.api.RepositoryManager;
import org.netbeans.modules.bugtracking.team.TeamRepositories;
import org.netbeans.modules.bugtracking.spi.RepositoryInfo;
import org.netbeans.modules.bugtracking.util.BugtrackingUtil;
import org.netbeans.modules.team.commons.LogUtils;
import org.netbeans.modules.bugtracking.commons.NBBugzillaUtils;
import org.netbeans.modules.team.spi.TeamAccessorUtils;
import org.openide.util.Exceptions;
import org.openide.util.NbPreferences;
import org.openide.util.RequestProcessor;
/**
*
* @author Tomas Stupka
*/
public class RepositoryRegistry {
/**
* A repository was created or removed, where old value is a Collection of all repositories
* before the change and new value a Collection of all repositories after the change.
*/
public static final String EVENT_REPOSITORIES_CHANGED = RepositoryManager.EVENT_REPOSITORIES_CHANGED; // NOI18N
private final PropertyChangeSupport changeSupport = new PropertyChangeSupport(this);
private static final String BUGTRACKING_REPO = "bugracking.repository_"; // NOI18N
private static final String DELIMITER = "<=>"; // NOI18N
private static RepositoryRegistry instance;
private final Semaphore repositorySemaphore = new Semaphore(1);
//@GuardedBy("repositorySemaphore")
private RepositoriesMap repositories;
private RepositoryRegistry() {
lockRepositories();
final long t = System.currentTimeMillis();
new RequestProcessor(RepositoryRegistry.class.getName()).post(new Runnable() {
@Override
public void run() {
final ProgressHandle[] ph = new ProgressHandle[1];
final Timer timer[] = new Timer[1];
timer[0] = new Timer(800, new ActionListener() {
@Override
public void actionPerformed(ActionEvent e) {
if(repositories == null) {
timer[0].stop();
ph[0] = ProgressHandleFactory.createSystemHandle("Initializing Bugtracking repositories");
ph[0].start();
}
}
});
timer[0].start();
try {
loadRepositories();
} finally {
releaseRepositoriesLock();
if(ph[0] != null) {
ph[0].finish();
}
timer[0].stop();
BugtrackingManager.LOG.log(Level.INFO, "Loading stored repositories took {0} millis.", (System.currentTimeMillis() - t));
}
}
});
}
public boolean isInitializing() {
return repositories == null;
}
/**
* Returns the singleton RepositoryRegistry instance
*
* @return
*/
public static synchronized RepositoryRegistry getInstance() {
if(instance == null) {
instance = new RepositoryRegistry();
}
return instance;
}
/**
* Returns all repositories
*
* @return
*/
public Collection getRepositories() {
lockRepositories();
try {
List l = repositories.getRepositories();
return new LinkedList(l);
} finally {
releaseRepositoriesLock();
}
}
/**
* Returns all known repositories incl. the Team ones
*
* @param pingOpenProjects if {@code false}, search only Team projects
* that are currently open in the Team dashboard;
* if {@code true}, search also all Team projects
* currently opened in the IDE
* @return repositories
*/
public Collection getKnownRepositories(boolean pingOpenProjects) {
return getKnownRepositories(pingOpenProjects, false);
}
/**
* Returns all known repositories incl. the Team ones
*
* @param pingOpenProjects if {@code false}, search only Team projects
* that are currently open in the Team dashboard;
* if {@code true}, search also all Team projects
* currently opened in the IDE
* @param onlyDashboardOpenProjects
* @return repositories
*/
public Collection getKnownRepositories(boolean pingOpenProjects, boolean onlyDashboardOpenProjects) {
Collection otherRepos = getRepositories();
Collection teamRepos = TeamRepositories.getInstance().getRepositories(pingOpenProjects, onlyDashboardOpenProjects);
List ret = new ArrayList(teamRepos.size() + otherRepos.size());
ret.addAll(otherRepos);
ret.addAll(teamRepos);
logRepositoryUsage(ret);
return ret;
}
/**
* Returns all repositories for the connector with the given ID
*
* @param connectorID
* @return
*/
public Collection getRepositories(String connectorID, boolean allKnown) {
LinkedList ret = new LinkedList();
lockRepositories();
try {
final Map m = repositories.get(connectorID);
if(m != null) {
ret.addAll(m.values());
}
} finally {
releaseRepositoriesLock();
}
if(allKnown) {
// team repos (not registered by user)
Collection repos = TeamRepositories.getInstance().getRepositories(false, true);
for (RepositoryImpl impl : repos) {
if(connectorID.equals(impl.getConnectorId())) {
ret.add(impl);
}
}
}
return ret;
}
public RepositoryImpl getRepository(String connectorId, String repoId, boolean allKnown) {
Collection repos = getRepositories(connectorId, allKnown);
for (RepositoryImpl repo : repos) {
if(repo.getId().equals(repoId)) {
return repo;
}
}
return null;
}
/**
* Add the given repository
*
* @param repository
*/
public void addRepository(RepositoryImpl repository) {
assert repository != null;
if(repository.isTeamRepository() && !NBBugzillaUtils.isNbRepository(repository.getUrl())) {
// we don't store team repositories - XXX shouldn't be even called
return;
}
lockRepositories();
try {
repositories.put(repository); // cache
putRepository(repository); // persist
} finally {
releaseRepositoriesLock();
}
fireRepositoriesChanged(null, Arrays.asList(repository));
}
/**
* Remove the given repository
*
* @param repository
*/
public void removeRepository(RepositoryImpl repository) {
lockRepositories();
try {
RepositoryInfo info = repository.getInfo();
String connectorID = info.getConnectorId();
// persist remove
getPreferences().remove(getRepositoryKey(info));
// remove from cache
repositories.remove(connectorID, repository);
} finally {
releaseRepositoriesLock();
}
fireRepositoriesChanged(Arrays.asList(repository), null);
}
/**
* remove a listener from this connector
* @param listener
*/
public synchronized void removePropertyChangeListener(PropertyChangeListener listener) {
changeSupport.removePropertyChangeListener(listener);
TeamRepositories.getInstance().removePropertyChangeListener(listener);
}
/**
* Add a listener to this connector to listen on events
* @param listener
*/
public synchronized void addPropertyChangeListener(PropertyChangeListener listener) {
changeSupport.addPropertyChangeListener(listener);
TeamRepositories.getInstance().addPropertyChangeListener(listener);
}
////////////////////////////////////////////////////////////////////////////////////////
// private
////////////////////////////////////////////////////////////////////////////////////////
/**
* for testing
*/
void flushRepositories() {
repositories = null;
}
private String getRepositoryKey(RepositoryInfo info) {
return BUGTRACKING_REPO + info.getConnectorId() + DELIMITER + info.getID();
}
void loadRepositories() {
RepositoriesMap map = new RepositoriesMap();
try {
migrateBugzilla();
migrateJira();
String[] ids = getRepositoryIds();
DelegatingConnector[] connectors = BugtrackingManager.getInstance().getConnectors();
if (ids != null) {
for (String id : ids) {
String[] idArray = id.split(DELIMITER);
String connectorId = idArray[0].substring(BUGTRACKING_REPO.length());
for (DelegatingConnector c : connectors) {
if(c.getID().equals(connectorId)) {
RepositoryInfo info = SPIAccessor.IMPL.read(getPreferences(), id);
if(info != null) {
Repository repo = c.createRepository(info);
if (repo != null) {
map.put(APIAccessor.IMPL.getImpl(repo));
}
}
}
}
}
}
for (DelegatingConnector c : connectors) {
if (BugtrackingManager.isLocalConnectorID(c.getID())) {
Repository repo = c.createRepository();
if (repo != null) {
map.put(APIAccessor.IMPL.getImpl(repo));
}
}
}
} finally {
repositories = map;
}
}
private String[] getRepositoryIds() {
return getKeysWithPrefix(BUGTRACKING_REPO);
}
/**
* package private for testing
*/
void putRepository(RepositoryImpl repository) {
RepositoryInfo info = repository.getInfo();
final String key = getRepositoryKey(info);
SPIAccessor.IMPL.store(getPreferences(), info, key);
}
private Preferences getPreferences() {
return NbPreferences.forModule(RepositoryRegistry.class);
}
private String[] getKeysWithPrefix(String prefix) {
String[] keys = null;
try {
keys = getPreferences().keys();
} catch (BackingStoreException ex) {
BugtrackingManager.LOG.log(Level.SEVERE, null, ex); // XXX
}
if (keys == null || keys.length == 0) {
return new String[0];
}
List ret = new ArrayList();
for (String key : keys) {
if (key.startsWith(prefix)) {
ret.add(key);
}
}
return ret.toArray(new String[0]);
}
/**
*
* @param oldRepositories - lists repositories which were available for the connector before the change
* @param newRepositories - lists repositories which are available for the connector after the change
*/
private void fireRepositoriesChanged(Collection oldRepositories, Collection newRepositories) {
changeSupport.firePropertyChange(EVENT_REPOSITORIES_CHANGED, oldRepositories, newRepositories);
}
private class RepositoriesMap extends HashMap> {
public void remove(String connectorID, RepositoryImpl repository) {
Map m = get(connectorID);
if(m != null) {
m.remove(repository.getId());
}
}
public void put(RepositoryImpl repository) {
String connectorID = repository.getInfo().getConnectorId();
Map m = get(connectorID);
if(m == null) {
m = new HashMap();
put(connectorID, m);
}
m.put(repository.getId(), repository);
}
List getRepositories() {
List ret = new LinkedList();
for (Entry> e : entrySet()) {
ret.addAll(e.getValue().values());
}
return ret;
}
}
///////////////////////////////////////////////////////////////////////////////////////////////////////
private static final String JIRA_REPO_ID = "jira.repository_"; // NOI18N
private static final String BUGZILLA_REPO_ID = "bugzilla.repository_"; // NOI18N
private static final String NB_BUGZILLA_USERNAME = "nbbugzilla.username"; // NOI18N
private static final String NB_BUGZILLA_PASSWORD = "nbbugzilla.password"; // NOI18N
private static final String REPOSITORY_SETTING_SHORT_LOGIN = "bugzilla.shortLoginEnabled"; // NOI18N
private void migrateBugzilla() {
Preferences preferences = getBugzillaPreferences();
String[] repoIds = getRepoIds(preferences, BUGZILLA_REPO_ID);
for (String id : repoIds) {
migrateBugzillaRepository(preferences, id);
preferences.remove(BUGZILLA_REPO_ID + id);
}
preferences.remove(NB_BUGZILLA_USERNAME);
}
private void migrateJira() {
Preferences preferences = getJiraPreferences();
String[] repoIds = getRepoIds(preferences, JIRA_REPO_ID);
for (String id : repoIds) {
migrateJiraRepository(preferences, id);
preferences.remove(JIRA_REPO_ID + id);
}
}
private String[] getRepoIds(Preferences preferences, String repoId) {
String[] keys = null;
try {
keys = preferences.keys();
} catch (BackingStoreException ex) {
BugtrackingManager.LOG.log(Level.SEVERE, null, ex);
}
if (keys == null || keys.length == 0) {
return new String[0];
}
List ret = new ArrayList();
for (String key : keys) {
if (key.startsWith(repoId)) {
ret.add(key.substring(repoId.length()));
}
}
return ret.toArray(new String[0]);
}
private void migrateBugzillaRepository(Preferences preferences, String repoID) {
String[] values = getRepositoryValues(preferences, BUGZILLA_REPO_ID, repoID);
if(values == null) {
return;
}
assert values.length == 3 || values.length == 6 || values.length == 7;
String url = values[0];
String user;
char[] password;
if(NBBugzillaUtils.isNbRepository(url)) {
user = getBugzillaNBUsername();
char[] psswdArray = getNBPassword();
password = psswdArray != null ? psswdArray : new char[0];
} else {
user = values[1];
password = BugtrackingUtil.readPassword(values[2], null, user, url);
}
String httpUser = values.length > 3 ? values[3] : null;
char[] httpPassword = values.length > 3 ? BugtrackingUtil.readPassword(values[4], "http", httpUser, url) : new char[0]; // NOI18N
String shortNameEnabled = "false"; // NOI18N
if (values.length > 5) {
shortNameEnabled = values[5];
}
String name;
if (values.length > 6) {
name = values[6];
} else {
name = repoID;
}
RepositoryInfo info = new RepositoryInfo(
repoID,
"org.netbeans.modules.bugzilla", // NOI18N
url,
name,
name,
user,
httpUser,
password,
httpPassword);
info.putValue(REPOSITORY_SETTING_SHORT_LOGIN, shortNameEnabled);
SPIAccessor.IMPL.store(getPreferences(), info, getRepositoryKey(info));
}
private void migrateJiraRepository(Preferences preferences, String repoID) {
String[] values = getRepositoryValues(preferences, JIRA_REPO_ID, repoID);
String url = values[0];
String user = values[1];
String password = new String(BugtrackingUtil.readPassword(values[2], null, user, url));
String httpUser = values.length > 3 ? values[3] : null;
String httpPassword = new String(values.length > 3 ? BugtrackingUtil.readPassword(values[4], "http", httpUser, url) : null); // NOI18N
String repoName;
if(values.length > 5) {
repoName = values[5];
} else {
repoName = repoID;
}
RepositoryInfo info = new RepositoryInfo(
repoID,
"org.netbeans.modules.jira", // NOI18N
url,
repoName,
repoName,
user,
httpUser,
password.toCharArray(),
httpPassword.toCharArray());
SPIAccessor.IMPL.store(getPreferences(), info, getRepositoryKey(info));
}
private static String[] getRepositoryValues(Preferences preferences, String repoPrefix, String repoID) {
String repoString = preferences.get(repoPrefix + repoID, ""); // NOI18N
if(repoString.equals("")) { // NOI18N
return null;
}
return repoString.split(DELIMITER);
}
private static Preferences getBugzillaPreferences() {
return NbPreferences.root().node("org/netbeans/modules/bugzilla"); // NOI18N
}
private static Preferences getJiraPreferences() {
return NbPreferences.root().node("org/netbeans/modules/jira"); // NOI18N
}
private static String getBugzillaNBUsername() {
String user = getBugzillaPreferences().get(NB_BUGZILLA_USERNAME, ""); // NOI18N
return user;
}
private static char[] getNBPassword() {
return Keyring.read(NB_BUGZILLA_PASSWORD);
}
private void logRepositoryUsage(Collection ret) {
for (RepositoryImpl repositoryImpl : ret) {
LogUtils.logRepositoryUsage(repositoryImpl.getConnectorId(), repositoryImpl.getUrl());
// log team usage
if (repositoryImpl.isTeamRepository()) {
TeamAccessorUtils.logTeamUsage(repositoryImpl.getUrl(), "ISSUE_TRACKING", LogUtils.getBugtrackingType(repositoryImpl.getConnectorId())); //NOI18N
}
}
}
private void lockRepositories() {
try {
repositorySemaphore.acquire();
} catch (InterruptedException ex) {
Exceptions.printStackTrace(ex);
}
}
private void releaseRepositoriesLock() {
repositorySemaphore.release();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy