1
0
Fork 0
libreoffice/comphelper/source/misc/configuration.cxx
Daniel Baumann 8e63e14cf6
Adding upstream version 4:25.2.3.
Signed-off-by: Daniel Baumann <daniel.baumann@progress-linux.org>
2025-06-22 16:20:04 +02:00

305 lines
9.9 KiB
C++

/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
/*
* This file is part of the LibreOffice project.
*
* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
*/
#include <sal/config.h>
#include <cassert>
#include <map>
#include <memory>
#include <mutex>
#include <string_view>
#include <com/sun/star/beans/PropertyAttribute.hpp>
#include <com/sun/star/configuration/ReadOnlyAccess.hpp>
#include <com/sun/star/configuration/ReadWriteAccess.hpp>
#include <com/sun/star/configuration/XReadWriteAccess.hpp>
#include <com/sun/star/configuration/theDefaultProvider.hpp>
#include <com/sun/star/container/XHierarchicalNameAccess.hpp>
#include <com/sun/star/container/XHierarchicalNameReplace.hpp>
#include <com/sun/star/container/XNameAccess.hpp>
#include <com/sun/star/container/XNameContainer.hpp>
#include <com/sun/star/lang/XLocalizable.hpp>
#include <com/sun/star/uno/Any.hxx>
#include <com/sun/star/uno/Reference.hxx>
#include <comphelper/solarmutex.hxx>
#include <comphelper/configuration.hxx>
#include <comphelper/configurationlistener.hxx>
#include <rtl/ustring.hxx>
#include <sal/log.hxx>
#include <i18nlangtag/languagetag.hxx>
namespace com::sun::star::uno { class XComponentContext; }
namespace {
OUString getDefaultLocale(
css::uno::Reference< css::uno::XComponentContext > const & context)
{
return LanguageTag(
css::uno::Reference< css::lang::XLocalizable >(
css::configuration::theDefaultProvider::get(context),
css::uno::UNO_QUERY_THROW)->
getLocale()).getBcp47(false);
}
OUString extendLocalizedPath(std::u16string_view path, OUString const & locale) {
SAL_WARN_IF(
locale.match("*"), "comphelper",
"Locale \"" << locale << "\" starts with \"*\"");
assert(locale.indexOf('&') == -1);
assert(locale.indexOf('"') == -1);
assert(locale.indexOf('\'') == -1);
return OUString::Concat(path) + "/['*" + locale + "']";
}
}
std::shared_ptr< comphelper::ConfigurationChanges >
comphelper::ConfigurationChanges::create(
css::uno::Reference<css::uno::XComponentContext> const & context)
{
return detail::ConfigurationWrapper::get(context).createChanges();
}
comphelper::ConfigurationChanges::~ConfigurationChanges() {}
void comphelper::ConfigurationChanges::commit() const {
access_->commitChanges();
}
comphelper::ConfigurationChanges::ConfigurationChanges(
css::uno::Reference< css::uno::XComponentContext > const & context):
access_(
css::configuration::ReadWriteAccess::create(
context, getDefaultLocale(context)))
{}
void comphelper::ConfigurationChanges::setPropertyValue(
OUString const & path, css::uno::Any const & value) const
{
access_->replaceByHierarchicalName(path, value);
}
css::uno::Reference< css::container::XHierarchicalNameReplace >
comphelper::ConfigurationChanges::getGroup(OUString const & path) const
{
return css::uno::Reference< css::container::XHierarchicalNameReplace >(
access_->getByHierarchicalName(path), css::uno::UNO_QUERY_THROW);
}
css::uno::Reference< css::container::XNameContainer >
comphelper::ConfigurationChanges::getSet(OUString const & path) const
{
return css::uno::Reference< css::container::XNameContainer >(
access_->getByHierarchicalName(path), css::uno::UNO_QUERY_THROW);
}
comphelper::detail::ConfigurationWrapper const &
comphelper::detail::ConfigurationWrapper::get(
css::uno::Reference<css::uno::XComponentContext> const & context)
{
static comphelper::detail::ConfigurationWrapper WRAPPER(context);
return WRAPPER;
}
comphelper::detail::ConfigurationWrapper::ConfigurationWrapper(
css::uno::Reference<css::uno::XComponentContext> const & context):
context_(context.is() ? context : comphelper::getProcessComponentContext()),
access_(css::configuration::ReadWriteAccess::create(context_, u"*"_ustr))
{}
comphelper::detail::ConfigurationWrapper::~ConfigurationWrapper() {}
bool comphelper::detail::ConfigurationWrapper::isReadOnly(OUString const & path)
const
{
return
(access_->getPropertyByHierarchicalName(path).Attributes
& css::beans::PropertyAttribute::READONLY)
!= 0;
}
css::uno::Any comphelper::detail::ConfigurationWrapper::getPropertyValue(std::u16string_view path) const
{
// should be short-circuited in ConfigurationProperty::get()
assert(!comphelper::IsFuzzing());
// Cache the configuration access, since some of the keys are used in hot code.
// Note that this cache is only used by the officecfg:: auto-generated code, using it for anything
// else would be unwise because the cache could end up containing stale entries.
static std::mutex gMutex;
static std::map<OUString, css::uno::Reference< css::container::XNameAccess >> gAccessMap;
sal_Int32 idx = path.rfind('/');
assert(idx!=-1);
OUString parentPath(path.substr(0, idx));
OUString childName(path.substr(idx+1));
std::scoped_lock aGuard(gMutex);
// check cache
auto it = gAccessMap.find(parentPath);
if (it == gAccessMap.end())
{
// not in the cache, look it up
css::uno::Reference<css::container::XNameAccess> access(
access_->getByHierarchicalName(parentPath), css::uno::UNO_QUERY_THROW);
it = gAccessMap.emplace(parentPath, access).first;
}
return it->second->getByName(childName);
}
void comphelper::detail::ConfigurationWrapper::setPropertyValue(
std::shared_ptr< ConfigurationChanges > const & batch,
OUString const & path, css::uno::Any const & value)
{
assert(batch);
batch->setPropertyValue(path, value);
}
css::uno::Any
comphelper::detail::ConfigurationWrapper::getLocalizedPropertyValue(
std::u16string_view path) const
{
return access_->getByHierarchicalName(
extendLocalizedPath(path, getDefaultLocale(context_)));
}
void comphelper::detail::ConfigurationWrapper::setLocalizedPropertyValue(
std::shared_ptr< ConfigurationChanges > const & batch,
OUString const & path, css::uno::Any const & value)
{
assert(batch);
batch->setPropertyValue(path, value);
}
css::uno::Reference< css::container::XHierarchicalNameAccess >
comphelper::detail::ConfigurationWrapper::getGroupReadOnly(
OUString const & path) const
{
return css::uno::Reference< css::container::XHierarchicalNameAccess >(
(css::configuration::ReadOnlyAccess::create(
context_, getDefaultLocale(context_))->
getByHierarchicalName(path)),
css::uno::UNO_QUERY_THROW);
}
css::uno::Reference< css::container::XHierarchicalNameReplace >
comphelper::detail::ConfigurationWrapper::getGroupReadWrite(
std::shared_ptr< ConfigurationChanges > const & batch,
OUString const & path)
{
assert(batch);
return batch->getGroup(path);
}
css::uno::Reference< css::container::XNameAccess >
comphelper::detail::ConfigurationWrapper::getSetReadOnly(
OUString const & path) const
{
return css::uno::Reference< css::container::XNameAccess >(
(css::configuration::ReadOnlyAccess::create(
context_, getDefaultLocale(context_))->
getByHierarchicalName(path)),
css::uno::UNO_QUERY_THROW);
}
css::uno::Reference< css::container::XNameContainer >
comphelper::detail::ConfigurationWrapper::getSetReadWrite(
std::shared_ptr< ConfigurationChanges > const & batch,
OUString const & path)
{
assert(batch);
return batch->getSet(path);
}
std::shared_ptr< comphelper::ConfigurationChanges >
comphelper::detail::ConfigurationWrapper::createChanges() const {
return std::shared_ptr< ConfigurationChanges >(
new ConfigurationChanges(context_));
}
void comphelper::ConfigurationListener::addListener(ConfigurationListenerPropertyBase *pListener)
{
maListeners.push_back( pListener );
mxConfig->addPropertyChangeListener( pListener->maName, this );
pListener->setProperty( mxConfig->getPropertyValue( pListener->maName ) );
}
void comphelper::ConfigurationListener::removeListener(ConfigurationListenerPropertyBase *pListener)
{
auto it = std::find( maListeners.begin(), maListeners.end(), pListener );
if ( it != maListeners.end() )
{
maListeners.erase( it );
mxConfig->removePropertyChangeListener( pListener->maName, this );
}
}
void comphelper::ConfigurationListener::dispose()
{
for (auto const& listener : maListeners)
{
mxConfig->removePropertyChangeListener( listener->maName, this );
listener->dispose();
}
maListeners.clear();
mxConfig.clear();
mbDisposed = true;
}
void SAL_CALL comphelper::ConfigurationListener::disposing(css::lang::EventObject const &)
{
dispose();
}
void SAL_CALL comphelper::ConfigurationListener::propertyChange(
css::beans::PropertyChangeEvent const &rEvt )
{
// Code is commonly used inside the SolarMutexGuard
// so to avoid concurrent writes to the property,
// and allow fast, lock-less access, guard here.
//
// Note that we are abusing rtl::Reference here to do acquire/release because,
// unlike osl::Guard, it is tolerant of null pointers, and on some code paths, the
// SolarMutex does not exist.
rtl::Reference<comphelper::SolarMutex> xMutexGuard( comphelper::SolarMutex::get() );
assert( rEvt.Source == mxConfig );
for (auto const& listener : maListeners)
{
if ( listener->maName == rEvt.PropertyName )
{
// ignore rEvt.NewValue - in theory it could be stale => not set.
css::uno::Any aValue = mxConfig->getPropertyValue( listener->maName );
listener->setProperty( aValue );
}
}
}
namespace comphelper {
static bool bIsFuzzing = false;
#if !defined(FUZZING_BUILD_MODE_UNSAFE_FOR_PRODUCTION)
bool IsFuzzing()
{
return bIsFuzzing;
}
#endif
void EnableFuzzing()
{
bIsFuzzing = true;
LanguageTag::disable_lt_tag_parse();
}
}
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */