/*
* SIP Communicator, the OpenSource Java VoIP and Instant Messaging client.
*
* Distributable under LGPL license.
* See terms of license at gnu.org.
*/
package net.java.sip.communicator.service.protocol;
import net.java.sip.communicator.service.protocol.event.*;
import org.osgi.framework.*;
/**
* Provides utilities to aid the manipulation of {@link AccountManager}.
*
* @author Lubomir Marinov
*/
public final class AccountManagerUtils
{
private static AccountManager getAccountManager(BundleContext bundleContext)
{
return
(AccountManager)
bundleContext.getService(
bundleContext.getServiceReference(
AccountManager.class.getName()));
}
/**
* Starts a specific Bundle
and waits for the
* AccountManager
available in a specific
* BundleContext
to load the stored accounts of a
* ProtocolProviderFactory
with a specific protocol name.
*
* @param bundleContextWithAccountManager
* the BundleContext
in which an
* AccountManager
service is registered
* @param bundleToStart
* the Bundle
to be started
* @param protocolNameToWait
* the protocol name of a ProtocolProviderFactory
to
* wait the end of the loading of the stored accounts for
* @throws BundleException
* @throws InterruptedException
* if any thread interrupted the current thread before or while
* the current thread was waiting for the loading of the stored
* accounts
*/
public static void startBundleAndWaitStoredAccountsLoaded(
BundleContext bundleContextWithAccountManager,
final Bundle bundleToStart,
final String protocolNameToWait)
throws BundleException,
InterruptedException
{
AccountManager accountManager
= getAccountManager(bundleContextWithAccountManager);
final boolean[] storedAccountsAreLoaded = new boolean[1];
AccountManagerListener listener = new AccountManagerListener()
{
public void handleAccountManagerEvent(AccountManagerEvent event)
{
if (AccountManagerEvent.STORED_ACCOUNTS_LOADED
!= event.getType())
return;
ProtocolProviderFactory factory = event.getFactory();
/*
* If the event is for a factory with a protocol name other than
* protocolNameToWait, it's not the one we're waiting for.
*/
if ((factory != null)
&& !protocolNameToWait
.equals(factory.getProtocolName()))
return;
/*
* If the event if for a factory which is no longer registered,
* then it's not the one we're waiting for because we're waiting
* for the specified bundle to start and register a factory.
*/
if (factory != null)
{
BundleContext bundleContext
= bundleToStart.getBundleContext();
/*
* If the specified bundle still hasn't started, the event
* cannot be the one we're waiting for.
*/
if (bundleContext == null)
return;
ServiceReference[] factoryRefs;
try
{
factoryRefs
= bundleContext
.getServiceReferences(
ProtocolProviderFactory.class.getName(),
"("
+ ProtocolProviderFactory.PROTOCOL
+ "="
+ protocolNameToWait
+ ")");
}
catch (InvalidSyntaxException isex)
{
/*
* Not likely so ignore it and assume the event is for
* a valid factory.
*/
factoryRefs = null;
}
if (factoryRefs != null)
{
boolean factoryIsRegistered = false;
for (ServiceReference factoryRef : factoryRefs)
if (factory == bundleContext.getService(factoryRef))
{
factoryIsRegistered = true;
break;
}
if (!factoryIsRegistered)
return;
}
}
synchronized (storedAccountsAreLoaded)
{
storedAccountsAreLoaded[0] = true;
storedAccountsAreLoaded.notify();
}
}
};
accountManager.addListener(listener);
try
{
bundleToStart.start();
while (true)
{
synchronized (storedAccountsAreLoaded)
{
if (storedAccountsAreLoaded[0])
{
break;
}
storedAccountsAreLoaded.wait();
}
}
}
finally
{
accountManager.removeListener(listener);
}
}
/**
* Prevents the creation of AccountManagerUtils
instances.
*/
private AccountManagerUtils()
{
}
}