1*b1cdbd2cSJim Jagielski /**************************************************************
2*b1cdbd2cSJim Jagielski  *
3*b1cdbd2cSJim Jagielski  * Licensed to the Apache Software Foundation (ASF) under one
4*b1cdbd2cSJim Jagielski  * or more contributor license agreements.  See the NOTICE file
5*b1cdbd2cSJim Jagielski  * distributed with this work for additional information
6*b1cdbd2cSJim Jagielski  * regarding copyright ownership.  The ASF licenses this file
7*b1cdbd2cSJim Jagielski  * to you under the Apache License, Version 2.0 (the
8*b1cdbd2cSJim Jagielski  * "License"); you may not use this file except in compliance
9*b1cdbd2cSJim Jagielski  * with the License.  You may obtain a copy of the License at
10*b1cdbd2cSJim Jagielski  *
11*b1cdbd2cSJim Jagielski  *   http://www.apache.org/licenses/LICENSE-2.0
12*b1cdbd2cSJim Jagielski  *
13*b1cdbd2cSJim Jagielski  * Unless required by applicable law or agreed to in writing,
14*b1cdbd2cSJim Jagielski  * software distributed under the License is distributed on an
15*b1cdbd2cSJim Jagielski  * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
16*b1cdbd2cSJim Jagielski  * KIND, either express or implied.  See the License for the
17*b1cdbd2cSJim Jagielski  * specific language governing permissions and limitations
18*b1cdbd2cSJim Jagielski  * under the License.
19*b1cdbd2cSJim Jagielski  *
20*b1cdbd2cSJim Jagielski  *************************************************************/
21*b1cdbd2cSJim Jagielski 
22*b1cdbd2cSJim Jagielski 
23*b1cdbd2cSJim Jagielski 
24*b1cdbd2cSJim Jagielski // MARKER(update_precomp.py): autogen include statement, do not remove
25*b1cdbd2cSJim Jagielski #include "precompiled_scripting.hxx"
26*b1cdbd2cSJim Jagielski #include <cppuhelper/implementationentry.hxx>
27*b1cdbd2cSJim Jagielski #include <cppuhelper/factory.hxx>
28*b1cdbd2cSJim Jagielski #include <tools/diagnose_ex.h>
29*b1cdbd2cSJim Jagielski 
30*b1cdbd2cSJim Jagielski #include <util/scriptingconstants.hxx>
31*b1cdbd2cSJim Jagielski #include <util/util.hxx>
32*b1cdbd2cSJim Jagielski 
33*b1cdbd2cSJim Jagielski #include <com/sun/star/container/XContentEnumerationAccess.hpp>
34*b1cdbd2cSJim Jagielski #include "ProviderCache.hxx"
35*b1cdbd2cSJim Jagielski 
36*b1cdbd2cSJim Jagielski using namespace com::sun::star;
37*b1cdbd2cSJim Jagielski using namespace com::sun::star::uno;
38*b1cdbd2cSJim Jagielski using namespace com::sun::star::script;
39*b1cdbd2cSJim Jagielski 
40*b1cdbd2cSJim Jagielski namespace func_provider
41*b1cdbd2cSJim Jagielski {
42*b1cdbd2cSJim Jagielski 
ProviderCache(const Reference<XComponentContext> & xContext,const Sequence<Any> & scriptContext)43*b1cdbd2cSJim Jagielski ProviderCache::ProviderCache( const Reference< XComponentContext >& xContext, const Sequence< Any >& scriptContext )
44*b1cdbd2cSJim Jagielski     throw ( RuntimeException ) : m_Sctx( scriptContext ), m_xContext( xContext )
45*b1cdbd2cSJim Jagielski {
46*b1cdbd2cSJim Jagielski     // initialise m_hProviderDetailsCache with details of ScriptProviders
47*b1cdbd2cSJim Jagielski     // will use createContentEnumeration
48*b1cdbd2cSJim Jagielski 
49*b1cdbd2cSJim Jagielski     m_xMgr = m_xContext->getServiceManager();
50*b1cdbd2cSJim Jagielski     ENSURE_OR_THROW( m_xMgr.is(), "ProviderCache::ProviderCache() failed to obtain ServiceManager" );
51*b1cdbd2cSJim Jagielski     populateCache();
52*b1cdbd2cSJim Jagielski }
53*b1cdbd2cSJim Jagielski 
54*b1cdbd2cSJim Jagielski 
ProviderCache(const Reference<XComponentContext> & xContext,const Sequence<Any> & scriptContext,const Sequence<::rtl::OUString> & blackList)55*b1cdbd2cSJim Jagielski ProviderCache::ProviderCache( const Reference< XComponentContext >& xContext, const Sequence< Any >& scriptContext, const Sequence< ::rtl::OUString >& blackList )
56*b1cdbd2cSJim Jagielski     throw ( RuntimeException ) : m_sBlackList( blackList ), m_Sctx( scriptContext ), m_xContext( xContext )
57*b1cdbd2cSJim Jagielski 
58*b1cdbd2cSJim Jagielski {
59*b1cdbd2cSJim Jagielski     // initialise m_hProviderDetailsCache with details of ScriptProviders
60*b1cdbd2cSJim Jagielski     // will use createContentEnumeration
61*b1cdbd2cSJim Jagielski 
62*b1cdbd2cSJim Jagielski     m_xMgr = m_xContext->getServiceManager();
63*b1cdbd2cSJim Jagielski     ENSURE_OR_THROW( m_xMgr.is(), "ProviderCache::ProviderCache() failed to obtain ServiceManager" );
64*b1cdbd2cSJim Jagielski     populateCache();
65*b1cdbd2cSJim Jagielski }
66*b1cdbd2cSJim Jagielski 
~ProviderCache()67*b1cdbd2cSJim Jagielski ProviderCache::~ProviderCache()
68*b1cdbd2cSJim Jagielski {
69*b1cdbd2cSJim Jagielski }
70*b1cdbd2cSJim Jagielski 
71*b1cdbd2cSJim Jagielski Reference< provider::XScriptProvider >
getProvider(const::rtl::OUString & providerName)72*b1cdbd2cSJim Jagielski ProviderCache::getProvider( const ::rtl::OUString& providerName )
73*b1cdbd2cSJim Jagielski {
74*b1cdbd2cSJim Jagielski     ::osl::Guard< osl::Mutex > aGuard( m_mutex );
75*b1cdbd2cSJim Jagielski     Reference< provider::XScriptProvider > provider;
76*b1cdbd2cSJim Jagielski     ProviderDetails_hash::iterator h_it = m_hProviderDetailsCache.find( providerName );
77*b1cdbd2cSJim Jagielski     if ( h_it != m_hProviderDetailsCache.end() )
78*b1cdbd2cSJim Jagielski     {
79*b1cdbd2cSJim Jagielski         if (  h_it->second.provider.is() )
80*b1cdbd2cSJim Jagielski         {
81*b1cdbd2cSJim Jagielski             provider = h_it->second.provider;
82*b1cdbd2cSJim Jagielski         }
83*b1cdbd2cSJim Jagielski 	else
84*b1cdbd2cSJim Jagielski 	{
85*b1cdbd2cSJim Jagielski 	    // need to create provider and insert into hash
86*b1cdbd2cSJim Jagielski             provider = createProvider( h_it->second );
87*b1cdbd2cSJim Jagielski 	}
88*b1cdbd2cSJim Jagielski     }
89*b1cdbd2cSJim Jagielski     return provider;
90*b1cdbd2cSJim Jagielski }
91*b1cdbd2cSJim Jagielski 
92*b1cdbd2cSJim Jagielski Sequence < Reference< provider::XScriptProvider > >
getAllProviders()93*b1cdbd2cSJim Jagielski ProviderCache::getAllProviders() throw ( RuntimeException )
94*b1cdbd2cSJim Jagielski {
95*b1cdbd2cSJim Jagielski     Sequence < Reference< provider::XScriptProvider > > providers (  m_hProviderDetailsCache.size() );
96*b1cdbd2cSJim Jagielski     // need to create providers that haven't been created already
97*b1cdbd2cSJim Jagielski     // so check what providers exist and what ones don't
98*b1cdbd2cSJim Jagielski 
99*b1cdbd2cSJim Jagielski     ::osl::Guard< osl::Mutex > aGuard( m_mutex );
100*b1cdbd2cSJim Jagielski     ProviderDetails_hash::iterator h_itEnd =  m_hProviderDetailsCache.end();
101*b1cdbd2cSJim Jagielski     ProviderDetails_hash::iterator h_it = m_hProviderDetailsCache.begin();
102*b1cdbd2cSJim Jagielski     // should assert if size !>  0
103*b1cdbd2cSJim Jagielski     if (  m_hProviderDetailsCache.size() )
104*b1cdbd2cSJim Jagielski     {
105*b1cdbd2cSJim Jagielski         sal_Int32 providerIndex = 0;
106*b1cdbd2cSJim Jagielski 	sal_Int32 index = 0;
107*b1cdbd2cSJim Jagielski         for ( index = 0; h_it !=  h_itEnd; ++h_it, index++ )
108*b1cdbd2cSJim Jagielski         {
109*b1cdbd2cSJim Jagielski             Reference< provider::XScriptProvider > xScriptProvider  = h_it->second.provider;
110*b1cdbd2cSJim Jagielski             if ( xScriptProvider.is() )
111*b1cdbd2cSJim Jagielski             {
112*b1cdbd2cSJim Jagielski                 providers[ providerIndex++ ] = xScriptProvider;
113*b1cdbd2cSJim Jagielski             }
114*b1cdbd2cSJim Jagielski             else
115*b1cdbd2cSJim Jagielski             {
116*b1cdbd2cSJim Jagielski                 // create provider
117*b1cdbd2cSJim Jagielski                 try
118*b1cdbd2cSJim Jagielski                 {
119*b1cdbd2cSJim Jagielski                     xScriptProvider  = createProvider( h_it->second );
120*b1cdbd2cSJim Jagielski                     providers[ providerIndex++ ] = xScriptProvider;
121*b1cdbd2cSJim Jagielski                 }
122*b1cdbd2cSJim Jagielski                 catch ( Exception& e )
123*b1cdbd2cSJim Jagielski                 {
124*b1cdbd2cSJim Jagielski                     ::rtl::OUString temp = OUSTR( "ProviderCache::getAllProviders: failed to create provider, " );
125*b1cdbd2cSJim Jagielski                     temp.concat( e.Message );
126*b1cdbd2cSJim Jagielski                     //throw RuntimeException( temp.concat( e.Message ),
127*b1cdbd2cSJim Jagielski                     //    Reference< XInterface >() );
128*b1cdbd2cSJim Jagielski                 }
129*b1cdbd2cSJim Jagielski             }
130*b1cdbd2cSJim Jagielski         }
131*b1cdbd2cSJim Jagielski 
132*b1cdbd2cSJim Jagielski         if ( providerIndex < index )
133*b1cdbd2cSJim Jagielski         {
134*b1cdbd2cSJim Jagielski             providers.realloc( providerIndex );
135*b1cdbd2cSJim Jagielski         }
136*b1cdbd2cSJim Jagielski 
137*b1cdbd2cSJim Jagielski     }
138*b1cdbd2cSJim Jagielski     else
139*b1cdbd2cSJim Jagielski     {
140*b1cdbd2cSJim Jagielski         OSL_TRACE("no available providers, something very wrong!!!");
141*b1cdbd2cSJim Jagielski     }
142*b1cdbd2cSJim Jagielski     return providers;
143*b1cdbd2cSJim Jagielski }
144*b1cdbd2cSJim Jagielski 
145*b1cdbd2cSJim Jagielski void
populateCache()146*b1cdbd2cSJim Jagielski ProviderCache::populateCache() throw ( RuntimeException )
147*b1cdbd2cSJim Jagielski {
148*b1cdbd2cSJim Jagielski     // wrong name in services.rdb
149*b1cdbd2cSJim Jagielski     ::rtl::OUString serviceName;
150*b1cdbd2cSJim Jagielski     ::osl::Guard< osl::Mutex > aGuard( m_mutex );
151*b1cdbd2cSJim Jagielski     try
152*b1cdbd2cSJim Jagielski     {
153*b1cdbd2cSJim Jagielski         ::rtl::OUString languageProviderName( RTL_CONSTASCII_USTRINGPARAM(
154*b1cdbd2cSJim Jagielski             "com.sun.star.script.provider.LanguageScriptProvider" ) );
155*b1cdbd2cSJim Jagielski 
156*b1cdbd2cSJim Jagielski         Reference< container::XContentEnumerationAccess > xEnumAccess = Reference< container::XContentEnumerationAccess >( m_xMgr, UNO_QUERY_THROW );
157*b1cdbd2cSJim Jagielski         Reference< container::XEnumeration > xEnum = xEnumAccess->createContentEnumeration ( languageProviderName );
158*b1cdbd2cSJim Jagielski 
159*b1cdbd2cSJim Jagielski         while ( xEnum->hasMoreElements() )
160*b1cdbd2cSJim Jagielski         {
161*b1cdbd2cSJim Jagielski 
162*b1cdbd2cSJim Jagielski             Reference< lang::XSingleComponentFactory > factory( xEnum->nextElement(), UNO_QUERY_THROW );
163*b1cdbd2cSJim Jagielski             Reference< lang::XServiceInfo > xServiceInfo( factory, UNO_QUERY_THROW );
164*b1cdbd2cSJim Jagielski 
165*b1cdbd2cSJim Jagielski             Sequence< ::rtl::OUString > serviceNames = xServiceInfo->getSupportedServiceNames();
166*b1cdbd2cSJim Jagielski 
167*b1cdbd2cSJim Jagielski             if ( serviceNames.getLength() > 0 )
168*b1cdbd2cSJim Jagielski             {
169*b1cdbd2cSJim Jagielski                 ::rtl::OUString searchString( RTL_CONSTASCII_USTRINGPARAM (
170*b1cdbd2cSJim Jagielski                     "com.sun.star.script.provider.ScriptProviderFor" ) );
171*b1cdbd2cSJim Jagielski 
172*b1cdbd2cSJim Jagielski                 for ( sal_Int32 index = 0; index < serviceNames.getLength(); index++ )
173*b1cdbd2cSJim Jagielski                 {
174*b1cdbd2cSJim Jagielski                     if ( serviceNames[ index ].indexOf( searchString ) == 0 && !isInBlackList(  serviceNames[ index ] ) )
175*b1cdbd2cSJim Jagielski                     {
176*b1cdbd2cSJim Jagielski                         serviceName = serviceNames[ index ];
177*b1cdbd2cSJim Jagielski                         ProviderDetails details;
178*b1cdbd2cSJim Jagielski                         details.factory = factory;
179*b1cdbd2cSJim Jagielski                         m_hProviderDetailsCache[ serviceName ] = details;
180*b1cdbd2cSJim Jagielski                         break;
181*b1cdbd2cSJim Jagielski                     }
182*b1cdbd2cSJim Jagielski                 }
183*b1cdbd2cSJim Jagielski             }
184*b1cdbd2cSJim Jagielski         }
185*b1cdbd2cSJim Jagielski     }
186*b1cdbd2cSJim Jagielski     catch ( Exception e )
187*b1cdbd2cSJim Jagielski     {
188*b1cdbd2cSJim Jagielski         ::rtl::OUString temp = OUSTR(
189*b1cdbd2cSJim Jagielski             "ProviderCache::populateCache: couldn't obtain XSingleComponentFactory for " );
190*b1cdbd2cSJim Jagielski         temp.concat( serviceName );
191*b1cdbd2cSJim Jagielski         throw RuntimeException( temp.concat( e.Message ), Reference< XInterface >() );
192*b1cdbd2cSJim Jagielski     }
193*b1cdbd2cSJim Jagielski }
194*b1cdbd2cSJim Jagielski 
195*b1cdbd2cSJim Jagielski Reference< provider::XScriptProvider >
createProvider(ProviderDetails & details)196*b1cdbd2cSJim Jagielski ProviderCache::createProvider( ProviderDetails& details ) throw ( RuntimeException )
197*b1cdbd2cSJim Jagielski {
198*b1cdbd2cSJim Jagielski     try
199*b1cdbd2cSJim Jagielski     {
200*b1cdbd2cSJim Jagielski         details.provider.set(
201*b1cdbd2cSJim Jagielski             details.factory->createInstanceWithArgumentsAndContext( m_Sctx, m_xContext ), UNO_QUERY_THROW );
202*b1cdbd2cSJim Jagielski     }
203*b1cdbd2cSJim Jagielski     catch ( Exception& e )
204*b1cdbd2cSJim Jagielski     {
205*b1cdbd2cSJim Jagielski         ::rtl::OUString temp = ::rtl::OUString::createFromAscii("ProviderCache::createProvider() Error creating provider from factory!!!");
206*b1cdbd2cSJim Jagielski         throw RuntimeException( temp.concat( e.Message ), Reference< XInterface >() );
207*b1cdbd2cSJim Jagielski     }
208*b1cdbd2cSJim Jagielski 
209*b1cdbd2cSJim Jagielski     return details.provider;
210*b1cdbd2cSJim Jagielski }
211*b1cdbd2cSJim Jagielski } //end namespace
212