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_desktop.hxx" 26 27 #include <cppuhelper/implbase1.hxx> 28 29 #include "comphelper/servicedecl.hxx" 30 #include "cppuhelper/exc_hlp.hxx" 31 #include "rtl/bootstrap.hxx" 32 #include "com/sun/star/deployment/ExtensionManager.hpp" 33 #include "com/sun/star/deployment/XExtensionManager.hpp" 34 #include "com/sun/star/deployment/thePackageManagerFactory.hpp" 35 #include "com/sun/star/deployment/XPackageManager.hpp" 36 #include "com/sun/star/deployment/XPackageManagerFactory.hpp" 37 #include "com/sun/star/deployment/XPackage.hpp" 38 #include "com/sun/star/deployment/InstallException.hpp" 39 #include "com/sun/star/deployment/VersionException.hpp" 40 #include "com/sun/star/deployment/LicenseException.hpp" 41 #include "com/sun/star/lang/XServiceInfo.hpp" 42 #include "com/sun/star/registry/XRegistryKey.hpp" 43 #include "com/sun/star/beans/Optional.hpp" 44 #include "com/sun/star/task/XInteractionApprove.hpp" 45 #include "com/sun/star/beans/Ambiguous.hpp" 46 #include "com/sun/star/uno/XComponentContext.hpp" 47 #include "com/sun/star/io/XInputStream.hpp" 48 #include "com/sun/star/util/XModifyBroadcaster.hpp" 49 #include "comphelper/sequence.hxx" 50 #include "xmlscript/xml_helper.hxx" 51 #include "osl/diagnose.h" 52 #include "dp_interact.h" 53 #include "dp_resource.h" 54 #include "dp_ucb.h" 55 #include "dp_identifier.hxx" 56 #include "dp_descriptioninfoset.hxx" 57 #include "dp_extensionmanager.hxx" 58 #include "dp_commandenvironments.hxx" 59 #include "dp_properties.hxx" 60 #include "boost/bind.hpp" 61 62 #include <list> 63 #include <hash_map> 64 #include <algorithm> 65 66 namespace deploy = com::sun::star::deployment; 67 namespace lang = com::sun::star::lang; 68 namespace registry = com::sun::star::registry; 69 namespace task = com::sun::star::task; 70 namespace ucb = com::sun::star::ucb; 71 namespace uno = com::sun::star::uno; 72 namespace beans = com::sun::star::beans; 73 namespace util = com::sun::star::util; 74 namespace css = com::sun::star; 75 76 77 //#define OUSTR(s) rtl::OUString(RTL_CONSTASCII_USTRINGPARAM(s)) 78 79 using ::com::sun::star::uno::Reference; 80 using ::rtl::OUString; 81 82 namespace { 83 84 struct CompIdentifiers 85 { 86 bool operator() (::std::vector<Reference<deploy::XPackage> > const & a, 87 ::std::vector<Reference<deploy::XPackage> > const & b) 88 { 89 90 if (getName(a).compareTo(getName(b)) < 0) 91 return true; 92 return false; 93 } 94 95 OUString getName(::std::vector<Reference<deploy::XPackage> > const & a); 96 }; 97 98 OUString CompIdentifiers::getName(::std::vector<Reference<deploy::XPackage> > const & a) 99 { 100 OSL_ASSERT(a.size() == 3); 101 //get the first non-null reference 102 Reference<deploy::XPackage> extension; 103 ::std::vector<Reference<deploy::XPackage> >::const_iterator it = a.begin(); 104 for (; it != a.end(); it++) 105 { 106 if (it->is()) 107 { 108 extension = *it; 109 break; 110 } 111 } 112 OSL_ASSERT(extension.is()); 113 return extension->getDisplayName(); 114 } 115 116 void writeLastModified(OUString & url, Reference<ucb::XCommandEnvironment> const & xCmdEnv) 117 { 118 //Write the lastmodified file 119 try { 120 ::rtl::Bootstrap::expandMacros(url); 121 ::ucbhelper::Content ucbStamp(url, xCmdEnv ); 122 dp_misc::erase_path( url, xCmdEnv ); 123 ::rtl::OString stamp("1" ); 124 Reference<css::io::XInputStream> xData( 125 ::xmlscript::createInputStream( 126 ::rtl::ByteSequence( 127 reinterpret_cast<sal_Int8 const *>(stamp.getStr()), 128 stamp.getLength() ) ) ); 129 ucbStamp.writeStream( xData, true /* replace existing */ ); 130 } 131 catch(...) 132 { 133 uno::Any exc(::cppu::getCaughtException()); 134 throw deploy::DeploymentException( 135 OUSTR("Failed to update") + url, 0, exc); 136 } 137 } 138 139 class ExtensionRemoveGuard 140 { 141 css::uno::Reference<css::deployment::XPackage> m_extension; 142 css::uno::Reference<css::deployment::XPackageManager> m_xPackageManager; 143 144 public: 145 ExtensionRemoveGuard(){}; 146 ExtensionRemoveGuard( 147 css::uno::Reference<css::deployment::XPackage> const & extension, 148 css::uno::Reference<css::deployment::XPackageManager> const & xPackageManager): 149 m_extension(extension), m_xPackageManager(xPackageManager) {} 150 ~ExtensionRemoveGuard(); 151 152 void set(css::uno::Reference<css::deployment::XPackage> const & extension, 153 css::uno::Reference<css::deployment::XPackageManager> const & xPackageManager) { 154 m_extension = extension; 155 m_xPackageManager = xPackageManager; 156 } 157 }; 158 159 ExtensionRemoveGuard::~ExtensionRemoveGuard() 160 { 161 try { 162 OSL_ASSERT(!(m_extension.is() && !m_xPackageManager.is())); 163 if (m_xPackageManager.is() && m_extension.is()) 164 m_xPackageManager->removePackage( 165 dp_misc::getIdentifier(m_extension), ::rtl::OUString(), 166 css::uno::Reference<css::task::XAbortChannel>(), 167 css::uno::Reference<css::ucb::XCommandEnvironment>()); 168 } catch (...) { 169 OSL_ASSERT(0); 170 } 171 } 172 173 } //end namespace 174 175 namespace dp_manager { 176 177 178 179 //------------------------------------------------------------------------------ 180 181 //ToDo: bundled extension 182 ExtensionManager::ExtensionManager( Reference< uno::XComponentContext > const& xContext) : 183 ::cppu::WeakComponentImplHelper1< css::deployment::XExtensionManager >(getMutex()), 184 m_xContext( xContext ) 185 { 186 m_xPackageManagerFactory = deploy::thePackageManagerFactory::get(m_xContext); 187 OSL_ASSERT(m_xPackageManagerFactory.is()); 188 189 m_repositoryNames.push_back(OUSTR("user")); 190 m_repositoryNames.push_back(OUSTR("shared")); 191 m_repositoryNames.push_back(OUSTR("bundled")); 192 } 193 194 //------------------------------------------------------------------------------ 195 196 ExtensionManager::~ExtensionManager() 197 { 198 } 199 200 Reference<deploy::XPackageManager> ExtensionManager::getUserRepository() 201 { 202 return m_xPackageManagerFactory->getPackageManager(OUSTR("user")); 203 } 204 Reference<deploy::XPackageManager> ExtensionManager::getSharedRepository() 205 { 206 return m_xPackageManagerFactory->getPackageManager(OUSTR("shared")); 207 } 208 Reference<deploy::XPackageManager> ExtensionManager::getBundledRepository() 209 { 210 return m_xPackageManagerFactory->getPackageManager(OUSTR("bundled")); 211 } 212 Reference<deploy::XPackageManager> ExtensionManager::getTmpRepository() 213 { 214 return m_xPackageManagerFactory->getPackageManager(OUSTR("tmp")); 215 } 216 Reference<deploy::XPackageManager> ExtensionManager::getBakRepository() 217 { 218 return m_xPackageManagerFactory->getPackageManager(OUSTR("bak")); 219 } 220 221 Reference<task::XAbortChannel> ExtensionManager::createAbortChannel() 222 throw (uno::RuntimeException) 223 { 224 return new dp_misc::AbortChannel; 225 } 226 227 css::uno::Reference<css::deployment::XPackageManager> 228 ExtensionManager::getPackageManager(::rtl::OUString const & repository) 229 throw (css::lang::IllegalArgumentException) 230 { 231 Reference<deploy::XPackageManager> xPackageManager; 232 if (repository.equals(OUSTR("user"))) 233 xPackageManager = getUserRepository(); 234 else if (repository.equals(OUSTR("shared"))) 235 xPackageManager = getSharedRepository(); 236 else if (repository.equals(OUSTR("bundled"))) 237 xPackageManager = getBundledRepository(); 238 else if (repository.equals(OUSTR("tmp"))) 239 xPackageManager = getTmpRepository(); 240 else if (repository.equals(OUSTR("bak"))) 241 xPackageManager = getBakRepository(); 242 else 243 throw lang::IllegalArgumentException( 244 OUSTR("No valid repository name provided."), 245 static_cast<cppu::OWeakObject*>(this), 0); 246 return xPackageManager; 247 } 248 249 250 /* 251 Enters the XPackage objects into a map. They must be all from the 252 same repository. The value type of the map is a vector, where each vector 253 represents an extension with a particular identifier. The first member 254 is represents the user extension, the second the shared extension and the 255 third the bundled extension. 256 */ 257 void ExtensionManager::addExtensionsToMap( 258 id2extensions & mapExt, 259 uno::Sequence<Reference<deploy::XPackage> > const & seqExt, 260 OUString const & repository) 261 { 262 //Determine the index in the vector where these extensions are to be 263 //added. 264 ::std::list<OUString>::const_iterator citNames = 265 m_repositoryNames.begin(); 266 int index = 0; 267 for (;citNames != m_repositoryNames.end(); citNames++, index++) 268 { 269 if (citNames->equals(repository)) 270 break; 271 } 272 273 for (int i = 0; i < seqExt.getLength(); i++) 274 { 275 Reference<deploy::XPackage> const & xExtension = seqExt[i]; 276 OUString id = dp_misc::getIdentifier(xExtension); 277 id2extensions::iterator ivec = mapExt.find(id); 278 if (ivec == mapExt.end()) 279 { 280 ::std::vector<Reference<deploy::XPackage> > vec(3); 281 vec[index] = xExtension; 282 mapExt[id] = vec; 283 } 284 else 285 { 286 ivec->second[index] = xExtension; 287 } 288 } 289 } 290 291 /* 292 returns a list containing extensions with the same identifier from 293 all repositories (user, shared, bundled) If one repository does not 294 have this extension, then the list contains an empty Referenc. The list 295 is ordered according to the priority of the repostories: 296 1. user 297 2. shared 298 3. bundled 299 300 The number of elements is always three, unless the number of repository 301 changes. 302 */ 303 ::std::list<Reference<deploy::XPackage> > 304 ExtensionManager::getExtensionsWithSameId( 305 OUString const & identifier, OUString const & fileName, 306 Reference< ucb::XCommandEnvironment> const & /*xCmdEnv*/) 307 308 { 309 ::std::list<Reference<deploy::XPackage> > extensionList; 310 try 311 { //will throw an exception if the extension does not exist 312 extensionList.push_back(getUserRepository()->getDeployedPackage( 313 identifier, fileName, Reference<ucb::XCommandEnvironment>())); 314 } catch(lang::IllegalArgumentException &) 315 { 316 extensionList.push_back(Reference<deploy::XPackage>()); 317 } 318 try 319 { 320 extensionList.push_back(getSharedRepository()->getDeployedPackage( 321 identifier, fileName, Reference<ucb::XCommandEnvironment>())); 322 } catch (lang::IllegalArgumentException &) 323 { 324 extensionList.push_back(Reference<deploy::XPackage>()); 325 } 326 try 327 { 328 extensionList.push_back(getBundledRepository()->getDeployedPackage( 329 identifier, fileName, Reference<ucb::XCommandEnvironment>())); 330 } catch (lang::IllegalArgumentException &) 331 { 332 extensionList.push_back(Reference<deploy::XPackage>()); 333 } 334 OSL_ASSERT(extensionList.size() == 3); 335 return extensionList; 336 } 337 338 uno::Sequence<Reference<deploy::XPackage> > 339 ExtensionManager::getExtensionsWithSameIdentifier( 340 OUString const & identifier, 341 OUString const & fileName, 342 Reference< ucb::XCommandEnvironment> const & xCmdEnv ) 343 throw ( 344 deploy::DeploymentException, 345 ucb::CommandFailedException, 346 lang::IllegalArgumentException, 347 uno::RuntimeException) 348 { 349 try 350 { 351 ::std::list<Reference<deploy::XPackage> > listExtensions = 352 getExtensionsWithSameId( 353 identifier, fileName, xCmdEnv); 354 sal_Bool bHasExtension = false; 355 356 //throw an IllegalArgumentException if there is no extension at all. 357 typedef ::std::list<Reference<deploy::XPackage> >::const_iterator CIT; 358 for (CIT i = listExtensions.begin(); i != listExtensions.end(); i++) 359 bHasExtension |= i->is(); 360 if (!bHasExtension) 361 throw lang::IllegalArgumentException( 362 OUSTR("Could not find extension: ") + identifier + OUSTR(", ") + fileName, 363 static_cast<cppu::OWeakObject*>(this), -1); 364 365 return comphelper::containerToSequence< 366 Reference<deploy::XPackage>, 367 ::std::list<Reference<deploy::XPackage> > 368 > (listExtensions); 369 } 370 catch (deploy::DeploymentException & ) 371 { 372 throw; 373 } 374 catch ( ucb::CommandFailedException & ) 375 { 376 throw; 377 } 378 catch (lang::IllegalArgumentException &) 379 { 380 throw; 381 } 382 catch (...) 383 { 384 uno::Any exc = ::cppu::getCaughtException(); 385 throw deploy::DeploymentException( 386 OUSTR("Extension Manager: exception during getExtensionsWithSameIdentifier"), 387 static_cast<OWeakObject*>(this), exc); 388 } 389 } 390 391 392 393 bool ExtensionManager::isUserDisabled( 394 OUString const & identifier, OUString const & fileName) 395 { 396 ::std::list<Reference<deploy::XPackage> > listExtensions; 397 398 try { 399 listExtensions = getExtensionsWithSameId(identifier, fileName); 400 } catch (lang::IllegalArgumentException & ) { 401 } 402 OSL_ASSERT(listExtensions.size() == 3); 403 404 return isUserDisabled( ::comphelper::containerToSequence< 405 Reference<deploy::XPackage>, 406 ::std::list<Reference<deploy::XPackage> > 407 > (listExtensions)); 408 } 409 410 bool ExtensionManager::isUserDisabled( 411 uno::Sequence<Reference<deploy::XPackage> > const & seqExtSameId) 412 { 413 OSL_ASSERT(seqExtSameId.getLength() == 3); 414 Reference<deploy::XPackage> const & userExtension = seqExtSameId[0]; 415 if (userExtension.is()) 416 { 417 beans::Optional<beans::Ambiguous<sal_Bool> > reg = 418 userExtension->isRegistered(Reference<task::XAbortChannel>(), 419 Reference<ucb::XCommandEnvironment>()); 420 //If the value is ambiguous is than we assume that the extension 421 //is enabled, but something went wrong during enabling. We do not 422 //automatically disable user extensions. 423 if (reg.IsPresent && 424 ! reg.Value.IsAmbiguous && ! reg.Value.Value) 425 return true; 426 } 427 return false; 428 } 429 430 /* 431 This method determines the active extension (XPackage.registerPackage) with a 432 particular identifier. 433 434 The parameter bUserDisabled determines if the user extension is disabled. 435 436 When the user repository contains an extension with the given identifier and 437 it is not disabled by the user, then it is always registered. Otherwise an 438 extension is only registered when there is no registered extension in one of 439 the repositories with a higher priority. That is, if the extension is from 440 the shared repository and an active extension with the same identifer is in 441 the user repository, then the extension is not registered. Similarly a 442 bundled extension is not registered if there is an active extension with the 443 same identifier in the shared or user repository. 444 */ 445 void ExtensionManager::activateExtension( 446 OUString const & identifier, OUString const & fileName, 447 bool bUserDisabled, 448 bool bStartup, 449 Reference<task::XAbortChannel> const & xAbortChannel, 450 Reference<ucb::XCommandEnvironment> const & xCmdEnv ) 451 { 452 ::std::list<Reference<deploy::XPackage> > listExtensions; 453 try { 454 listExtensions = getExtensionsWithSameId(identifier, fileName); 455 } catch (lang::IllegalArgumentException &) { 456 } 457 OSL_ASSERT(listExtensions.size() == 3); 458 459 activateExtension( 460 ::comphelper::containerToSequence< 461 Reference<deploy::XPackage>, 462 ::std::list<Reference<deploy::XPackage> > 463 > (listExtensions), 464 bUserDisabled, bStartup, xAbortChannel, xCmdEnv); 465 466 fireModified(); 467 } 468 469 void ExtensionManager::activateExtension( 470 uno::Sequence<Reference<deploy::XPackage> > const & seqExt, 471 bool bUserDisabled, 472 bool bStartup, 473 Reference<task::XAbortChannel> const & xAbortChannel, 474 Reference<ucb::XCommandEnvironment> const & xCmdEnv ) 475 { 476 bool bActive = false; 477 sal_Int32 len = seqExt.getLength(); 478 for (sal_Int32 i = 0; i < len; i++) 479 { 480 Reference<deploy::XPackage> const & aExt = seqExt[i]; 481 if (aExt.is()) 482 { 483 //get the registration value of the current iteration 484 beans::Optional<beans::Ambiguous<sal_Bool> > optReg = 485 aExt->isRegistered(xAbortChannel, xCmdEnv); 486 //If nothing can be registered then break 487 if (!optReg.IsPresent) 488 break; 489 490 //Check if this is a disabled user extension, 491 if (i == 0 && bUserDisabled) 492 { 493 aExt->revokePackage(xAbortChannel, xCmdEnv); 494 continue; 495 } 496 497 //If we have already determined an active extension then we must 498 //make sure to unregister all extensions with the same id in 499 //repositories with a lower priority 500 if (bActive) 501 { 502 aExt->revokePackage(xAbortChannel, xCmdEnv); 503 } 504 else 505 { 506 //This is the first extension in the ordered list, which becomes 507 //the active extension 508 bActive = true; 509 //Register if not already done. 510 //reregister if the value is ambiguous, which indicates that 511 //something went wrong during last registration. 512 aExt->registerPackage(bStartup, xAbortChannel, xCmdEnv); 513 } 514 } 515 } 516 } 517 518 Reference<deploy::XPackage> ExtensionManager::backupExtension( 519 OUString const & identifier, OUString const & fileName, 520 Reference<deploy::XPackageManager> const & xPackageManager, 521 Reference<ucb::XCommandEnvironment> const & xCmdEnv ) 522 { 523 Reference<deploy::XPackage> xBackup; 524 Reference<ucb::XCommandEnvironment> tmpCmdEnv( 525 new TmpRepositoryCommandEnv(xCmdEnv->getInteractionHandler())); 526 Reference<deploy::XPackage> xOldExtension; 527 xOldExtension = xPackageManager->getDeployedPackage( 528 identifier, fileName, tmpCmdEnv); 529 530 if (xOldExtension.is()) 531 { 532 xBackup = getTmpRepository()->addPackage( 533 xOldExtension->getURL(), uno::Sequence<beans::NamedValue>(), 534 OUString(), Reference<task::XAbortChannel>(), tmpCmdEnv); 535 536 OSL_ENSURE(xBackup.is(), "Failed to backup extension"); 537 } 538 return xBackup; 539 } 540 541 //The supported package types are actually determined by the registry. However 542 //creating a registry 543 //(desktop/source/deployment/registry/dp_registry.cxx:PackageRegistryImpl) will 544 //create all the backends, so that the registry can obtain from them the package 545 //types. Creating the registry will also set up the registry folder containing 546 //all the subfolders for the respective backends. 547 //Because all repositories support the same backends, we can just delegate this 548 //call to one of the repositories. 549 uno::Sequence< Reference<deploy::XPackageTypeInfo> > 550 ExtensionManager::getSupportedPackageTypes() 551 throw (uno::RuntimeException) 552 { 553 return getUserRepository()->getSupportedPackageTypes(); 554 } 555 //Do some necessary checks and user interaction. This function does not 556 //aquire the extension manager mutex and that mutex must not be aquired 557 //when this function is called. doChecksForAddExtension does synchronous 558 //user interactions which may require aquiring the solar mutex. 559 //Returns true if the extension can be installed. 560 bool ExtensionManager::doChecksForAddExtension( 561 Reference<deploy::XPackageManager> const & xPackageMgr, 562 uno::Sequence<beans::NamedValue> const & properties, 563 css::uno::Reference<css::deployment::XPackage> const & xTmpExtension, 564 Reference<task::XAbortChannel> const & xAbortChannel, 565 Reference<ucb::XCommandEnvironment> const & xCmdEnv, 566 Reference<deploy::XPackage> & out_existingExtension ) 567 throw (deploy::DeploymentException, 568 ucb::CommandFailedException, 569 ucb::CommandAbortedException, 570 lang::IllegalArgumentException, 571 uno::RuntimeException) 572 { 573 try 574 { 575 Reference<deploy::XPackage> xOldExtension; 576 const OUString sIdentifier = dp_misc::getIdentifier(xTmpExtension); 577 const OUString sFileName = xTmpExtension->getName(); 578 const OUString sDisplayName = xTmpExtension->getDisplayName(); 579 const OUString sVersion = xTmpExtension->getVersion(); 580 581 try 582 { 583 xOldExtension = xPackageMgr->getDeployedPackage( 584 sIdentifier, sFileName, xCmdEnv); 585 out_existingExtension = xOldExtension; 586 } 587 catch (lang::IllegalArgumentException &) 588 { 589 } 590 bool bCanInstall = false; 591 592 //This part is not guarded against other threads removing, adding, disabling ... 593 //etc. the same extension. 594 //checkInstall is safe because it notifies the user if the extension is not yet 595 //installed in the same repository. Because addExtension has its own guard 596 //(m_addMutex), another thread cannot add the extension in the meantime. 597 //checkUpdate is called if the same extension exists in the same 598 //repository. The user is asked if they want to replace it. Another 599 //thread 600 //could already remove the extension. So asking the user was not 601 //necessary. No harm is done. The other thread may also ask the user 602 //if he wants to remove the extension. This depends on the 603 //XCommandEnvironment which it passes to removeExtension. 604 if (xOldExtension.is()) 605 { 606 //throws a CommandFailedException if the user cancels 607 //the action. 608 checkUpdate(sVersion, sDisplayName,xOldExtension, xCmdEnv); 609 } 610 else 611 { 612 //throws a CommandFailedException if the user cancels 613 //the action. 614 checkInstall(sDisplayName, xCmdEnv); 615 } 616 //Prevent showing the license if requested. 617 Reference<ucb::XCommandEnvironment> _xCmdEnv(xCmdEnv); 618 ExtensionProperties props(OUString(), properties, Reference<ucb::XCommandEnvironment>()); 619 620 dp_misc::DescriptionInfoset info(dp_misc::getDescriptionInfoset(xTmpExtension->getURL())); 621 const ::boost::optional<dp_misc::SimpleLicenseAttributes> licenseAttributes = 622 info.getSimpleLicenseAttributes(); 623 624 if (licenseAttributes && licenseAttributes->suppressIfRequired 625 && props.isSuppressedLicense()) 626 _xCmdEnv = Reference<ucb::XCommandEnvironment>( 627 new NoLicenseCommandEnv(xCmdEnv->getInteractionHandler())); 628 629 bCanInstall = xTmpExtension->checkPrerequisites( 630 xAbortChannel, _xCmdEnv, xOldExtension.is() || props.isExtensionUpdate()) == 0 ? true : false; 631 632 return bCanInstall; 633 } 634 catch (deploy::DeploymentException& ) { 635 throw; 636 } catch (ucb::CommandFailedException & ) { 637 throw; 638 } catch (ucb::CommandAbortedException & ) { 639 throw; 640 } catch (lang::IllegalArgumentException &) { 641 throw; 642 } catch (uno::RuntimeException &) { 643 throw; 644 } catch (uno::Exception &) { 645 uno::Any excOccurred = ::cppu::getCaughtException(); 646 deploy::DeploymentException exc( 647 OUSTR("Extension Manager: exception in doChecksForAddExtension"), 648 static_cast<OWeakObject*>(this), excOccurred); 649 throw exc; 650 } catch (...) { 651 throw uno::RuntimeException( 652 OUSTR("Extension Manager: unexpected exception in doChecksForAddExtension"), 653 static_cast<OWeakObject*>(this)); 654 } 655 } 656 657 // Only add to shared and user repository 658 Reference<deploy::XPackage> ExtensionManager::addExtension( 659 OUString const & url, uno::Sequence<beans::NamedValue> const & properties, 660 OUString const & repository, 661 Reference<task::XAbortChannel> const & xAbortChannel, 662 Reference<ucb::XCommandEnvironment> const & xCmdEnv ) 663 throw (deploy::DeploymentException, 664 ucb::CommandFailedException, 665 ucb::CommandAbortedException, 666 lang::IllegalArgumentException, 667 uno::RuntimeException) 668 { 669 Reference<deploy::XPackage> xNewExtension; 670 //Determine the repository to use 671 Reference<deploy::XPackageManager> xPackageManager; 672 if (repository.equals(OUSTR("user"))) 673 xPackageManager = getUserRepository(); 674 else if (repository.equals(OUSTR("shared"))) 675 xPackageManager = getSharedRepository(); 676 else 677 throw lang::IllegalArgumentException( 678 OUSTR("No valid repository name provided."), 679 static_cast<cppu::OWeakObject*>(this), 0); 680 //We must make sure that the xTmpExtension is not create twice, because this 681 //would remove the first one. 682 ::osl::MutexGuard addGuard(m_addMutex); 683 684 Reference<deploy::XPackage> xTmpExtension = 685 getTempExtension(url, xAbortChannel, xCmdEnv); 686 //Make sure the extension is removed from the tmp repository in case 687 //of an exception 688 ExtensionRemoveGuard tmpExtensionRemoveGuard(xTmpExtension, getTmpRepository()); 689 ExtensionRemoveGuard bakExtensionRemoveGuard; 690 const OUString sIdentifier = dp_misc::getIdentifier(xTmpExtension); 691 const OUString sFileName = xTmpExtension->getName(); 692 Reference<deploy::XPackage> xOldExtension; 693 Reference<deploy::XPackage> xExtensionBackup; 694 695 uno::Any excOccurred2; 696 bool bUserDisabled = false; 697 bool bCanInstall = doChecksForAddExtension( 698 xPackageManager, 699 properties, 700 xTmpExtension, 701 xAbortChannel, 702 xCmdEnv, 703 xOldExtension ); 704 705 { 706 // In this garded section (getMutex) we must not use the argument xCmdEnv 707 // because it may bring up dialogs (XInteractionHandler::handle) this 708 //may potententially deadlock. See issue 709 //http://qa.openoffice.org/issues/show_bug.cgi?id=114933 710 //By not providing xCmdEnv the underlying APIs will throw an exception if 711 //the XInteractionRequest cannot be handled 712 ::osl::MutexGuard guard(getMutex()); 713 714 if (bCanInstall) 715 { 716 try 717 { 718 bUserDisabled = isUserDisabled(sIdentifier, sFileName); 719 if (xOldExtension.is()) 720 { 721 try 722 { 723 xOldExtension->revokePackage( 724 xAbortChannel, Reference<ucb::XCommandEnvironment>()); 725 //save the old user extension in case the user aborts 726 xExtensionBackup = getBakRepository()->importExtension( 727 xOldExtension, Reference<task::XAbortChannel>(), 728 Reference<ucb::XCommandEnvironment>()); 729 bakExtensionRemoveGuard.set(xExtensionBackup, getBakRepository()); 730 } 731 catch (lang::DisposedException &) 732 { 733 //Another thread might have removed the extension meanwhile 734 } 735 } 736 //check again dependencies but prevent user interaction, 737 //We can disregard the license, because the user must have already 738 //accepted it, whe we called checkPrerequisites the first time 739 SilentCheckPrerequisitesCommandEnv * pSilentCommandEnv = 740 new SilentCheckPrerequisitesCommandEnv(); 741 Reference<ucb::XCommandEnvironment> silentCommandEnv(pSilentCommandEnv); 742 743 sal_Int32 failedPrereq = xTmpExtension->checkPrerequisites( 744 xAbortChannel, silentCommandEnv, true); 745 if (failedPrereq == 0) 746 { 747 xNewExtension = xPackageManager->addPackage( 748 url, properties, OUString(), xAbortChannel, 749 Reference<ucb::XCommandEnvironment>()); 750 //If we add a user extension and there is already one which was 751 //disabled by a user, then the newly installed one is enabled. If we 752 //add to another repository then the user extension remains 753 //disabled. 754 bool bUserDisabled2 = bUserDisabled; 755 if (repository.equals(OUSTR("user"))) 756 bUserDisabled2 = false; 757 758 ::rtl::OUString const name(xNewExtension->getName()); 759 activateExtension( 760 dp_misc::getIdentifier(xNewExtension), 761 name, bUserDisabled2, false, xAbortChannel, 762 Reference<ucb::XCommandEnvironment>()); 763 } 764 else 765 { 766 if (pSilentCommandEnv->m_Exception.hasValue()) 767 ::cppu::throwException(pSilentCommandEnv->m_Exception); 768 else if ( pSilentCommandEnv->m_UnknownException.hasValue()) 769 ::cppu::throwException(pSilentCommandEnv->m_UnknownException); 770 else 771 throw deploy::DeploymentException ( 772 OUSTR("Extension Manager: exception during addExtension, ckeckPrerequisites failed"), 773 static_cast<OWeakObject*>(this), uno::Any()); 774 } 775 } 776 catch (deploy::DeploymentException& ) { 777 excOccurred2 = ::cppu::getCaughtException(); 778 } catch (ucb::CommandFailedException & ) { 779 excOccurred2 = ::cppu::getCaughtException(); 780 } catch (ucb::CommandAbortedException & ) { 781 excOccurred2 = ::cppu::getCaughtException(); 782 } catch (lang::IllegalArgumentException &) { 783 excOccurred2 = ::cppu::getCaughtException(); 784 } catch (uno::RuntimeException &) { 785 excOccurred2 = ::cppu::getCaughtException(); 786 } catch (...) { 787 excOccurred2 = ::cppu::getCaughtException(); 788 deploy::DeploymentException exc( 789 OUSTR("Extension Manager: exception during addExtension, url: ") 790 + url, static_cast<OWeakObject*>(this), excOccurred2); 791 excOccurred2 <<= exc; 792 } 793 } 794 795 if (excOccurred2.hasValue()) 796 { 797 //It does not matter what exception is thrown. We try to 798 //recover the original status. 799 //If the user aborted installation then a ucb::CommandAbortedException 800 //is thrown. 801 //Use a private AbortChannel so the user cannot interrupt. 802 try 803 { 804 if (xExtensionBackup.is()) 805 { 806 Reference<deploy::XPackage> xRestored = 807 xPackageManager->importExtension( 808 xExtensionBackup, Reference<task::XAbortChannel>(), 809 Reference<ucb::XCommandEnvironment>()); 810 } 811 activateExtension( 812 sIdentifier, sFileName, bUserDisabled, false, 813 Reference<task::XAbortChannel>(), Reference<ucb::XCommandEnvironment>()); 814 } 815 catch (...) 816 { 817 } 818 ::cppu::throwException(excOccurred2); 819 } 820 } // leaving the garded section (getMutex()) 821 822 try 823 { 824 fireModified(); 825 826 }catch (deploy::DeploymentException& ) { 827 throw; 828 } catch (ucb::CommandFailedException & ) { 829 throw; 830 } catch (ucb::CommandAbortedException & ) { 831 throw; 832 } catch (lang::IllegalArgumentException &) { 833 throw; 834 } catch (uno::RuntimeException &) { 835 throw; 836 } catch (uno::Exception &) { 837 uno::Any excOccurred = ::cppu::getCaughtException(); 838 deploy::DeploymentException exc( 839 OUSTR("Extension Manager: exception in doChecksForAddExtension"), 840 static_cast<OWeakObject*>(this), excOccurred); 841 throw exc; 842 } catch (...) { 843 throw uno::RuntimeException( 844 OUSTR("Extension Manager: unexpected exception in doChecksForAddExtension"), 845 static_cast<OWeakObject*>(this)); 846 } 847 848 return xNewExtension; 849 } 850 851 void ExtensionManager::removeExtension( 852 OUString const & identifier, OUString const & fileName, 853 OUString const & repository, 854 Reference<task::XAbortChannel> const & xAbortChannel, 855 Reference<ucb::XCommandEnvironment> const & xCmdEnv ) 856 throw (deploy::DeploymentException, 857 ucb::CommandFailedException, 858 ucb::CommandAbortedException, 859 lang::IllegalArgumentException, 860 uno::RuntimeException) 861 { 862 uno::Any excOccurred1; 863 Reference<deploy::XPackage> xExtensionBackup; 864 Reference<deploy::XPackageManager> xPackageManager; 865 bool bUserDisabled = false; 866 ::osl::MutexGuard guard(getMutex()); 867 try 868 { 869 //Determine the repository to use 870 if (repository.equals(OUSTR("user"))) 871 xPackageManager = getUserRepository(); 872 else if (repository.equals(OUSTR("shared"))) 873 xPackageManager = getSharedRepository(); 874 else 875 throw lang::IllegalArgumentException( 876 OUSTR("No valid repository name provided."), 877 static_cast<cppu::OWeakObject*>(this), 0); 878 879 bUserDisabled = isUserDisabled(identifier, fileName); 880 //Backup the extension, in case the user cancels the action 881 xExtensionBackup = backupExtension( 882 identifier, fileName, xPackageManager, xCmdEnv); 883 884 //revoke the extension if it is active 885 Reference<deploy::XPackage> xOldExtension = 886 xPackageManager->getDeployedPackage( 887 identifier, fileName, xCmdEnv); 888 xOldExtension->revokePackage(xAbortChannel, xCmdEnv); 889 890 xPackageManager->removePackage( 891 identifier, fileName, xAbortChannel, xCmdEnv); 892 activateExtension(identifier, fileName, bUserDisabled, false, 893 xAbortChannel, xCmdEnv); 894 fireModified(); 895 } 896 catch (deploy::DeploymentException& ) { 897 excOccurred1 = ::cppu::getCaughtException(); 898 } catch (ucb::CommandFailedException & ) { 899 excOccurred1 = ::cppu::getCaughtException(); 900 } catch (ucb::CommandAbortedException & ) { 901 excOccurred1 = ::cppu::getCaughtException(); 902 } catch (lang::IllegalArgumentException &) { 903 excOccurred1 = ::cppu::getCaughtException(); 904 } catch (uno::RuntimeException &) { 905 excOccurred1 = ::cppu::getCaughtException(); 906 } catch (...) { 907 excOccurred1 = ::cppu::getCaughtException(); 908 deploy::DeploymentException exc( 909 OUSTR("Extension Manager: exception during removeEtension"), 910 static_cast<OWeakObject*>(this), excOccurred1); 911 excOccurred1 <<= exc; 912 } 913 914 if (excOccurred1.hasValue()) 915 { 916 //User aborted installation, restore the previous situation. 917 //Use a private AbortChannel so the user cannot interrupt. 918 try 919 { 920 Reference<ucb::XCommandEnvironment> tmpCmdEnv( 921 new TmpRepositoryCommandEnv(xCmdEnv->getInteractionHandler())); 922 if (xExtensionBackup.is()) 923 { 924 Reference<deploy::XPackage> xRestored = 925 xPackageManager->importExtension( 926 xExtensionBackup, Reference<task::XAbortChannel>(), 927 tmpCmdEnv); 928 activateExtension( 929 identifier, fileName, bUserDisabled, false, 930 Reference<task::XAbortChannel>(), 931 tmpCmdEnv); 932 933 getTmpRepository()->removePackage( 934 dp_misc::getIdentifier(xExtensionBackup), 935 xExtensionBackup->getName(), xAbortChannel, xCmdEnv); 936 fireModified(); 937 } 938 } 939 catch (...) 940 { 941 } 942 ::cppu::throwException(excOccurred1); 943 } 944 945 if (xExtensionBackup.is()) 946 getTmpRepository()->removePackage( 947 dp_misc::getIdentifier(xExtensionBackup), 948 xExtensionBackup->getName(), xAbortChannel, xCmdEnv); 949 } 950 951 // Only enable extensions from shared and user repository 952 void ExtensionManager::enableExtension( 953 Reference<deploy::XPackage> const & extension, 954 Reference<task::XAbortChannel> const & xAbortChannel, 955 Reference<ucb::XCommandEnvironment> const & xCmdEnv) 956 throw (deploy::DeploymentException, 957 ucb::CommandFailedException, 958 ucb::CommandAbortedException, 959 lang::IllegalArgumentException, 960 uno::RuntimeException) 961 { 962 ::osl::MutexGuard guard(getMutex()); 963 bool bUserDisabled = false; 964 uno::Any excOccurred; 965 try 966 { 967 if (!extension.is()) 968 return; 969 OUString repository = extension->getRepositoryName(); 970 if (!repository.equals(OUSTR("user"))) 971 throw lang::IllegalArgumentException( 972 OUSTR("No valid repository name provided."), 973 static_cast<cppu::OWeakObject*>(this), 0); 974 975 bUserDisabled = isUserDisabled(dp_misc::getIdentifier(extension), 976 extension->getName()); 977 978 activateExtension(dp_misc::getIdentifier(extension), 979 extension->getName(), false, false, 980 xAbortChannel, xCmdEnv); 981 } 982 catch (deploy::DeploymentException& ) { 983 excOccurred = ::cppu::getCaughtException(); 984 } catch (ucb::CommandFailedException & ) { 985 excOccurred = ::cppu::getCaughtException(); 986 } catch (ucb::CommandAbortedException & ) { 987 excOccurred = ::cppu::getCaughtException(); 988 } catch (lang::IllegalArgumentException &) { 989 excOccurred = ::cppu::getCaughtException(); 990 } catch (uno::RuntimeException &) { 991 excOccurred = ::cppu::getCaughtException(); 992 } catch (...) { 993 excOccurred = ::cppu::getCaughtException(); 994 deploy::DeploymentException exc( 995 OUSTR("Extension Manager: exception during enableExtension"), 996 static_cast<OWeakObject*>(this), excOccurred); 997 excOccurred <<= exc; 998 } 999 1000 if (excOccurred.hasValue()) 1001 { 1002 try 1003 { 1004 activateExtension(dp_misc::getIdentifier(extension), 1005 extension->getName(), bUserDisabled, false, 1006 xAbortChannel, xCmdEnv); 1007 } 1008 catch (...) 1009 { 1010 } 1011 ::cppu::throwException(excOccurred); 1012 } 1013 } 1014 1015 /** 1016 */ 1017 sal_Int32 ExtensionManager::checkPrerequisitesAndEnable( 1018 Reference<deploy::XPackage> const & extension, 1019 Reference<task::XAbortChannel> const & xAbortChannel, 1020 Reference<ucb::XCommandEnvironment> const & xCmdEnv) 1021 throw (deploy::DeploymentException, 1022 ucb::CommandFailedException, 1023 ucb::CommandAbortedException, 1024 lang::IllegalArgumentException, 1025 uno::RuntimeException) 1026 { 1027 try 1028 { 1029 if (!extension.is()) 1030 return 0; 1031 ::osl::MutexGuard guard(getMutex()); 1032 sal_Int32 ret = 0; 1033 Reference<deploy::XPackageManager> mgr = 1034 getPackageManager(extension->getRepositoryName()); 1035 ret = mgr->checkPrerequisites(extension, xAbortChannel, xCmdEnv); 1036 if (ret) 1037 { 1038 //There are some unfulfilled prerequisites, try to revoke 1039 extension->revokePackage(xAbortChannel, xCmdEnv); 1040 } 1041 const OUString id(dp_misc::getIdentifier(extension)); 1042 activateExtension(id, extension->getName(), 1043 isUserDisabled(id, extension->getName()), false, 1044 xAbortChannel, xCmdEnv); 1045 return ret; 1046 } 1047 catch (deploy::DeploymentException& ) { 1048 throw; 1049 } catch (ucb::CommandFailedException & ) { 1050 throw; 1051 } catch (ucb::CommandAbortedException & ) { 1052 throw; 1053 } catch (lang::IllegalArgumentException &) { 1054 throw; 1055 } catch (uno::RuntimeException &) { 1056 throw; 1057 } catch (...) { 1058 uno::Any excOccurred = ::cppu::getCaughtException(); 1059 deploy::DeploymentException exc( 1060 OUSTR("Extension Manager: exception during disableExtension"), 1061 static_cast<OWeakObject*>(this), excOccurred); 1062 throw exc; 1063 } 1064 } 1065 1066 1067 void ExtensionManager::disableExtension( 1068 Reference<deploy::XPackage> const & extension, 1069 Reference<task::XAbortChannel> const & xAbortChannel, 1070 Reference<ucb::XCommandEnvironment> const & xCmdEnv ) 1071 throw (deploy::DeploymentException, 1072 ucb::CommandFailedException, 1073 ucb::CommandAbortedException, 1074 lang::IllegalArgumentException, 1075 uno::RuntimeException) 1076 { 1077 ::osl::MutexGuard guard(getMutex()); 1078 uno::Any excOccurred; 1079 bool bUserDisabled = false; 1080 try 1081 { 1082 if (!extension.is()) 1083 return; 1084 const OUString repository( extension->getRepositoryName()); 1085 if (!repository.equals(OUSTR("user"))) 1086 throw lang::IllegalArgumentException( 1087 OUSTR("No valid repository name provided."), 1088 static_cast<cppu::OWeakObject*>(this), 0); 1089 1090 const OUString id(dp_misc::getIdentifier(extension)); 1091 bUserDisabled = isUserDisabled(id, extension->getName()); 1092 1093 activateExtension(id, extension->getName(), true, false, 1094 xAbortChannel, xCmdEnv); 1095 } 1096 catch (deploy::DeploymentException& ) { 1097 excOccurred = ::cppu::getCaughtException(); 1098 } catch (ucb::CommandFailedException & ) { 1099 excOccurred = ::cppu::getCaughtException(); 1100 } catch (ucb::CommandAbortedException & ) { 1101 excOccurred = ::cppu::getCaughtException(); 1102 } catch (lang::IllegalArgumentException &) { 1103 excOccurred = ::cppu::getCaughtException(); 1104 } catch (uno::RuntimeException &) { 1105 excOccurred = ::cppu::getCaughtException(); 1106 } catch (...) { 1107 excOccurred = ::cppu::getCaughtException(); 1108 deploy::DeploymentException exc( 1109 OUSTR("Extension Manager: exception during disableExtension"), 1110 static_cast<OWeakObject*>(this), excOccurred); 1111 excOccurred <<= exc; 1112 } 1113 1114 if (excOccurred.hasValue()) 1115 { 1116 try 1117 { 1118 activateExtension(dp_misc::getIdentifier(extension), 1119 extension->getName(), bUserDisabled, false, 1120 xAbortChannel, xCmdEnv); 1121 } 1122 catch (...) 1123 { 1124 } 1125 ::cppu::throwException(excOccurred); 1126 } 1127 } 1128 1129 uno::Sequence< Reference<deploy::XPackage> > 1130 ExtensionManager::getDeployedExtensions( 1131 OUString const & repository, 1132 Reference<task::XAbortChannel> const &xAbort, 1133 Reference<ucb::XCommandEnvironment> const & xCmdEnv ) 1134 throw (deploy::DeploymentException, 1135 ucb::CommandFailedException, 1136 ucb::CommandAbortedException, 1137 lang::IllegalArgumentException, 1138 uno::RuntimeException) 1139 { 1140 return getPackageManager(repository)->getDeployedPackages( 1141 xAbort, xCmdEnv); 1142 } 1143 1144 Reference<deploy::XPackage> 1145 ExtensionManager::getDeployedExtension( 1146 OUString const & repository, 1147 OUString const & identifier, 1148 OUString const & filename, 1149 Reference<ucb::XCommandEnvironment> const & xCmdEnv ) 1150 throw (deploy::DeploymentException, 1151 ucb::CommandFailedException, 1152 lang::IllegalArgumentException, 1153 uno::RuntimeException) 1154 { 1155 return getPackageManager(repository)->getDeployedPackage( 1156 identifier, filename, xCmdEnv); 1157 } 1158 1159 uno::Sequence< uno::Sequence<Reference<deploy::XPackage> > > 1160 ExtensionManager::getAllExtensions( 1161 Reference<task::XAbortChannel> const & xAbort, 1162 Reference<ucb::XCommandEnvironment> const & xCmdEnv ) 1163 throw (deploy::DeploymentException, 1164 ucb::CommandFailedException, 1165 ucb::CommandAbortedException, 1166 lang::IllegalArgumentException, 1167 uno::RuntimeException) 1168 { 1169 try 1170 { 1171 id2extensions mapExt; 1172 1173 uno::Sequence<Reference<deploy::XPackage> > userExt = 1174 getUserRepository()->getDeployedPackages(xAbort, xCmdEnv); 1175 addExtensionsToMap(mapExt, userExt, OUSTR("user")); 1176 uno::Sequence<Reference<deploy::XPackage> > sharedExt = 1177 getSharedRepository()->getDeployedPackages(xAbort, xCmdEnv); 1178 addExtensionsToMap(mapExt, sharedExt, OUSTR("shared")); 1179 uno::Sequence<Reference<deploy::XPackage> > bundledExt = 1180 getBundledRepository()->getDeployedPackages(xAbort, xCmdEnv); 1181 addExtensionsToMap(mapExt, bundledExt, OUSTR("bundled")); 1182 1183 //copy the values of the map to a vector for sorting 1184 ::std::vector< ::std::vector<Reference<deploy::XPackage> > > 1185 vecExtensions; 1186 id2extensions::const_iterator mapIt = mapExt.begin(); 1187 for (;mapIt != mapExt.end(); mapIt++) 1188 vecExtensions.push_back(mapIt->second); 1189 1190 //sort the element according to the identifier 1191 ::std::sort(vecExtensions.begin(), vecExtensions.end(), CompIdentifiers()); 1192 1193 ::std::vector< ::std::vector<Reference<deploy::XPackage> > >::const_iterator 1194 citVecVec = vecExtensions.begin(); 1195 sal_Int32 j = 0; 1196 uno::Sequence< uno::Sequence<Reference<deploy::XPackage> > > seqSeq(vecExtensions.size()); 1197 for (;citVecVec != vecExtensions.end(); citVecVec++, j++) 1198 { 1199 seqSeq[j] = comphelper::containerToSequence(*citVecVec); 1200 } 1201 return seqSeq; 1202 1203 } catch (deploy::DeploymentException& ) { 1204 throw; 1205 } catch (ucb::CommandFailedException & ) { 1206 throw; 1207 } catch (ucb::CommandAbortedException & ) { 1208 throw; 1209 } catch (lang::IllegalArgumentException &) { 1210 throw; 1211 } catch (uno::RuntimeException &) { 1212 throw; 1213 } catch (...) { 1214 uno::Any exc = ::cppu::getCaughtException(); 1215 throw deploy::DeploymentException( 1216 OUSTR("Extension Manager: exception during enableExtension"), 1217 static_cast<OWeakObject*>(this), exc); 1218 } 1219 } 1220 1221 //only to be called from unopkg!!! 1222 void ExtensionManager::reinstallDeployedExtensions( 1223 OUString const & repository, 1224 Reference<task::XAbortChannel> const & xAbortChannel, 1225 Reference<ucb::XCommandEnvironment> const & xCmdEnv ) 1226 throw (deploy::DeploymentException, 1227 ucb::CommandFailedException, ucb::CommandAbortedException, 1228 lang::IllegalArgumentException, uno::RuntimeException) 1229 { 1230 try 1231 { 1232 Reference<deploy::XPackageManager> 1233 xPackageManager = getPackageManager(repository); 1234 1235 ::osl::MutexGuard guard(getMutex()); 1236 xPackageManager->reinstallDeployedPackages(xAbortChannel, xCmdEnv); 1237 //We must sync here, otherwise we will get exceptions when extensions 1238 //are removed. 1239 dp_misc::syncRepositories(xCmdEnv); 1240 const uno::Sequence< Reference<deploy::XPackage> > extensions( 1241 xPackageManager->getDeployedPackages(xAbortChannel, xCmdEnv)); 1242 1243 for ( sal_Int32 pos = 0; pos < extensions.getLength(); ++pos ) 1244 { 1245 try 1246 { 1247 const OUString id = dp_misc::getIdentifier(extensions[ pos ]); 1248 const OUString fileName = extensions[ pos ]->getName(); 1249 OSL_ASSERT(id.getLength()); 1250 activateExtension(id, fileName, false, true, xAbortChannel, xCmdEnv ); 1251 } 1252 catch (lang::DisposedException &) 1253 { 1254 } 1255 } 1256 } catch (deploy::DeploymentException& ) { 1257 throw; 1258 } catch (ucb::CommandFailedException & ) { 1259 throw; 1260 } catch (ucb::CommandAbortedException & ) { 1261 throw; 1262 } catch (lang::IllegalArgumentException &) { 1263 throw; 1264 } catch (uno::RuntimeException &) { 1265 throw; 1266 } catch (...) { 1267 uno::Any exc = ::cppu::getCaughtException(); 1268 throw deploy::DeploymentException( 1269 OUSTR("Extension Manager: exception during enableExtension"), 1270 static_cast<OWeakObject*>(this), exc); 1271 } 1272 } 1273 1274 /** Works on the bundled repository. That is using the variables 1275 BUNDLED_EXTENSIONS and BUNDLED_EXTENSIONS_USER. 1276 */ 1277 void ExtensionManager::synchronizeBundledPrereg( 1278 Reference<task::XAbortChannel> const & xAbortChannel, 1279 Reference<ucb::XCommandEnvironment> const & xCmdEnv ) 1280 throw (deploy::DeploymentException, 1281 uno::RuntimeException) 1282 { 1283 try 1284 { 1285 String sSynchronizingBundled(StrSyncRepository::get()); 1286 sSynchronizingBundled.SearchAndReplaceAllAscii( "%NAME", OUSTR("bundled")); 1287 dp_misc::ProgressLevel progressBundled(xCmdEnv, sSynchronizingBundled); 1288 1289 Reference<deploy::XPackageManagerFactory> xPackageManagerFactory( 1290 deploy::thePackageManagerFactory::get(m_xContext)); 1291 1292 Reference<deploy::XPackageManager> xMgr = 1293 xPackageManagerFactory->getPackageManager(OUSTR("bundled_prereg")); 1294 xMgr->synchronize(xAbortChannel, xCmdEnv); 1295 progressBundled.update(OUSTR("\n\n")); 1296 1297 uno::Sequence<Reference<deploy::XPackage> > extensions = xMgr->getDeployedPackages( 1298 xAbortChannel, xCmdEnv); 1299 try 1300 { 1301 for (sal_Int32 i = 0; i < extensions.getLength(); i++) 1302 { 1303 extensions[i]->registerPackage(true, xAbortChannel, xCmdEnv); 1304 } 1305 } 1306 catch (...) 1307 { 1308 OSL_ASSERT(0); 1309 } 1310 OUString lastSyncBundled(RTL_CONSTASCII_USTRINGPARAM( 1311 "$BUNDLED_EXTENSIONS_PREREG/lastsynchronized")); 1312 writeLastModified(lastSyncBundled, xCmdEnv); 1313 1314 } catch (deploy::DeploymentException& ) { 1315 throw; 1316 } catch (ucb::CommandFailedException & ) { 1317 throw; 1318 } catch (ucb::CommandAbortedException & ) { 1319 throw; 1320 } catch (lang::IllegalArgumentException &) { 1321 throw; 1322 } catch (uno::RuntimeException &) { 1323 throw; 1324 } catch (...) { 1325 uno::Any exc = ::cppu::getCaughtException(); 1326 throw deploy::DeploymentException( 1327 OUSTR("Extension Manager: exception in synchronize"), 1328 static_cast<OWeakObject*>(this), exc); 1329 } 1330 } 1331 1332 sal_Bool ExtensionManager::synchronize( 1333 Reference<task::XAbortChannel> const & xAbortChannel, 1334 Reference<ucb::XCommandEnvironment> const & xCmdEnv ) 1335 throw (deploy::DeploymentException, 1336 ucb::CommandFailedException, 1337 ucb::CommandAbortedException, 1338 lang::IllegalArgumentException, 1339 uno::RuntimeException) 1340 { 1341 try 1342 { 1343 sal_Bool bModified = sal_False; 1344 1345 ::osl::MutexGuard guard(getMutex()); 1346 String sSynchronizingShared(StrSyncRepository::get()); 1347 sSynchronizingShared.SearchAndReplaceAllAscii( "%NAME", OUSTR("shared")); 1348 dp_misc::ProgressLevel progressShared(xCmdEnv, sSynchronizingShared); 1349 bModified = getSharedRepository()->synchronize(xAbortChannel, xCmdEnv); 1350 progressShared.update(OUSTR("\n\n")); 1351 1352 String sSynchronizingBundled(StrSyncRepository::get()); 1353 sSynchronizingBundled.SearchAndReplaceAllAscii( "%NAME", OUSTR("bundled")); 1354 dp_misc::ProgressLevel progressBundled(xCmdEnv, sSynchronizingBundled); 1355 bModified |= getBundledRepository()->synchronize(xAbortChannel, xCmdEnv); 1356 progressBundled.update(OUSTR("\n\n")); 1357 1358 //Always determine the active extension. This is necessary for the 1359 //first-start optimization. The setup creates the registration data for the 1360 //bundled extensions (brand_layer/share/prereg/bundled), which is copied to the user 1361 //installation (user_installation/extension/bundled) when a user starts OOo 1362 //for the first time after running setup. All bundled extensions are registered 1363 //at that moment. However, extensions with the same identifier can be in the 1364 //shared or user repository, in which case the respective bundled extensions must 1365 //be revoked. 1366 try 1367 { 1368 const uno::Sequence<uno::Sequence<Reference<deploy::XPackage> > > 1369 seqSeqExt = getAllExtensions(xAbortChannel, xCmdEnv); 1370 for (sal_Int32 i = 0; i < seqSeqExt.getLength(); i++) 1371 { 1372 uno::Sequence<Reference<deploy::XPackage> > const & seqExt = 1373 seqSeqExt[i]; 1374 activateExtension(seqExt, isUserDisabled(seqExt), true, 1375 xAbortChannel, xCmdEnv); 1376 } 1377 } 1378 catch (...) 1379 { 1380 //We catch the exception, so we can write the lastmodified file 1381 //so we will no repeat this everytime OOo starts. 1382 OSL_ENSURE(0, "Extensions Manager: synchronize"); 1383 } 1384 OUString lastSyncBundled(RTL_CONSTASCII_USTRINGPARAM( 1385 "$BUNDLED_EXTENSIONS_USER/lastsynchronized")); 1386 writeLastModified(lastSyncBundled, xCmdEnv); 1387 OUString lastSyncShared(RTL_CONSTASCII_USTRINGPARAM( 1388 "$SHARED_EXTENSIONS_USER/lastsynchronized")); 1389 writeLastModified(lastSyncShared, xCmdEnv); 1390 return bModified; 1391 } catch (deploy::DeploymentException& ) { 1392 throw; 1393 } catch (ucb::CommandFailedException & ) { 1394 throw; 1395 } catch (ucb::CommandAbortedException & ) { 1396 throw; 1397 } catch (lang::IllegalArgumentException &) { 1398 throw; 1399 } catch (uno::RuntimeException &) { 1400 throw; 1401 } catch (...) { 1402 uno::Any exc = ::cppu::getCaughtException(); 1403 throw deploy::DeploymentException( 1404 OUSTR("Extension Manager: exception in synchronize"), 1405 static_cast<OWeakObject*>(this), exc); 1406 } 1407 } 1408 1409 // Notify the user when a new extension is to be installed. This is only the 1410 // case when one uses the system integration to install an extension (double 1411 // clicking on .oxt file etc.)). The function must only be called if there is no 1412 // extension with the same identifier already deployed. Then the checkUpdate 1413 // function will inform the user that the extension is about to be installed In 1414 // case the user cancels the installation a CommandFailed exception is 1415 // thrown. 1416 void ExtensionManager::checkInstall( 1417 OUString const & displayName, 1418 Reference<ucb::XCommandEnvironment> const & cmdEnv) 1419 { 1420 uno::Any request( 1421 deploy::InstallException( 1422 OUSTR("Extension ") + displayName + 1423 OUSTR(" is about to be installed."), 1424 static_cast<OWeakObject *>(this), displayName)); 1425 bool approve = false, abort = false; 1426 if (! dp_misc::interactContinuation( 1427 request, task::XInteractionApprove::static_type(), 1428 cmdEnv, &approve, &abort )) 1429 { 1430 OSL_ASSERT( !approve && !abort ); 1431 throw deploy::DeploymentException( 1432 dp_misc::getResourceString(RID_STR_ERROR_WHILE_ADDING) + displayName, 1433 static_cast<OWeakObject *>(this), request ); 1434 } 1435 if (abort || !approve) 1436 throw ucb::CommandFailedException( 1437 dp_misc::getResourceString(RID_STR_ERROR_WHILE_ADDING) + displayName, 1438 static_cast<OWeakObject *>(this), request ); 1439 } 1440 1441 /* The function will make the user interaction in case there is an extension 1442 installed with the same id. This function may only be called if there is already 1443 an extension. 1444 */ 1445 void ExtensionManager::checkUpdate( 1446 OUString const & newVersion, 1447 OUString const & newDisplayName, 1448 Reference<deploy::XPackage> const & oldExtension, 1449 Reference<ucb::XCommandEnvironment> const & xCmdEnv ) 1450 { 1451 // package already deployed, interact --force: 1452 uno::Any request( 1453 (deploy::VersionException( 1454 dp_misc::getResourceString( 1455 RID_STR_PACKAGE_ALREADY_ADDED ) + newDisplayName, 1456 static_cast<OWeakObject *>(this), newVersion, newDisplayName, 1457 oldExtension ) ) ); 1458 bool replace = false, abort = false; 1459 if (! dp_misc::interactContinuation( 1460 request, task::XInteractionApprove::static_type(), 1461 xCmdEnv, &replace, &abort )) { 1462 OSL_ASSERT( !replace && !abort ); 1463 throw deploy::DeploymentException( 1464 dp_misc::getResourceString( 1465 RID_STR_ERROR_WHILE_ADDING) + newDisplayName, 1466 static_cast<OWeakObject *>(this), request ); 1467 } 1468 if (abort || !replace) 1469 throw ucb::CommandFailedException( 1470 dp_misc::getResourceString( 1471 RID_STR_PACKAGE_ALREADY_ADDED) + newDisplayName, 1472 static_cast<OWeakObject *>(this), request ); 1473 } 1474 1475 Reference<deploy::XPackage> ExtensionManager::getTempExtension( 1476 OUString const & url, 1477 Reference<task::XAbortChannel> const & xAbortChannel, 1478 Reference<ucb::XCommandEnvironment> const & /*xCmdEnv*/) 1479 1480 { 1481 Reference<ucb::XCommandEnvironment> tmpCmdEnvA(new TmpRepositoryCommandEnv()); 1482 Reference<deploy::XPackage> xTmpPackage = getTmpRepository()->addPackage( 1483 url, uno::Sequence<beans::NamedValue>(),OUString(), xAbortChannel, tmpCmdEnvA); 1484 if (!xTmpPackage.is()) 1485 { 1486 throw deploy::DeploymentException( 1487 OUSTR("Extension Manager: Failed to create temporary XPackage for url: ") + url, 1488 static_cast<OWeakObject*>(this), uno::Any()); 1489 1490 } 1491 return xTmpPackage; 1492 } 1493 1494 uno::Sequence<Reference<deploy::XPackage> > SAL_CALL 1495 ExtensionManager::getExtensionsWithUnacceptedLicenses( 1496 OUString const & repository, 1497 Reference<ucb::XCommandEnvironment> const & xCmdEnv) 1498 throw (deploy::DeploymentException, 1499 uno::RuntimeException) 1500 { 1501 Reference<deploy::XPackageManager> 1502 xPackageManager = getPackageManager(repository); 1503 ::osl::MutexGuard guard(getMutex()); 1504 return xPackageManager->getExtensionsWithUnacceptedLicenses(xCmdEnv); 1505 } 1506 1507 sal_Bool ExtensionManager::isReadOnlyRepository(::rtl::OUString const & repository) 1508 throw (uno::RuntimeException) 1509 { 1510 return getPackageManager(repository)->isReadOnly(); 1511 } 1512 //------------------------------------------------------------------------------ 1513 //------------------------------------------------------------------------------ 1514 //------------------------------------------------------------------------------ 1515 1516 namespace sdecl = comphelper::service_decl; 1517 sdecl::class_<ExtensionManager> servicePIP; 1518 extern sdecl::ServiceDecl const serviceDecl( 1519 servicePIP, 1520 // a private one: 1521 "com.sun.star.comp.deployment.ExtensionManager", 1522 "com.sun.star.comp.deployment.ExtensionManager"); 1523 1524 //------------------------------------------------------------------------------ 1525 bool singleton_entries( 1526 uno::Reference< registry::XRegistryKey > const & xRegistryKey ) 1527 { 1528 try { 1529 uno::Reference< registry::XRegistryKey > xKey( 1530 xRegistryKey->createKey( 1531 serviceDecl.getImplementationName() + 1532 // xxx todo: use future generated function to get singleton name 1533 OUSTR("/UNO/SINGLETONS/" 1534 "com.sun.star.deployment.ExtensionManager") ) ); 1535 xKey->setStringValue( serviceDecl.getSupportedServiceNames()[0] ); 1536 return true; 1537 } 1538 catch (registry::InvalidRegistryException & exc) { 1539 (void) exc; // avoid warnings 1540 OSL_ENSURE( 0, ::rtl::OUStringToOString( 1541 exc.Message, RTL_TEXTENCODING_UTF8 ).getStr() ); 1542 return false; 1543 } 1544 } 1545 1546 // XModifyBroadcaster 1547 //______________________________________________________________________________ 1548 void ExtensionManager::addModifyListener( 1549 Reference<util::XModifyListener> const & xListener ) 1550 throw (uno::RuntimeException) 1551 { 1552 check(); 1553 rBHelper.addListener( ::getCppuType( &xListener ), xListener ); 1554 } 1555 1556 //______________________________________________________________________________ 1557 void ExtensionManager::removeModifyListener( 1558 Reference<util::XModifyListener> const & xListener ) 1559 throw (uno::RuntimeException) 1560 { 1561 check(); 1562 rBHelper.removeListener( ::getCppuType( &xListener ), xListener ); 1563 } 1564 1565 void ExtensionManager::check() 1566 { 1567 ::osl::MutexGuard guard( getMutex() ); 1568 if (rBHelper.bInDispose || rBHelper.bDisposed) { 1569 throw lang::DisposedException( 1570 OUSTR("ExtensionManager instance has already been disposed!"), 1571 static_cast<OWeakObject *>(this) ); 1572 } 1573 } 1574 1575 void ExtensionManager::fireModified() 1576 { 1577 ::cppu::OInterfaceContainerHelper * pContainer = rBHelper.getContainer( 1578 util::XModifyListener::static_type() ); 1579 if (pContainer != 0) { 1580 pContainer->forEach<util::XModifyListener>( 1581 boost::bind(&util::XModifyListener::modified, _1, 1582 lang::EventObject(static_cast<OWeakObject *>(this))) ); 1583 } 1584 } 1585 1586 } // namespace dp_manager 1587 1588 1589