Version 7.6.3.2-android, tag libreoffice-7.6.3.2-android
[LibreOffice.git] / desktop / source / pkgchk / unopkg / unopkg_misc.cxx
blob553952917748218d5c87735ddc9c0f9657bdf17a
1 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
2 /*
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>
45 #include <dp_misc.h>
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;
53 namespace unopkg {
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;
77 if (!opt.isEmpty())
79 if (opt.equalsAsciiL(
80 option_info.m_name, option_info.m_name_length ))
82 return &option_info;
86 SAL_WARN( "desktop", opt );
87 return nullptr;
91 bool isOption( OptionInfo const * option_info, sal_uInt32 * pIndex )
93 assert(option_info != nullptr);
94 if (osl_getCommandArgCount() <= *pIndex)
95 return false;
97 OUString arg;
98 osl_getCommandArg( *pIndex, &arg.pData );
99 sal_Int32 len = arg.getLength();
101 if (len < 2 || arg[ 0 ] != '-')
102 return false;
104 if (len == 2 && arg[ 1 ] == option_info->m_short_option)
106 ++(*pIndex);
107 dp_misc::TRACE(__FILE__ ": identified option \'\'"
108 + OUStringChar( option_info->m_short_option ) + "\n");
109 return true;
111 if (arg[ 1 ] == '-' && rtl_ustr_ascii_compare(
112 arg.pData->buffer + 2, option_info->m_name ) == 0)
114 ++(*pIndex);
115 dp_misc::TRACE(__FILE__ ": identified option \'"
116 + OUString::createFromAscii(option_info->m_name) + "\'\n");
117 return true;
119 return false;
123 bool isBootstrapVariable(sal_uInt32 * pIndex)
125 OSL_ASSERT(osl_getCommandArgCount() >= *pIndex);
127 OUString arg;
128 osl_getCommandArg(*pIndex, &arg.pData);
129 if (arg.match("-env:"))
131 ++(*pIndex);
132 return true;
134 return false;
138 bool readArgument(
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: "
148 + *pValue + "\n");
149 ++(*pIndex);
150 return true;
152 --(*pIndex);
154 return false;
158 OUString const & getExecutableDir()
160 static const OUString EXEC =
161 []()
163 OUString path;
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( '/' ) );
168 }();
169 return EXEC;
173 OUString const & getProcessWorkingDir()
175 static const OUString WORKING =
176 []()
178 OUString workingDir;
179 utl::Bootstrap::getProcessWorkingDir(workingDir);
180 return workingDir;
181 }();
182 return WORKING;
186 OUString makeAbsoluteFileUrl(
187 OUString const & sys_path, OUString const & base_url )
189 // system path to file url
190 OUString file_url;
191 oslFileError rc = osl_getFileURLFromSystemPath( sys_path.pData, &file_url.pData );
192 if ( rc != osl_File_E_None) {
193 OUString tempPath;
194 if ( osl_getSystemPathFromFileURL( sys_path.pData, &tempPath.pData) != osl_File_E_None )
196 throw RuntimeException("cannot get file url from system path: " +
197 sys_path );
199 file_url = sys_path;
202 OUString abs;
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;
215 namespace {
218 void printf_space( sal_Int32 space )
220 while (space--)
221 dp_misc::writeConsole(u" ");
225 void printf_line(
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"));
233 void printf_package(
234 Reference<deployment::XPackage> const & xPackage,
235 Reference<XCommandEnvironment> const & xCmdEnv, sal_Int32 level )
237 beans::Optional< OUString > id(
238 level == 0
239 ? beans::Optional< OUString >(
240 true, dp_misc::getIdentifier( xPackage ) )
241 : xPackage->getIdentifier() );
242 if (id.IsPresent)
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 ) );
251 OUString value;
252 if (option.IsPresent) {
253 beans::Ambiguous<sal_Bool> const & reg = option.Value;
254 if (reg.IsAmbiguous)
255 value = "unknown";
256 else
257 value = reg.Value ? std::u16string_view(u"yes") : std::u16string_view(u"no");
259 else
260 value = "n/a";
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())
271 return;
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");
285 } // anon namespace
287 static void printf_unaccepted_licenses(
288 Reference<deployment::XPackage> const & ext)
290 OUString id(
291 dp_misc::getIdentifier(ext) );
292 printf_line( u"Identifier", id, 0 );
293 printf_space(1);
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");
310 else
312 int index = 0;
313 for (auto const& extension : allExtensions)
315 if (vecUnaccepted[index])
316 printf_unaccepted_licenses(extension);
317 else
318 printf_package( extension, xCmdEnv, level );
319 dp_misc::writeConsole(u"\n");
320 ++index;
326 namespace {
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 );
343 return xContext;
347 Reference<XComponentContext> connectToOffice(
348 Reference<XComponentContext> const & xLocalComponentContext,
349 bool verbose )
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" );
357 if (verbose)
359 dp_misc::writeConsole(Concat2View(
360 "Raising process: " + appURL +
361 "\nArguments: --nologo --nodefault " + args[2] +
362 "\n"));
365 ::dp_misc::raiseProcess( appURL, args );
367 if (verbose)
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 ),
374 UNO_QUERY_THROW );
375 if (verbose)
376 dp_misc::writeConsole(u"OK.\n");
378 return xRet;
381 } // anon namespace
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()
388 OUString ret;
389 OUString sBootstrap("${$BRAND_BASE_DIR/" LIBO_ETC_FOLDER "/" SAL_CONFIGFILE("bootstrap") ":UserInstallation}");
390 rtl::Bootstrap::expandMacros(sBootstrap);
391 OUString sAbs;
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))
398 ret = sBootstrap;
402 return ret;
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 ) );
421 else
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();
430 if (bGui)
432 //We show a message box or print to the console that there
433 //is another instance already running
434 if ( ! InitVCL() )
435 throw RuntimeException( "Cannot initialize VCL!" );
437 std::unique_ptr<weld::MessageDialog> xWarn(Application::CreateMessageDialog(nullptr,
438 VclMessageType::Warning, VclButtonsType::Ok,
439 sMsg));
440 xWarn->set_title(utl::ConfigManager::getProductName());
441 xWarn->run();
443 DeInitVCL();
446 throw LockFileException(sError + sMsg);
450 return xComponentContext;
455 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */