blob: 05c37015a5e7a4ab06669f95e2989645fdf9bb09 [file] [log] [blame]
/**************************************************************
*
* 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.
*
*************************************************************/
// MARKER(update_precomp.py): autogen include statement, do not remove
#include "precompiled_desktop.hxx"
#include "dp_manager.h"
#include "dp_resource.h"
#include "cppuhelper/compbase1.hxx"
#include "comphelper/servicedecl.hxx"
#include "com/sun/star/deployment/thePackageManagerFactory.hpp"
using namespace ::dp_misc;
using namespace ::com::sun::star;
using namespace ::com::sun::star::uno;
using ::rtl::OUString;
namespace dp_manager {
namespace factory {
typedef ::cppu::WeakComponentImplHelper1<
deployment::XPackageManagerFactory > t_pmfac_helper;
//==============================================================================
class PackageManagerFactoryImpl : private MutexHolder, public t_pmfac_helper
{
Reference<XComponentContext> m_xComponentContext;
Reference<deployment::XPackageManager> m_xUserMgr;
Reference<deployment::XPackageManager> m_xSharedMgr;
Reference<deployment::XPackageManager> m_xBundledMgr;
Reference<deployment::XPackageManager> m_xTmpMgr;
Reference<deployment::XPackageManager> m_xBakMgr;
typedef ::std::hash_map<
OUString, WeakReference<deployment::XPackageManager>,
::rtl::OUStringHash > t_string2weakref;
t_string2weakref m_managers;
protected:
inline void check();
virtual void SAL_CALL disposing();
public:
virtual ~PackageManagerFactoryImpl();
PackageManagerFactoryImpl(
Reference<XComponentContext> const & xComponentContext );
// XPackageManagerFactory
virtual Reference<deployment::XPackageManager> SAL_CALL getPackageManager(
OUString const & context ) throw (RuntimeException);
};
//==============================================================================
namespace sdecl = comphelper::service_decl;
sdecl::class_<PackageManagerFactoryImpl> servicePMFI;
extern sdecl::ServiceDecl const serviceDecl(
servicePMFI,
// a private one:
"com.sun.star.comp.deployment.PackageManagerFactory",
"com.sun.star.comp.deployment.PackageManagerFactory" );
//==============================================================================
bool singleton_entries(
Reference<registry::XRegistryKey> const & xRegistryKey )
{
try {
Reference<registry::XRegistryKey> xKey(
xRegistryKey->createKey(
serviceDecl.getImplementationName() +
// xxx todo: use future generated function to get singleton name
OUSTR("/UNO/SINGLETONS/"
"com.sun.star.deployment.thePackageManagerFactory") ) );
xKey->setStringValue( serviceDecl.getSupportedServiceNames()[0] );
return true;
}
catch (registry::InvalidRegistryException & exc) {
(void) exc; // avoid warnings
OSL_ENSURE( 0, ::rtl::OUStringToOString(
exc.Message, RTL_TEXTENCODING_UTF8 ).getStr() );
return false;
}
}
//______________________________________________________________________________
PackageManagerFactoryImpl::PackageManagerFactoryImpl(
Reference<XComponentContext> const & xComponentContext )
: t_pmfac_helper( getMutex() ),
m_xComponentContext( xComponentContext )
{
}
//______________________________________________________________________________
PackageManagerFactoryImpl::~PackageManagerFactoryImpl()
{
}
//______________________________________________________________________________
inline void PackageManagerFactoryImpl::check()
{
::osl::MutexGuard guard( getMutex() );
if (rBHelper.bInDispose || rBHelper.bDisposed)
{
throw lang::DisposedException(
OUSTR("PackageManagerFactory instance has already been disposed!"),
static_cast<OWeakObject *>(this) );
}
}
//______________________________________________________________________________
void PackageManagerFactoryImpl::disposing()
{
// dispose all managers:
::osl::MutexGuard guard( getMutex() );
t_string2weakref::const_iterator iPos( m_managers.begin() );
t_string2weakref::const_iterator const iEnd( m_managers.end() );
for ( ; iPos != iEnd; ++iPos )
try_dispose( iPos->second );
m_managers = t_string2weakref();
// the below are already disposed:
m_xUserMgr.clear();
m_xSharedMgr.clear();
m_xBundledMgr.clear();
m_xTmpMgr.clear();
m_xBakMgr.clear();
}
// XPackageManagerFactory
//______________________________________________________________________________
Reference<deployment::XPackageManager>
PackageManagerFactoryImpl::getPackageManager( OUString const & context )
throw (RuntimeException)
{
Reference< deployment::XPackageManager > xRet;
::osl::ResettableMutexGuard guard( getMutex() );
check();
t_string2weakref::const_iterator const iFind( m_managers.find( context ) );
if (iFind != m_managers.end()) {
xRet = iFind->second;
if (xRet.is())
return xRet;
}
guard.clear();
xRet.set( PackageManagerImpl::create( m_xComponentContext, context ) );
guard.reset();
::std::pair< t_string2weakref::iterator, bool > insertion(
m_managers.insert( t_string2weakref::value_type( context, xRet ) ) );
if (insertion.second)
{
OSL_ASSERT( insertion.first->second.get() == xRet );
// hold user, shared mgrs for whole process: live deployment
if (context.equalsAsciiL( RTL_CONSTASCII_STRINGPARAM("user") ))
m_xUserMgr = xRet;
else if (context.equalsAsciiL( RTL_CONSTASCII_STRINGPARAM("shared") ))
m_xSharedMgr = xRet;
else if (context.equalsAsciiL( RTL_CONSTASCII_STRINGPARAM("bundled") ))
m_xBundledMgr = xRet;
else if (context.equalsAsciiL( RTL_CONSTASCII_STRINGPARAM("tmp") ))
m_xTmpMgr = xRet;
else if (context.equalsAsciiL( RTL_CONSTASCII_STRINGPARAM("bak") ))
m_xBakMgr = xRet;
}
else
{
Reference< deployment::XPackageManager > xAlreadyIn(
insertion.first->second );
if (xAlreadyIn.is())
{
guard.clear();
try_dispose( xRet );
xRet = xAlreadyIn;
}
else
{
insertion.first->second = xRet;
}
}
return xRet;
}
} // namespace factory
} // namespace dp_manager