office-gobmx/include/comphelper/configuration.hxx
Xisco Fauli cfcd6b2557 tdf#157042: Revert "re-apply "optimize ConfigurationProperty::get()""
This reverts commit 3a4a00a51a.

<x1sc0> noelgrandin, regarding tdf#157042, the commit was reverted in 24-2 and 7-6 branches but not in master so I was wondering what to do next. There are clear steps on how to reproduce it in comment 27 but it seems the crash is not reproducible with a debug build ( according to comment 37 )
<noelgrandin> x1sc0, let me try to reproduce that
<noelgrandin> x1sc0, I cant reproduce, please just revert that on master

Change-Id: I45dcf8f4b422e1a19eaa41ec7614db569b5aac7c
Reviewed-on: https://gerrit.libreoffice.org/c/core/+/163125
Reviewed-by: Noel Grandin <noel.grandin@collabora.co.uk>
Tested-by: Jenkins
Reviewed-by: Xisco Fauli <xiscofauli@libreoffice.org>
2024-02-08 21:33:15 +01:00

391 lines
13 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/.
*/
#ifndef INCLUDED_COMPHELPER_CONFIGURATION_HXX
#define INCLUDED_COMPHELPER_CONFIGURATION_HXX
#include <sal/config.h>
#include <optional>
#include <string_view>
#include <com/sun/star/uno/Any.hxx>
#include <com/sun/star/uno/Reference.h>
#include <comphelper/comphelperdllapi.h>
#include <comphelper/processfactory.hxx>
#include <sal/types.h>
#include <memory>
namespace com::sun::star {
namespace configuration { class XReadWriteAccess; }
namespace container {
class XHierarchicalNameAccess;
class XHierarchicalNameReplace;
class XNameAccess;
class XNameContainer;
}
namespace uno { class XComponentContext; }
}
namespace comphelper {
namespace detail { class ConfigurationWrapper; }
/// A batch of configuration changes that is committed as a whole.
///
/// Client code needs to call commit explicitly; otherwise the changes are lost
/// when the instance is destroyed.
///
/// This is the only class from this header file that client code should use
/// directly.
class COMPHELPER_DLLPUBLIC ConfigurationChanges {
public:
static std::shared_ptr<ConfigurationChanges> create(
css::uno::Reference<css::uno::XComponentContext> const & context
= css::uno::Reference<css::uno::XComponentContext>());
~ConfigurationChanges();
void commit() const;
private:
ConfigurationChanges(const ConfigurationChanges&) = delete;
ConfigurationChanges& operator=(const ConfigurationChanges&) = delete;
SAL_DLLPRIVATE ConfigurationChanges(
css::uno::Reference< css::uno::XComponentContext >
const & context);
SAL_DLLPRIVATE void setPropertyValue(
OUString const & path, css::uno::Any const & value)
const;
SAL_DLLPRIVATE css::uno::Reference<
css::container::XHierarchicalNameReplace >
getGroup(OUString const & path) const;
SAL_DLLPRIVATE
css::uno::Reference< css::container::XNameContainer >
getSet(OUString const & path) const;
css::uno::Reference<
css::configuration::XReadWriteAccess > access_;
friend class detail::ConfigurationWrapper;
};
namespace detail {
/// @internal
class COMPHELPER_DLLPUBLIC ConfigurationWrapper {
public:
static ConfigurationWrapper const & get(
css::uno::Reference<css::uno::XComponentContext> const & context);
bool isReadOnly(OUString const & path) const;
css::uno::Any getPropertyValue(std::u16string_view path) const;
static void setPropertyValue(
std::shared_ptr< ConfigurationChanges > const & batch,
OUString const & path, css::uno::Any const & value);
css::uno::Any getLocalizedPropertyValue(
std::u16string_view path) const;
static void setLocalizedPropertyValue(
std::shared_ptr< ConfigurationChanges > const & batch,
OUString const & path, css::uno::Any const & value);
css::uno::Reference<
css::container::XHierarchicalNameAccess >
getGroupReadOnly(OUString const & path) const;
static css::uno::Reference<
css::container::XHierarchicalNameReplace >
getGroupReadWrite(
std::shared_ptr< ConfigurationChanges > const & batch,
OUString const & path);
css::uno::Reference< css::container::XNameAccess >
getSetReadOnly(OUString const & path) const;
static css::uno::Reference< css::container::XNameContainer >
getSetReadWrite(
std::shared_ptr< ConfigurationChanges > const & batch,
OUString const & path);
std::shared_ptr< ConfigurationChanges > createChanges() const;
private:
SAL_DLLPRIVATE explicit ConfigurationWrapper(
css::uno::Reference<css::uno::XComponentContext> const & context);
SAL_DLLPRIVATE ~ConfigurationWrapper();
ConfigurationWrapper(const ConfigurationWrapper&) = delete;
ConfigurationWrapper& operator=(const ConfigurationWrapper&) = delete;
css::uno::Reference< css::uno::XComponentContext > context_;
css::uno::Reference< css::configuration::XReadWriteAccess > access_;
// should really be a css.configuration.ReadOnlyAccess (with added
// css.beans.XHierarchicalPropertySetInfo), but then
// configmgr::Access::asProperty() would report all properties as
// READONLY, so isReadOnly() would not work
};
/// @internal
template< typename T > struct Convert {
static css::uno::Any toAny(T const & value)
{ return css::uno::Any(value); }
static T fromAny(css::uno::Any const & value)
{ return value.get< T >(); }
private:
Convert(const Convert&) = delete;
Convert& operator=(const Convert&) = delete;
Convert() = delete;
~Convert() = delete;
};
/// @internal
template< typename T > struct Convert< std::optional< T > >
{
static css::uno::Any toAny(std::optional< T > const & value) {
return value
? css::uno::Any(*value)
: css::uno::Any();
}
static std::optional< T > fromAny(css::uno::Any const & value)
{
return value.hasValue()
? std::optional< T >(value.get< T >()) : std::optional< T >();
}
private:
Convert(const Convert&) = delete;
Convert& operator=(const Convert&) = delete;
Convert() = delete;
~Convert() = delete;
};
}
// Avoid using the config layer and rely on defaults which is only useful
// for special test tool targets (typically fuzzing) where start-up speed
// is of the essence
#if defined(FUZZING_BUILD_MODE_UNSAFE_FOR_PRODUCTION)
constexpr bool IsFuzzing() { return true; }
#else
COMPHELPER_DLLPUBLIC bool IsFuzzing();
#endif
COMPHELPER_DLLPUBLIC void EnableFuzzing();
/// A type-safe wrapper around a (non-localized) configuration property.
///
/// Automatically generated headers for the various configuration properties
/// derive from this template and make available its member functions to access
/// each given configuration property.
template< typename T, typename U > struct ConfigurationProperty
{
/// Get the read-only status of the given (non-localized) configuration
/// property.
static bool isReadOnly(
css::uno::Reference<css::uno::XComponentContext> const & context
= css::uno::Reference<css::uno::XComponentContext>())
{
return detail::ConfigurationWrapper::get(context).isReadOnly(T::path());
}
/// Get the value of the given (non-localized) configuration property.
///
/// For nillable properties, U is of type std::optional<U'>.
static U get(
css::uno::Reference<css::uno::XComponentContext> const & context
= css::uno::Reference<css::uno::XComponentContext>())
{
if (comphelper::IsFuzzing())
return U();
// Folding this into one statement causes a bogus error at least with
// Red Hat GCC 4.6.2-1:
css::uno::Any a(
detail::ConfigurationWrapper::get(context).getPropertyValue(
T::path()));
return detail::Convert< U >::fromAny(a);
}
/// Set the value of the given (non-localized) configuration property, via a
/// given changes batch.
///
/// For nillable properties, U is of type std::optional<U'>.
static void set(
U const & value,
std::shared_ptr< ConfigurationChanges > const & batch)
{
comphelper::detail::ConfigurationWrapper::setPropertyValue(
batch, T::path(), detail::Convert< U >::toAny(value));
}
private:
ConfigurationProperty(const ConfigurationProperty&) = delete;
ConfigurationProperty& operator=(const ConfigurationProperty&) = delete;
ConfigurationProperty() = delete;
~ConfigurationProperty() = delete;
};
/// A type-safe wrapper around a localized configuration property.
///
/// Automatically generated headers for the various localized configuration
/// properties derive from this template and make available its member functions
/// to access each given localized configuration property.
template< typename T, typename U > struct ConfigurationLocalizedProperty
{
/// Get the read-only status of the given (localized) configuration
/// property.
static bool isReadOnly(
css::uno::Reference<css::uno::XComponentContext> const & context
= css::uno::Reference<css::uno::XComponentContext>())
{
return detail::ConfigurationWrapper::get(context).isReadOnly(T::path());
}
/// Get the value of the given localized configuration property, for the
/// locale currently set at the
/// com.sun.star.configuration.theDefaultProvider.
///
/// For nillable properties, U is of type std::optional<U'>.
static U get(
css::uno::Reference<css::uno::XComponentContext> const & context
= css::uno::Reference<css::uno::XComponentContext>())
{
// Folding this into one statement causes a bogus error at least with
// Red Hat GCC 4.6.2-1:
css::uno::Any a(
detail::ConfigurationWrapper::get(context).
getLocalizedPropertyValue(T::path()));
return detail::Convert< U >::fromAny(a);
}
/// Set the value of the given localized configuration property, for the
/// locale currently set at the
/// com.sun.star.configuration.theDefaultProvider, via a given changes
/// batch.
///
/// For nillable properties, U is of type std::optional<U'>.
static void set(
U const & value,
std::shared_ptr< ConfigurationChanges > const & batch)
{
comphelper::detail::ConfigurationWrapper::setLocalizedPropertyValue(
batch, T::path(), detail::Convert< U >::toAny(value));
}
private:
ConfigurationLocalizedProperty(const ConfigurationLocalizedProperty&) = delete;
ConfigurationLocalizedProperty& operator=(const ConfigurationLocalizedProperty&) = delete;
ConfigurationLocalizedProperty() = delete;
~ConfigurationLocalizedProperty() = delete;
};
/// A type-safe wrapper around a configuration group.
///
/// Automatically generated headers for the various configuration groups derive
/// from this template and make available its member functions to access each
/// given configuration group.
template< typename T > struct ConfigurationGroup {
/// Get the read-only status of the given configuration group.
static bool isReadOnly(
css::uno::Reference<css::uno::XComponentContext> const & context
= css::uno::Reference<css::uno::XComponentContext>())
{
return detail::ConfigurationWrapper::get(context).isReadOnly(T::path());
}
/// Get read-only access to the given configuration group.
static css::uno::Reference<
css::container::XHierarchicalNameAccess >
get(css::uno::Reference<css::uno::XComponentContext> const & context
= css::uno::Reference<css::uno::XComponentContext>())
{
return detail::ConfigurationWrapper::get(context).getGroupReadOnly(
T::path());
}
/// Get read/write access to the given configuration group, storing any
/// modifications via the given changes batch.
static css::uno::Reference<
css::container::XHierarchicalNameReplace >
get(std::shared_ptr< ConfigurationChanges > const & batch)
{
return comphelper::detail::ConfigurationWrapper::getGroupReadWrite(
batch, T::path());
}
private:
ConfigurationGroup(const ConfigurationGroup&) = delete;
ConfigurationGroup& operator=(const ConfigurationGroup&) = delete;
ConfigurationGroup() = delete;
~ConfigurationGroup() = delete;
};
/// A type-safe wrapper around a configuration set.
///
/// Automatically generated headers for the various configuration sets derive
/// from this template and make available its member functions to access each
/// given configuration set.
template< typename T > struct ConfigurationSet {
/// Get the read-only status of the given configuration set.
static bool isReadOnly(
css::uno::Reference<css::uno::XComponentContext> const & context
= css::uno::Reference<css::uno::XComponentContext>())
{
return detail::ConfigurationWrapper::get(context).isReadOnly(T::path());
}
/// Get read-only access to the given configuration set.
static
css::uno::Reference< css::container::XNameAccess >
get(css::uno::Reference<css::uno::XComponentContext> const & context
= css::uno::Reference<css::uno::XComponentContext>())
{
return detail::ConfigurationWrapper::get(context).getSetReadOnly(
T::path());
}
/// Get read/write access to the given configuration set, storing any
/// modifications via the given changes batch.
static
css::uno::Reference< css::container::XNameContainer >
get(std::shared_ptr< ConfigurationChanges > const & batch)
{
return comphelper::detail::ConfigurationWrapper::getSetReadWrite(
batch, T::path());
}
private:
ConfigurationSet(const ConfigurationSet&) = delete;
ConfigurationSet& operator=(const ConfigurationSet&) = delete;
ConfigurationSet() = delete;
~ConfigurationSet() = delete;
};
}
#endif
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */