|           Line data    Source code 
       1             : /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
       2             : /*
       3             :  * This file is part of the LibreOffice project.
       4             :  *
       5             :  * This Source Code Form is subject to the terms of the Mozilla Public
       6             :  * License, v. 2.0. If a copy of the MPL was not distributed with this
       7             :  * file, You can obtain one at http://mozilla.org/MPL/2.0/.
       8             :  *
       9             :  * This file incorporates work covered by the following license notice:
      10             :  *
      11             :  *   Licensed to the Apache Software Foundation (ASF) under one or more
      12             :  *   contributor license agreements. See the NOTICE file distributed
      13             :  *   with this work for additional information regarding copyright
      14             :  *   ownership. The ASF licenses this file to you under the Apache
      15             :  *   License, Version 2.0 (the "License"); you may not use this file
      16             :  *   except in compliance with the License. You may obtain a copy of
      17             :  *   the License at http://www.apache.org/licenses/LICENSE-2.0 .
      18             :  */
      19             : 
      20             : #include <jobs/configaccess.hxx>
      21             : #include <general.h>
      22             : #include <services.h>
      23             : 
      24             : #include <com/sun/star/beans/XPropertySet.hpp>
      25             : #include <com/sun/star/beans/XMultiHierarchicalPropertySet.hpp>
      26             : #include <com/sun/star/configuration/theDefaultProvider.hpp>
      27             : #include <com/sun/star/container/XNameAccess.hpp>
      28             : #include <com/sun/star/beans/PropertyValue.hpp>
      29             : #include <com/sun/star/util/XChangesBatch.hpp>
      30             : 
      31             : #include <unotools/configpaths.hxx>
      32             : #include <rtl/ustrbuf.hxx>
      33             : 
      34             : namespace framework{
      35             : 
      36             : /**
      37             :     @short  open the configuration of this job
      38             :     @descr  We open the configuration of this job only. Not the whole package or the whole
      39             :             job set. We are interested on our own properties only.
      40             :             We set the opened configuration access as our member. So any following method,
      41             :             which needs cfg access, can use it. That prevent us against multiple open/close requests.
      42             :             But you can use this method to upgrade an already opened configuration too.
      43             : 
      44             :     @param  eMode
      45             :                 force opening of the configuration access in readonly or in read/write mode
      46             :  */
      47        2864 : ConfigAccess::ConfigAccess( /*IN*/ const css::uno::Reference< css::uno::XComponentContext >& rxContext,
      48             :                             /*IN*/ const OUString&                                    sRoot )
      49             :     : m_xContext    ( rxContext)
      50             :     , m_sRoot       ( sRoot    )
      51        2864 :     , m_eMode       ( E_CLOSED )
      52             : {
      53        2864 : }
      54             : 
      55             : /**
      56             :     @short  last chance to close an open configuration access point
      57             :     @descr  In case our user forgot to close this configuration point
      58             :             in the right way, normally he will run into some trouble -
      59             :             e.g. losing data.
      60             :  */
      61        5718 : ConfigAccess::~ConfigAccess()
      62             : {
      63        2859 :     close();
      64        2859 : }
      65             : 
      66             : /**
      67             :     @short  return the internal mode of this instance
      68             :     @descr  May be the outside user need any information about successfully opened
      69             :             or closed config access point objects. He can control the internal mode to do so.
      70             : 
      71             :     @return The internal open state of this object.
      72             :  */
      73        3435 : ConfigAccess::EOpenMode ConfigAccess::getMode() const
      74             : {
      75        3435 :     osl::MutexGuard g(m_mutex);
      76        3435 :     return m_eMode;
      77             : }
      78             : 
      79             : /**
      80             :     @short  open the configuration access in the specified mode
      81             :     @descr  We set the opened configuration access as our member. So any following method,
      82             :             which needs cfg access, can use it. That prevent us against multiple open/close requests.
      83             :             But you can use this method to upgrade an already opened configuration too.
      84             :             It's possible to open a config access in READONLY mode first and "open" it at a second
      85             :             time within the mode READWRITE. Then we will upgrade it. Dowgrade will be possible too.
      86             : 
      87             :             But note: closing will be done explicitly by calling method close() ... not by
      88             :             downgrading with mode CLOSED!
      89             : 
      90             :     @param  eMode
      91             :                 force (re)opening of the configuration access in readonly or in read/write mode
      92             :  */
      93        2864 : void ConfigAccess::open( /*IN*/ EOpenMode eMode )
      94             : {
      95        2864 :     osl::MutexGuard g(m_mutex);
      96             : 
      97             :     // check if configuration is already open in the right mode.
      98             :     // By the way: Don't allow closing by using this method!
      99        2864 :     if (
     100        2864 :         (eMode  !=E_CLOSED) &&
     101        2864 :         (m_eMode!=eMode   )
     102             :        )
     103             :     {
     104             :         // We have to close the old access point without any question here.
     105             :         // It will be open again using the new mode.
     106             :         // can be called without checks! It does the checks by itself ...
     107             :         // e.g. for already closed or not opened configuration.
     108             :         // Flushing of all made changes will be done here too.
     109        2864 :         close();
     110             : 
     111             :         // create the configuration provider, which provides sub access points
     112        2864 :         css::uno::Reference< css::lang::XMultiServiceFactory > xConfigProvider = css::configuration::theDefaultProvider::get(m_xContext);
     113        5728 :         css::beans::PropertyValue aParam;
     114        2864 :         aParam.Name    = "nodepath";
     115        2864 :         aParam.Value <<= m_sRoot;
     116             : 
     117        5728 :         css::uno::Sequence< css::uno::Any > lParams(1);
     118        2864 :         lParams[0] <<= aParam;
     119             : 
     120             :         // open it
     121             :         try
     122             :         {
     123        2864 :             if (eMode==E_READONLY)
     124        2864 :                 m_xConfig = xConfigProvider->createInstanceWithArguments(SERVICENAME_CFGREADACCESS  , lParams);
     125             :             else
     126           0 :             if (eMode==E_READWRITE)
     127           0 :                 m_xConfig = xConfigProvider->createInstanceWithArguments(SERVICENAME_CFGUPDATEACCESS, lParams);
     128             :         }
     129           0 :         catch(const css::uno::Exception& ex)
     130             :         {
     131             :             (void) ex; // avoid warning
     132             :             SAL_INFO("fwk", "open config: " << ex.Message);
     133             :         }
     134             : 
     135        2864 :         m_eMode = E_CLOSED;
     136        2864 :         if (m_xConfig.is())
     137        5728 :             m_eMode = eMode;
     138        2864 :     }
     139        2864 : }
     140             : 
     141             : /**
     142             :     @short  close the internal opened configuration access and flush all changes
     143             :     @descr  It checks, if the given access is valid and react in the right way.
     144             :             It flushes all changes ... so nobody else must know this state.
     145             :  */
     146        8583 : void ConfigAccess::close()
     147             : {
     148        8583 :     osl::MutexGuard g(m_mutex);
     149             :     // check already closed configuration
     150        8583 :     if (m_xConfig.is())
     151             :     {
     152        2860 :         css::uno::Reference< css::util::XChangesBatch > xFlush(m_xConfig, css::uno::UNO_QUERY);
     153        2860 :         if (xFlush.is())
     154           0 :             xFlush->commitChanges();
     155        2860 :         m_xConfig = css::uno::Reference< css::uno::XInterface >();
     156        2860 :         m_eMode   = E_CLOSED;
     157        8583 :     }
     158        8583 : }
     159             : 
     160             : /**
     161             :     @short  provides an access to the internal wrapped configuration access
     162             :     @descr  It's not allowed to safe this c++ (!) reference outside. You have
     163             :             to use it directly. Further you must use our public lock member m_aLock
     164             :             to synchronize your code with our internal structures and our interface
     165             :             methods. Acquire it before you call cfg() and release it afterwards immediately.
     166             : 
     167             :             E.g.:   ConfigAccess aAccess(...);
     168             :                     Guard aReadLock(aAccess.m_aLock);
     169             :                     Reference< XPropertySet > xSet(aAccess.cfg(), UNO_QUERY);
     170             :                     Any aProp = xSet->getPropertyValue("...");
     171             :                     aReadLock.unlock();
     172             : 
     173             :     @attention  During this time it's not allowed to call the methods open() or close()!
     174             :                 Otherwhise you will change your own referenced config access. Anything will
     175             :                 be possible then.
     176             : 
     177             :     @return A c++(!) reference to the uno instance of the configuration access point.
     178             :  */
     179        3440 : const css::uno::Reference< css::uno::XInterface >& ConfigAccess::cfg()
     180             : {
     181             :     // must be synchronized from outside!
     182             :     // => no lock here ...
     183        3440 :     return m_xConfig;
     184             : }
     185             : 
     186             : } // namespace framework
     187             : 
     188             : /* vim:set shiftwidth=4 softtabstop=4 expandtab: */
 |