1 /**************************************************************
2  *
3  * Licensed to the Apache Software Foundation (ASF) under one
4  * or more contributor license agreements.  See the NOTICE file
5  * distributed with this work for additional information
6  * regarding copyright ownership.  The ASF licenses this file
7  * to you under the Apache License, Version 2.0 (the
8  * "License"); you may not use this file except in compliance
9  * with the License.  You may obtain a copy of the License at
10  *
11  *   http://www.apache.org/licenses/LICENSE-2.0
12  *
13  * Unless required by applicable law or agreed to in writing,
14  * software distributed under the License is distributed on an
15  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16  * KIND, either express or implied.  See the License for the
17  * specific language governing permissions and limitations
18  * under the License.
19  *
20  *************************************************************/
21 
22 
23 
24 // MARKER(update_precomp.py): autogen include statement, do not remove
25 #include "precompiled_framework.hxx"
26 
27 #include "services/modulemanager.hxx"
28 #include "services/frame.hxx"
29 
30 //_______________________________________________
31 // own includes
32 #include <threadhelp/readguard.hxx>
33 #include <threadhelp/writeguard.hxx>
34 #include <services.h>
35 
36 //_______________________________________________
37 // interface includes
38 #include <com/sun/star/frame/XFrame.hpp>
39 #include <com/sun/star/frame/XController.hpp>
40 #include <com/sun/star/frame/XModel.hpp>
41 #include <com/sun/star/frame/XModule.hpp>
42 #include <comphelper/configurationhelper.hxx>
43 #include <comphelper/sequenceashashmap.hxx>
44 #include <comphelper/sequenceasvector.hxx>
45 #include <comphelper/enumhelper.hxx>
46 
47 //_______________________________________________
48 // other includes
49 #include <rtl/logfile.hxx>
50 
51 namespace framework
52 {
53 
54 static const ::rtl::OUString CFGPATH_FACTORIES     = ::rtl::OUString::createFromAscii("/org.openoffice.Setup/Office/Factories");
55 static const ::rtl::OUString MODULEPROP_IDENTIFIER = ::rtl::OUString::createFromAscii("ooSetupFactoryModuleIdentifier" );
56 
57 /*-----------------------------------------------
58     04.12.2003 09:32
59 -----------------------------------------------*/
60 DEFINE_XINTERFACE_7(ModuleManager                                    ,
61                     OWeakObject                                      ,
62                     DIRECT_INTERFACE(css::lang::XTypeProvider       ),
63                     DIRECT_INTERFACE(css::lang::XServiceInfo        ),
64                     DIRECT_INTERFACE(css::container::XNameReplace   ),
65                     DIRECT_INTERFACE(css::container::XNameAccess    ),
66                     DIRECT_INTERFACE(css::container::XElementAccess ),
67                     DIRECT_INTERFACE(css::container::XContainerQuery),
68                     DIRECT_INTERFACE(css::frame::XModuleManager     ))
69 
70 /*-----------------------------------------------
71     04.12.2003 09:32
72 -----------------------------------------------*/
73 DEFINE_XTYPEPROVIDER_7(ModuleManager                  ,
74                        css::lang::XTypeProvider       ,
75                        css::lang::XServiceInfo        ,
76                        css::container::XNameReplace   ,
77                        css::container::XNameAccess    ,
78                        css::container::XElementAccess ,
79                        css::container::XContainerQuery,
80                        css::frame::XModuleManager     )
81 
82 /*-----------------------------------------------
83     04.12.2003 09:35
84 -----------------------------------------------*/
85 DEFINE_XSERVICEINFO_ONEINSTANCESERVICE(ModuleManager                   ,
86                                        ::cppu::OWeakObject             ,
87                                        SERVICENAME_MODULEMANAGER       ,
88                                        IMPLEMENTATIONNAME_MODULEMANAGER)
89 
90 /*-----------------------------------------------
91     04.12.2003 09:35
92 -----------------------------------------------*/
93 DEFINE_INIT_SERVICE(
94                     ModuleManager,
95                     {
96                         /*Attention
97                             I think we don't need any mutex or lock here ... because we are called by our own static method impl_createInstance()
98                             to create a new instance of this class by our own supported service factory.
99                             see macro DEFINE_XSERVICEINFO_MULTISERVICE and "impl_initService()" for further informations!
100                         */
101                     }
102                    )
103 
104 /*-----------------------------------------------
105     04.12.2003 09:30
106 -----------------------------------------------*/
107 ModuleManager::ModuleManager(const css::uno::Reference< css::lang::XMultiServiceFactory >& xSMGR)
108     : ThreadHelpBase(     )
109     , m_xSMGR       (xSMGR)
110 {
111 }
112 
113 /*-----------------------------------------------
114     10.12.2003 11:59
115 -----------------------------------------------*/
116 ModuleManager::~ModuleManager()
117 {
118     if (m_xCFG.is())
119         m_xCFG.clear();
120 }
121 
122 /*-----------------------------------------------
123     10.12.2003 11:02
124 -----------------------------------------------*/
125 ::rtl::OUString SAL_CALL ModuleManager::identify(const css::uno::Reference< css::uno::XInterface >& xModule)
126     throw(css::lang::IllegalArgumentException,
127           css::frame::UnknownModuleException,
128           css::uno::RuntimeException         )
129 {
130     // valid parameter?
131     css::uno::Reference< css::frame::XFrame >      xFrame     (xModule, css::uno::UNO_QUERY);
132     css::uno::Reference< css::awt::XWindow >       xWindow    (xModule, css::uno::UNO_QUERY);
133     css::uno::Reference< css::frame::XController > xController(xModule, css::uno::UNO_QUERY);
134     css::uno::Reference< css::frame::XModel >      xModel     (xModule, css::uno::UNO_QUERY);
135 
136     if (
137         (!xFrame.is()     ) &&
138         (!xWindow.is()    ) &&
139         (!xController.is()) &&
140         (!xModel.is()     )
141        )
142     {
143         throw css::lang::IllegalArgumentException(
144                 ::rtl::OUString::createFromAscii("Given module is not a frame nor a window, controller or model."),
145                 static_cast< ::cppu::OWeakObject* >(this),
146                 1);
147     }
148 
149 	if (xFrame.is())
150     {
151 		xController = xFrame->getController();
152         xWindow     = xFrame->getComponentWindow();
153     }
154     if (xController.is())
155         xModel = xController->getModel();
156 
157     // modules are implemented by the deepest component in hierarchy ...
158     // Means: model -> controller -> window
159     // No fallbacks to higher components are allowed !
160     // Note : A frame provides access to module components only ... but it's not a module by himself.
161 
162     ::rtl::OUString sModule;
163     if (xModel.is())
164         sModule = implts_identify(xModel);
165     else
166     if (xController.is())
167         sModule = implts_identify(xController);
168     else
169     if (xWindow.is())
170         sModule = implts_identify(xWindow);
171 
172     if (sModule.getLength() < 1)
173         throw css::frame::UnknownModuleException(
174                 ::rtl::OUString::createFromAscii("Can't find suitable module for the given component."),
175                 static_cast< ::cppu::OWeakObject* >(this));
176 
177     return sModule;
178 }
179 
180 /*-----------------------------------------------
181     08.03.2007 09:55
182 -----------------------------------------------*/
183 void SAL_CALL ModuleManager::replaceByName(const ::rtl::OUString& sName ,
184                                            const css::uno::Any&   aValue)
185     throw (css::lang::IllegalArgumentException   ,
186            css::container::NoSuchElementException,
187            css::lang::WrappedTargetException     ,
188            css::uno::RuntimeException            )
189 {
190     ::comphelper::SequenceAsHashMap lProps(aValue);
191     if (lProps.empty() )
192     {
193         throw css::lang::IllegalArgumentException(
194                 ::rtl::OUString::createFromAscii("No properties given to replace part of module."),
195                 static_cast< css::container::XNameAccess* >(this),
196                 2);
197     }
198 
199     // SAFE -> ----------------------------------
200     ReadGuard aReadLock(m_aLock);
201     css::uno::Reference< css::lang::XMultiServiceFactory > xSMGR = m_xSMGR;
202     aReadLock.unlock();
203     // <- SAFE ----------------------------------
204 
205     // get access to the element
206     // Note: Dont use impl_getConfig() method here. Because it creates a readonly access only, further
207     // it cache it as a member of this module manager instance. If we change some props there ... but dont
208     // flush changes (because an error occurred) we will read them later. If we use a different config access
209     // we can close it without a flush ... and our read data wont be affected .-)
210     css::uno::Reference< css::uno::XInterface >         xCfg      = ::comphelper::ConfigurationHelper::openConfig(
211                                                                         xSMGR,
212                                                                         CFGPATH_FACTORIES,
213                                                                         ::comphelper::ConfigurationHelper::E_STANDARD);
214     css::uno::Reference< css::container::XNameAccess >  xModules (xCfg, css::uno::UNO_QUERY_THROW);
215     css::uno::Reference< css::container::XNameReplace > xModule  ;
216 
217     xModules->getByName(sName) >>= xModule;
218     if (!xModule.is())
219     {
220         throw css::uno::RuntimeException(
221                 ::rtl::OUString::createFromAscii("Was not able to get write access to the requested module entry inside configuration."),
222                 static_cast< css::container::XNameAccess* >(this));
223     }
224 
225     ::comphelper::SequenceAsHashMap::const_iterator pProp;
226     for (  pProp  = lProps.begin();
227            pProp != lProps.end()  ;
228          ++pProp                  )
229     {
230         const ::rtl::OUString& sPropName  = pProp->first;
231         const css::uno::Any&   aPropValue = pProp->second;
232 
233         // let "NoSuchElementException" out ! We support the same API ...
234         // and without a flush() at the end all changed data before will be ignored !
235         xModule->replaceByName(sPropName, aPropValue);
236     }
237 
238     ::comphelper::ConfigurationHelper::flush(xCfg);
239 }
240 
241 /*-----------------------------------------------
242     10.12.2003 12:05
243 -----------------------------------------------*/
244 css::uno::Any SAL_CALL ModuleManager::getByName(const ::rtl::OUString& sName)
245     throw(css::container::NoSuchElementException,
246           css::lang::WrappedTargetException     ,
247           css::uno::RuntimeException            )
248 {
249     // get access to the element
250     css::uno::Reference< css::container::XNameAccess > xCFG = implts_getConfig();
251     css::uno::Reference< css::container::XNameAccess > xModule;
252     xCFG->getByName(sName) >>= xModule;
253     if (!xModule.is())
254     {
255         throw css::uno::RuntimeException(
256                 ::rtl::OUString::createFromAscii("Was not able to get write access to the requested module entry inside configuration."),
257                 static_cast< css::container::XNameAccess* >(this));
258     }
259 
260     // convert it to seq< PropertyValue >
261     const css::uno::Sequence< ::rtl::OUString > lPropNames = xModule->getElementNames();
262           ::comphelper::SequenceAsHashMap       lProps     ;
263           sal_Int32                             c          = lPropNames.getLength();
264           sal_Int32                             i          = 0;
265 
266     lProps[MODULEPROP_IDENTIFIER] <<= sName;
267     for (i=0; i<c; ++i)
268     {
269         const ::rtl::OUString& sPropName         = lPropNames[i];
270                                lProps[sPropName] = xModule->getByName(sPropName);
271     }
272 
273     return css::uno::makeAny(lProps.getAsConstPropertyValueList());
274 }
275 
276 /*-----------------------------------------------
277     10.12.2003 11:58
278 -----------------------------------------------*/
279 css::uno::Sequence< ::rtl::OUString > SAL_CALL ModuleManager::getElementNames()
280     throw(css::uno::RuntimeException)
281 {
282     css::uno::Reference< css::container::XNameAccess > xCFG = implts_getConfig();
283     return xCFG->getElementNames();
284 }
285 
286 /*-----------------------------------------------
287     10.12.2003 11:57
288 -----------------------------------------------*/
289 sal_Bool SAL_CALL ModuleManager::hasByName(const ::rtl::OUString& sName)
290     throw(css::uno::RuntimeException)
291 {
292     css::uno::Reference< css::container::XNameAccess > xCFG = implts_getConfig();
293     return xCFG->hasByName(sName);
294 }
295 
296 /*-----------------------------------------------
297     10.12.2003 11:35
298 -----------------------------------------------*/
299 css::uno::Type SAL_CALL ModuleManager::getElementType()
300     throw(css::uno::RuntimeException)
301 {
302     return ::getCppuType((const css::uno::Sequence< css::beans::PropertyValue >*)0);
303 }
304 
305 /*-----------------------------------------------
306     10.12.2003 11:56
307 -----------------------------------------------*/
308 sal_Bool SAL_CALL ModuleManager::hasElements()
309     throw(css::uno::RuntimeException)
310 {
311     css::uno::Reference< css::container::XNameAccess > xCFG = implts_getConfig();
312     return xCFG->hasElements();
313 }
314 
315 /*-----------------------------------------------
316     07.03.2007 12:55
317 -----------------------------------------------*/
318 css::uno::Reference< css::container::XEnumeration > SAL_CALL ModuleManager::createSubSetEnumerationByQuery(const ::rtl::OUString&)
319     throw(css::uno::RuntimeException)
320 {
321     return css::uno::Reference< css::container::XEnumeration >();
322 }
323 
324 /*-----------------------------------------------
325     07.03.2007 12:55
326 -----------------------------------------------*/
327 css::uno::Reference< css::container::XEnumeration > SAL_CALL ModuleManager::createSubSetEnumerationByProperties(const css::uno::Sequence< css::beans::NamedValue >& lProperties)
328     throw(css::uno::RuntimeException)
329 {
330     ::comphelper::SequenceAsHashMap                 lSearchProps (lProperties);
331     css::uno::Sequence< ::rtl::OUString >           lModules     = getElementNames();
332     sal_Int32                                       c            = lModules.getLength();
333     sal_Int32                                       i            = 0;
334     ::comphelper::SequenceAsVector< css::uno::Any > lResult      ;
335 
336     for (i=0; i<c; ++i)
337     {
338         try
339         {
340             const ::rtl::OUString&                sModule      = lModules[i];
341                   ::comphelper::SequenceAsHashMap lModuleProps = getByName(sModule);
342 
343             if (lModuleProps.match(lSearchProps))
344                 lResult.push_back(css::uno::makeAny(lModuleProps.getAsConstPropertyValueList()));
345         }
346         catch(const css::uno::Exception&)
347             {}
348     }
349 
350     ::comphelper::OAnyEnumeration*                      pEnum = new ::comphelper::OAnyEnumeration(lResult.getAsConstList());
351     css::uno::Reference< css::container::XEnumeration > xEnum(static_cast< css::container::XEnumeration* >(pEnum), css::uno::UNO_QUERY_THROW);
352     return xEnum;
353 }
354 
355 /*-----------------------------------------------
356     14.12.2003 09:45
357 -----------------------------------------------*/
358 css::uno::Reference< css::container::XNameAccess > ModuleManager::implts_getConfig()
359     throw(css::uno::RuntimeException)
360 {
361     // SAFE -> ----------------------------------
362     ReadGuard aReadLock(m_aLock);
363     if (m_xCFG.is())
364         return m_xCFG;
365     css::uno::Reference< css::lang::XMultiServiceFactory > xSMGR = m_xSMGR;
366     aReadLock.unlock();
367     // <- SAFE ----------------------------------
368 
369     css::uno::Reference< css::uno::XInterface > xCfg;
370     try
371     {
372         xCfg = ::comphelper::ConfigurationHelper::openConfig(
373                     xSMGR,
374                     CFGPATH_FACTORIES,
375                     ::comphelper::ConfigurationHelper::E_READONLY);
376     }
377     catch(const css::uno::RuntimeException& exRun)
378         { throw exRun; }
379     catch(const css::uno::Exception&)
380         { xCfg.clear(); }
381 
382     // SAFE -> ----------------------------------
383     WriteGuard aWriteLock(m_aLock);
384     m_xCFG = css::uno::Reference< css::container::XNameAccess >(xCfg, css::uno::UNO_QUERY_THROW);
385     return m_xCFG;
386     // <- SAFE ----------------------------------
387 }
388 
389 /*-----------------------------------------------
390     30.01.2004 07:54
391 -----------------------------------------------*/
392 ::rtl::OUString ModuleManager::implts_identify(const css::uno::Reference< css::uno::XInterface >& xComponent)
393 {
394     // Search for an optional (!) interface XModule first.
395     // Its used to overrule an existing service name. Used e.g. by our database form designer
396     // which uses a writer module internaly.
397     css::uno::Reference< css::frame::XModule > xModule(xComponent, css::uno::UNO_QUERY);
398     if (xModule.is())
399         return xModule->getIdentifier();
400 
401     // detect modules in a generic way ...
402     // comparing service names with configured entries ...
403     css::uno::Reference< css::lang::XServiceInfo > xInfo(xComponent, css::uno::UNO_QUERY);
404     if (!xInfo.is())
405         return ::rtl::OUString();
406 
407     const css::uno::Sequence< ::rtl::OUString > lKnownModules = getElementNames();
408     const ::rtl::OUString*                      pKnownModules = lKnownModules.getConstArray();
409           sal_Int32                             c             = lKnownModules.getLength();
410           sal_Int32                             i             = 0;
411 
412     for (i=0; i<c; ++i)
413     {
414         if (xInfo->supportsService(pKnownModules[i]))
415             return pKnownModules[i];
416     }
417 
418     return ::rtl::OUString();
419 }
420 
421 } // namespace framework
422