1 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
3 * This file is part of the LibreOffice project.
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/.
9 * This file incorporates work covered by the following license notice:
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 .
20 #include <sal/config.h>
22 #include <string_view>
24 #include <config_folders.h>
26 #include <vcl/svapp.hxx>
27 #include <vcl/weld.hxx>
28 #include <rtl/bootstrap.hxx>
29 #include <rtl/ustrbuf.hxx>
30 #include <sal/log.hxx>
31 #include <osl/process.h>
32 #include <osl/file.hxx>
33 #include <unotools/configmgr.hxx>
34 #include <unotools/bootstrap.hxx>
35 #include <cppuhelper/bootstrap.hxx>
36 #include <comphelper/sequence.hxx>
37 #include <comphelper/processfactory.hxx>
39 #include <com/sun/star/lang/XMultiServiceFactory.hpp>
40 #include <com/sun/star/ucb/UniversalContentBroker.hpp>
42 #include <strings.hrc>
43 #include "unopkg_shared.h"
44 #include <dp_identifier.hxx>
46 #include <dp_shared.hxx>
47 #include <lockfile.hxx>
49 using namespace ::com::sun::star
;
50 using namespace ::com::sun::star::uno
;
51 using namespace ::com::sun::star::ucb
;
55 OUString
toString( OptionInfo
const * info
)
57 assert(info
!= nullptr);
58 OUStringBuffer
buf("--");
59 buf
.appendAscii(info
->m_name
);
60 if (info
->m_short_option
!= '\0')
62 buf
.append(" (short -" + OUStringChar(info
->m_short_option
) + ")");
64 if (info
->m_has_argument
)
65 buf
.append(" <argument>" );
66 return buf
.makeStringAndClear();
70 OptionInfo
const * getOptionInfo(
71 OptionInfo
const * list
,
72 OUString
const & opt
)
74 for ( ; list
->m_name
!= nullptr; ++list
)
76 OptionInfo
const & option_info
= *list
;
80 option_info
.m_name
, option_info
.m_name_length
))
86 SAL_WARN( "desktop", opt
);
91 bool isOption( OptionInfo
const * option_info
, sal_uInt32
* pIndex
)
93 assert(option_info
!= nullptr);
94 if (osl_getCommandArgCount() <= *pIndex
)
98 osl_getCommandArg( *pIndex
, &arg
.pData
);
99 sal_Int32 len
= arg
.getLength();
101 if (len
< 2 || arg
[ 0 ] != '-')
104 if (len
== 2 && arg
[ 1 ] == option_info
->m_short_option
)
107 dp_misc::TRACE(__FILE__
": identified option \'\'"
108 + OUStringChar( option_info
->m_short_option
) + "\n");
111 if (arg
[ 1 ] == '-' && rtl_ustr_ascii_compare(
112 arg
.pData
->buffer
+ 2, option_info
->m_name
) == 0)
115 dp_misc::TRACE(__FILE__
": identified option \'"
116 + OUString::createFromAscii(option_info
->m_name
) + "\'\n");
123 bool isBootstrapVariable(sal_uInt32
* pIndex
)
125 OSL_ASSERT(osl_getCommandArgCount() >= *pIndex
);
128 osl_getCommandArg(*pIndex
, &arg
.pData
);
129 if (arg
.match("-env:"))
139 OUString
* pValue
, OptionInfo
const * option_info
, sal_uInt32
* pIndex
)
141 if (isOption( option_info
, pIndex
))
143 if (*pIndex
< osl_getCommandArgCount())
145 OSL_ASSERT( pValue
!= nullptr );
146 osl_getCommandArg( *pIndex
, &pValue
->pData
);
147 dp_misc::TRACE(__FILE__
": argument value: "
158 OUString
const & getExecutableDir()
160 static const OUString EXEC
=
164 if (osl_getExecutableFile( &path
.pData
) != osl_Process_E_None
) {
165 throw RuntimeException("cannot locate executable directory!",nullptr);
167 return path
.copy( 0, path
.lastIndexOf( '/' ) );
173 OUString
const & getProcessWorkingDir()
175 static const OUString WORKING
=
179 utl::Bootstrap::getProcessWorkingDir(workingDir
);
186 OUString
makeAbsoluteFileUrl(
187 OUString
const & sys_path
, OUString
const & base_url
)
189 // system path to file url
191 oslFileError rc
= osl_getFileURLFromSystemPath( sys_path
.pData
, &file_url
.pData
);
192 if ( rc
!= osl_File_E_None
) {
194 if ( osl_getSystemPathFromFileURL( sys_path
.pData
, &tempPath
.pData
) != osl_File_E_None
)
196 throw RuntimeException("cannot get file url from system path: " +
203 if (osl_getAbsoluteFileURL(
204 base_url
.pData
, file_url
.pData
, &abs
.pData
) != osl_File_E_None
)
206 throw RuntimeException(
207 "making absolute file url failed: \"" + base_url
208 + "\" (base-url) and \"" + file_url
+ "\" (file-url)!" );
210 return abs
[ abs
.getLength() -1 ] == '/'
211 ? abs
.copy( 0, abs
.getLength() -1 ) : abs
;
218 void printf_space( sal_Int32 space
)
221 dp_misc::writeConsole(u
" ");
226 std::u16string_view name
, std::u16string_view value
, sal_Int32 level
)
228 printf_space( level
);
229 dp_misc::writeConsole(Concat2View(OUString::Concat(name
) + ": " + value
+ "\n"));
234 Reference
<deployment::XPackage
> const & xPackage
,
235 Reference
<XCommandEnvironment
> const & xCmdEnv
, sal_Int32 level
)
237 beans::Optional
< OUString
> id(
239 ? beans::Optional
< OUString
>(
240 true, dp_misc::getIdentifier( xPackage
) )
241 : xPackage
->getIdentifier() );
243 printf_line( u
"Identifier", id
.Value
, level
);
244 OUString
version(xPackage
->getVersion());
245 if (!version
.isEmpty())
246 printf_line( u
"Version", version
, level
+ 1 );
247 printf_line( u
"URL", xPackage
->getURL(), level
+ 1 );
249 beans::Optional
< beans::Ambiguous
<sal_Bool
> > option(
250 xPackage
->isRegistered( Reference
<task::XAbortChannel
>(), xCmdEnv
) );
252 if (option
.IsPresent
) {
253 beans::Ambiguous
<sal_Bool
> const & reg
= option
.Value
;
257 value
= reg
.Value
? std::u16string_view(u
"yes") : std::u16string_view(u
"no");
261 printf_line( u
"is registered", value
, level
+ 1 );
263 const Reference
<deployment::XPackageTypeInfo
> xPackageType(
264 xPackage
->getPackageType() );
265 OSL_ASSERT( xPackageType
.is() );
266 if (xPackageType
.is()) {
267 printf_line( u
"Media-Type", xPackageType
->getMediaType(), level
+ 1 );
269 printf_line( u
"Description", xPackage
->getDescription(), level
+ 1 );
270 if (!xPackage
->isBundle())
273 Sequence
< Reference
<deployment::XPackage
> > seq(
274 xPackage
->getBundle( Reference
<task::XAbortChannel
>(), xCmdEnv
) );
275 printf_space( level
+ 1 );
276 dp_misc::writeConsole(u
"bundled Packages: {\n");
277 std::vector
<Reference
<deployment::XPackage
> >vec_bundle
;
278 ::comphelper::sequenceToContainer(vec_bundle
, seq
);
279 printf_packages( vec_bundle
, std::vector
<bool>(vec_bundle
.size()),
280 xCmdEnv
, level
+ 2 );
281 printf_space( level
+ 1 );
282 dp_misc::writeConsole(u
"}\n");
287 static void printf_unaccepted_licenses(
288 Reference
<deployment::XPackage
> const & ext
)
291 dp_misc::getIdentifier(ext
) );
292 printf_line( u
"Identifier", id
, 0 );
294 dp_misc::writeConsole(u
"License not accepted\n\n");
298 void printf_packages(
299 std::vector
< Reference
<deployment::XPackage
> > const & allExtensions
,
300 std::vector
<bool> const & vecUnaccepted
,
301 Reference
<XCommandEnvironment
> const & xCmdEnv
, sal_Int32 level
)
303 OSL_ASSERT(allExtensions
.size() == vecUnaccepted
.size());
305 if (allExtensions
.empty())
307 printf_space( level
);
308 dp_misc::writeConsole(u
"<none>\n");
313 for (auto const& extension
: allExtensions
)
315 if (vecUnaccepted
[index
])
316 printf_unaccepted_licenses(extension
);
318 printf_package( extension
, xCmdEnv
, level
);
319 dp_misc::writeConsole(u
"\n");
329 Reference
<XComponentContext
> bootstrapStandAlone()
331 Reference
<XComponentContext
> xContext
=
332 ::cppu::defaultBootstrap_InitialComponentContext();
334 Reference
<lang::XMultiServiceFactory
> xServiceManager(
335 xContext
->getServiceManager(), UNO_QUERY_THROW
);
336 // set global process service factory used by unotools config helpers
337 ::comphelper::setProcessServiceFactory( xServiceManager
);
339 // Initialize the UCB (for backwards compatibility, in case some code still
340 // uses plain createInstance w/o args directly to obtain an instance):
341 UniversalContentBroker::create( xContext
);
347 Reference
<XComponentContext
> connectToOffice(
348 Reference
<XComponentContext
> const & xLocalComponentContext
,
351 OUString
pipeId( ::dp_misc::generateRandomPipeId() );
352 OUString acceptArg
= "--accept=pipe,name=" + pipeId
+ ";urp;";
354 Sequence
<OUString
> args
{ "--nologo", "--nodefault", acceptArg
};
355 OUString
appURL( getExecutableDir() + "/soffice" );
359 dp_misc::writeConsole(Concat2View(
360 "Raising process: " + appURL
+
361 "\nArguments: --nologo --nodefault " + args
[2] +
365 ::dp_misc::raiseProcess( appURL
, args
);
368 dp_misc::writeConsole(u
"OK. Connecting...");
370 OUString sUnoUrl
= "uno:pipe,name=" + pipeId
+ ";urp;StarOffice.ComponentContext";
371 Reference
<XComponentContext
> xRet(
372 ::dp_misc::resolveUnoURL(
373 sUnoUrl
, xLocalComponentContext
),
376 dp_misc::writeConsole(u
"OK.\n");
383 /** returns the path to the lock file used by unopkg.
384 @return the path. An empty string signifies an error.
386 static OUString
getLockFilePath()
389 OUString
sBootstrap("${$BRAND_BASE_DIR/" LIBO_ETC_FOLDER
"/" SAL_CONFIGFILE("bootstrap") ":UserInstallation}");
390 rtl::Bootstrap::expandMacros(sBootstrap
);
392 if (::osl::File::E_None
== ::osl::File::getAbsoluteFileURL(
393 sBootstrap
, ".lock", sAbs
))
395 if (::osl::File::E_None
==
396 ::osl::File::getSystemPathFromFileURL(sAbs
, sBootstrap
))
405 Reference
<XComponentContext
> getUNO(
406 bool verbose
, bool bGui
, const OUString
& sTempDir
,
407 Reference
<XComponentContext
> & out_localContext
)
409 // do not create any user data (for the root user) in --shared mode:
410 if (!sTempDir
.isEmpty())
411 rtl::Bootstrap::set("UserInstallation", sTempDir
);
413 // hold lock during process runtime:
414 static ::desktop::Lockfile
s_lockfile( false /* no IPC server */ );
415 Reference
<XComponentContext
> xComponentContext( bootstrapStandAlone() );
416 out_localContext
= xComponentContext
;
417 if (::dp_misc::office_is_running()) {
418 xComponentContext
.set(
419 connectToOffice( xComponentContext
, verbose
) );
423 if (! s_lockfile
.check( nullptr ))
425 OUString
sMsg(DpResId(RID_STR_CONCURRENTINSTANCE
));
426 OUString
sError(DpResId(RID_STR_UNOPKG_ERROR
));
428 sMsg
+= "\n" + getLockFilePath();
432 //We show a message box or print to the console that there
433 //is another instance already running
435 throw RuntimeException( "Cannot initialize VCL!" );
437 std::unique_ptr
<weld::MessageDialog
> xWarn(Application::CreateMessageDialog(nullptr,
438 VclMessageType::Warning
, VclButtonsType::Ok
,
440 xWarn
->set_title(utl::ConfigManager::getProductName());
446 throw LockFileException(sError
+ sMsg
);
450 return xComponentContext
;
455 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */