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_ucb.hxx" 26 27 /************************************************************************** 28 TODO 29 ************************************************************************** 30 31 *************************************************************************/ 32 33 #include <osl/diagnose.h> 34 #include "osl/doublecheckedlocking.h" 35 #include <rtl/uri.hxx> 36 #include <rtl/ustrbuf.hxx> 37 #include <ucbhelper/contentidentifier.hxx> 38 #include <ucbhelper/propertyvalueset.hxx> 39 #include <ucbhelper/simpleinteractionrequest.hxx> 40 #include <ucbhelper/cancelcommandexecution.hxx> 41 42 #include <com/sun/star/beans/PropertyAttribute.hpp> 43 #include <com/sun/star/beans/PropertySetInfoChange.hpp> 44 #include <com/sun/star/beans/PropertySetInfoChangeEvent.hpp> 45 #include <com/sun/star/beans/PropertyValue.hpp> 46 #include <com/sun/star/io/XActiveDataSink.hpp> 47 #include <com/sun/star/io/XOutputStream.hpp> 48 #include <com/sun/star/lang/IllegalAccessException.hpp> 49 #include <com/sun/star/task/PasswordContainerInteractionHandler.hpp> 50 #include <com/sun/star/ucb/CommandEnvironment.hpp> 51 #include <com/sun/star/ucb/CommandFailedException.hpp> 52 #include <com/sun/star/ucb/ContentInfoAttribute.hpp> 53 #include <com/sun/star/ucb/InsertCommandArgument.hpp> 54 #include <com/sun/star/ucb/InteractiveBadTransferURLException.hpp> 55 #include <com/sun/star/ucb/InteractiveAugmentedIOException.hpp> 56 #include "com/sun/star/ucb/InteractiveLockingLockedException.hpp" 57 #include "com/sun/star/ucb/InteractiveLockingLockExpiredException.hpp" 58 #include "com/sun/star/ucb/InteractiveLockingNotLockedException.hpp" 59 #include <com/sun/star/ucb/InteractiveNetworkConnectException.hpp> 60 #include <com/sun/star/ucb/InteractiveNetworkGeneralException.hpp> 61 #include <com/sun/star/ucb/InteractiveNetworkReadException.hpp> 62 #include <com/sun/star/ucb/InteractiveNetworkResolveNameException.hpp> 63 #include <com/sun/star/ucb/InteractiveNetworkWriteException.hpp> 64 #include <com/sun/star/ucb/MissingInputStreamException.hpp> 65 #include <com/sun/star/ucb/MissingPropertiesException.hpp> 66 #include <com/sun/star/ucb/NameClash.hpp> 67 #include <com/sun/star/ucb/NameClashException.hpp> 68 #include <com/sun/star/ucb/OpenCommandArgument2.hpp> 69 #include <com/sun/star/ucb/OpenMode.hpp> 70 #include <com/sun/star/ucb/PostCommandArgument2.hpp> 71 #include <com/sun/star/ucb/TransferInfo.hpp> 72 #include <com/sun/star/ucb/UnsupportedCommandException.hpp> 73 #include <com/sun/star/ucb/UnsupportedDataSinkException.hpp> 74 #include <com/sun/star/ucb/UnsupportedNameClashException.hpp> 75 #include <com/sun/star/ucb/UnsupportedOpenModeException.hpp> 76 #include <com/sun/star/ucb/XCommandInfo.hpp> 77 #include <com/sun/star/ucb/XPersistentPropertySet.hpp> 78 #include <com/sun/star/uno/XComponentContext.hpp> 79 80 #include "webdavcontent.hxx" 81 #include "webdavprovider.hxx" 82 #include "webdavresultset.hxx" 83 #include "ContentProperties.hxx" 84 #include "SerfUri.hxx" 85 #include "UCBDeadPropertyValue.hxx" 86 87 using namespace com::sun::star; 88 using namespace http_dav_ucp; 89 90 //========================================================================= 91 //========================================================================= 92 // 93 // Content Implementation. 94 // 95 //========================================================================= 96 //========================================================================= 97 98 //========================================================================= 99 // ctr for content on an existing webdav resource 100 Content::Content( 101 const uno::Reference< lang::XMultiServiceFactory >& rxSMgr, 102 ContentProvider* pProvider, 103 const uno::Reference< ucb::XContentIdentifier >& Identifier, 104 rtl::Reference< DAVSessionFactory > const & rSessionFactory ) 105 throw ( ucb::ContentCreationException ) 106 : ContentImplHelper( rxSMgr, pProvider, Identifier ), 107 m_eResourceType( UNKNOWN ), 108 m_pProvider( pProvider ), 109 m_bTransient( false ), 110 m_bCollection( false ), 111 m_bDidGetOrHead( false ) 112 { 113 try 114 { 115 m_xResAccess.reset( new DAVResourceAccess( 116 rxSMgr, 117 rSessionFactory, 118 Identifier->getContentIdentifier() ) ); 119 120 SerfUri aURI( Identifier->getContentIdentifier() ); 121 m_aEscapedTitle = aURI.GetPathBaseName(); 122 } 123 catch ( DAVException const & ) 124 { 125 throw ucb::ContentCreationException(); 126 } 127 } 128 129 //========================================================================= 130 // ctr for content on an non-existing webdav resource 131 Content::Content( 132 const uno::Reference< lang::XMultiServiceFactory >& rxSMgr, 133 ContentProvider* pProvider, 134 const uno::Reference< ucb::XContentIdentifier >& Identifier, 135 rtl::Reference< DAVSessionFactory > const & rSessionFactory, 136 sal_Bool isCollection ) 137 throw ( ucb::ContentCreationException ) 138 : ContentImplHelper( rxSMgr, pProvider, Identifier ), 139 m_eResourceType( UNKNOWN ), 140 m_pProvider( pProvider ), 141 m_bTransient( true ), 142 m_bCollection( isCollection ), 143 m_bDidGetOrHead( false ) 144 { 145 try 146 { 147 m_xResAccess.reset( new DAVResourceAccess( 148 rxSMgr, rSessionFactory, Identifier->getContentIdentifier() ) ); 149 } 150 catch ( DAVException const & ) 151 { 152 throw ucb::ContentCreationException(); 153 } 154 155 // Do not set m_aEscapedTitle here! Content::insert relays on this!!! 156 } 157 158 //========================================================================= 159 // virtual 160 Content::~Content() 161 { 162 } 163 164 //========================================================================= 165 // 166 // XInterface methods. 167 // 168 //========================================================================= 169 170 // virtual 171 void SAL_CALL Content::acquire() 172 throw( ) 173 { 174 ContentImplHelper::acquire(); 175 } 176 177 //========================================================================= 178 // virtual 179 void SAL_CALL Content::release() 180 throw( ) 181 { 182 ContentImplHelper::release(); 183 } 184 185 //========================================================================= 186 // virtual 187 uno::Any SAL_CALL Content::queryInterface( const uno::Type & rType ) 188 throw ( uno::RuntimeException ) 189 { 190 // Note: isFolder may require network activities! So call it only 191 // if it is really necessary!!! 192 uno::Any aRet = cppu::queryInterface( 193 rType, 194 static_cast< ucb::XContentCreator * >( this ) ); 195 if ( aRet.hasValue() ) 196 { 197 try 198 { 199 uno::Reference< beans::XPropertySet > const xProps( 200 m_xSMgr, uno::UNO_QUERY_THROW ); 201 uno::Reference< uno::XComponentContext > xCtx; 202 xCtx.set( xProps->getPropertyValue( 203 rtl::OUString( 204 RTL_CONSTASCII_USTRINGPARAM( "DefaultContext" ) ) ), 205 uno::UNO_QUERY_THROW ); 206 207 uno::Reference< task::XInteractionHandler > xIH( 208 task::PasswordContainerInteractionHandler::create( xCtx ) ); 209 210 // Supply a command env to isFolder() that contains an interaction 211 // handler that uses the password container service to obtain 212 // credentials without displaying a password gui. 213 214 uno::Reference< ucb::XCommandEnvironment > xCmdEnv( 215 ucb::CommandEnvironment::create( 216 xCtx, 217 xIH, 218 uno::Reference< ucb::XProgressHandler >() ) ); 219 220 return isFolder( xCmdEnv ) ? aRet : uno::Any(); 221 } 222 catch ( uno::RuntimeException const & ) 223 { 224 throw; 225 } 226 catch ( uno::Exception const & ) 227 { 228 return uno::Any(); 229 } 230 } 231 return aRet.hasValue() ? aRet : ContentImplHelper::queryInterface( rType ); 232 } 233 234 //========================================================================= 235 // 236 // XTypeProvider methods. 237 // 238 //========================================================================= 239 240 XTYPEPROVIDER_COMMON_IMPL( Content ); 241 242 //========================================================================= 243 // virtual 244 uno::Sequence< uno::Type > SAL_CALL Content::getTypes() 245 throw( uno::RuntimeException ) 246 { 247 sal_Bool bFolder = sal_False; 248 try 249 { 250 bFolder 251 = isFolder( uno::Reference< ucb::XCommandEnvironment >() ); 252 } 253 catch ( uno::RuntimeException const & ) 254 { 255 throw; 256 } 257 catch ( uno::Exception const & ) 258 { 259 } 260 261 cppu::OTypeCollection * pCollection = 0; 262 263 if ( bFolder ) 264 { 265 static cppu::OTypeCollection* pFolderTypes = 0; 266 267 pCollection = pFolderTypes; 268 if ( !pCollection ) 269 { 270 osl::Guard< osl::Mutex > aGuard( osl::Mutex::getGlobalMutex() ); 271 272 pCollection = pFolderTypes; 273 if ( !pCollection ) 274 { 275 static cppu::OTypeCollection aCollection( 276 CPPU_TYPE_REF( lang::XTypeProvider ), 277 CPPU_TYPE_REF( lang::XServiceInfo ), 278 CPPU_TYPE_REF( lang::XComponent ), 279 CPPU_TYPE_REF( ucb::XContent ), 280 CPPU_TYPE_REF( ucb::XCommandProcessor ), 281 CPPU_TYPE_REF( beans::XPropertiesChangeNotifier ), 282 CPPU_TYPE_REF( ucb::XCommandInfoChangeNotifier ), 283 CPPU_TYPE_REF( beans::XPropertyContainer ), 284 CPPU_TYPE_REF( beans::XPropertySetInfoChangeNotifier ), 285 CPPU_TYPE_REF( container::XChild ), 286 CPPU_TYPE_REF( ucb::XContentCreator ) ); // !! 287 pCollection = &aCollection; 288 OSL_DOUBLE_CHECKED_LOCKING_MEMORY_BARRIER(); 289 pFolderTypes = pCollection; 290 } 291 } 292 else { 293 OSL_DOUBLE_CHECKED_LOCKING_MEMORY_BARRIER(); 294 } 295 } 296 else 297 { 298 static cppu::OTypeCollection* pDocumentTypes = 0; 299 300 pCollection = pDocumentTypes; 301 if ( !pCollection ) 302 { 303 osl::Guard< osl::Mutex > aGuard( osl::Mutex::getGlobalMutex() ); 304 305 pCollection = pDocumentTypes; 306 if ( !pCollection ) 307 { 308 static cppu::OTypeCollection aCollection( 309 CPPU_TYPE_REF( lang::XTypeProvider ), 310 CPPU_TYPE_REF( lang::XServiceInfo ), 311 CPPU_TYPE_REF( lang::XComponent ), 312 CPPU_TYPE_REF( ucb::XContent ), 313 CPPU_TYPE_REF( ucb::XCommandProcessor ), 314 CPPU_TYPE_REF( beans::XPropertiesChangeNotifier ), 315 CPPU_TYPE_REF( ucb::XCommandInfoChangeNotifier ), 316 CPPU_TYPE_REF( beans::XPropertyContainer ), 317 CPPU_TYPE_REF( beans::XPropertySetInfoChangeNotifier ), 318 CPPU_TYPE_REF( container::XChild ) ); 319 pCollection = &aCollection; 320 OSL_DOUBLE_CHECKED_LOCKING_MEMORY_BARRIER(); 321 pDocumentTypes = pCollection; 322 } 323 } 324 else { 325 OSL_DOUBLE_CHECKED_LOCKING_MEMORY_BARRIER(); 326 } 327 } 328 329 return (*pCollection).getTypes(); 330 } 331 332 //========================================================================= 333 // 334 // XServiceInfo methods. 335 // 336 //========================================================================= 337 338 // virtual 339 rtl::OUString SAL_CALL Content::getImplementationName() 340 throw( uno::RuntimeException ) 341 { 342 return rtl::OUString::createFromAscii( 343 "com.sun.star.comp.ucb.WebDAVContent" ); 344 } 345 346 //========================================================================= 347 // virtual 348 uno::Sequence< rtl::OUString > SAL_CALL Content::getSupportedServiceNames() 349 throw( uno::RuntimeException ) 350 { 351 uno::Sequence< rtl::OUString > aSNS( 1 ); 352 aSNS.getArray()[ 0 ] 353 = rtl::OUString::createFromAscii( WEBDAV_CONTENT_SERVICE_NAME ); 354 return aSNS; 355 } 356 357 //========================================================================= 358 // 359 // XContent methods. 360 // 361 //========================================================================= 362 363 // virtual 364 rtl::OUString SAL_CALL Content::getContentType() 365 throw( uno::RuntimeException ) 366 { 367 sal_Bool bFolder = sal_False; 368 try 369 { 370 bFolder 371 = isFolder( uno::Reference< ucb::XCommandEnvironment >() ); 372 } 373 catch ( uno::RuntimeException const & ) 374 { 375 throw; 376 } 377 catch ( uno::Exception const & ) 378 { 379 } 380 381 if ( bFolder ) 382 return rtl::OUString::createFromAscii( WEBDAV_COLLECTION_TYPE ); 383 384 return rtl::OUString::createFromAscii( WEBDAV_CONTENT_TYPE ); 385 } 386 387 //========================================================================= 388 // 389 // XCommandProcessor methods. 390 // 391 //========================================================================= 392 393 // virtual 394 uno::Any SAL_CALL Content::execute( 395 const ucb::Command& aCommand, 396 sal_Int32 /*CommandId*/, 397 const uno::Reference< ucb::XCommandEnvironment >& Environment ) 398 throw( uno::Exception, 399 ucb::CommandAbortedException, 400 uno::RuntimeException ) 401 { 402 OSL_TRACE( ">>>>> Content::execute: start: command: %s, env: %s", 403 rtl::OUStringToOString( aCommand.Name, 404 RTL_TEXTENCODING_UTF8 ).getStr(), 405 Environment.is() ? "present" : "missing" ); 406 407 uno::Any aRet; 408 409 if ( aCommand.Name.equalsAsciiL( 410 RTL_CONSTASCII_STRINGPARAM( "getPropertyValues" ) ) ) 411 { 412 ////////////////////////////////////////////////////////////////// 413 // getPropertyValues 414 ////////////////////////////////////////////////////////////////// 415 416 uno::Sequence< beans::Property > Properties; 417 if ( !( aCommand.Argument >>= Properties ) ) 418 { 419 ucbhelper::cancelCommandExecution( 420 uno::makeAny( lang::IllegalArgumentException( 421 rtl::OUString::createFromAscii( 422 "Wrong argument type!" ), 423 static_cast< cppu::OWeakObject * >( this ), 424 -1 ) ), 425 Environment ); 426 // Unreachable 427 } 428 429 aRet <<= getPropertyValues( Properties, Environment ); 430 } 431 else if ( aCommand.Name.equalsAsciiL( 432 RTL_CONSTASCII_STRINGPARAM( "setPropertyValues" ) ) ) 433 { 434 ////////////////////////////////////////////////////////////////// 435 // setPropertyValues 436 ////////////////////////////////////////////////////////////////// 437 438 uno::Sequence< beans::PropertyValue > aProperties; 439 if ( !( aCommand.Argument >>= aProperties ) ) 440 { 441 ucbhelper::cancelCommandExecution( 442 uno::makeAny( lang::IllegalArgumentException( 443 rtl::OUString::createFromAscii( 444 "Wrong argument type!" ), 445 static_cast< cppu::OWeakObject * >( this ), 446 -1 ) ), 447 Environment ); 448 // Unreachable 449 } 450 451 if ( !aProperties.getLength() ) 452 { 453 ucbhelper::cancelCommandExecution( 454 uno::makeAny( lang::IllegalArgumentException( 455 rtl::OUString::createFromAscii( 456 "No properties!" ), 457 static_cast< cppu::OWeakObject * >( this ), 458 -1 ) ), 459 Environment ); 460 // Unreachable 461 } 462 463 aRet <<= setPropertyValues( aProperties, Environment ); 464 } 465 else if ( aCommand.Name.equalsAsciiL( 466 RTL_CONSTASCII_STRINGPARAM( "getPropertySetInfo" ) ) ) 467 { 468 ////////////////////////////////////////////////////////////////// 469 // getPropertySetInfo 470 ////////////////////////////////////////////////////////////////// 471 472 // Note: Implemented by base class. 473 aRet <<= getPropertySetInfo( Environment, 474 sal_False /* don't cache data */ ); 475 } 476 else if ( aCommand.Name.equalsAsciiL( 477 RTL_CONSTASCII_STRINGPARAM( "getCommandInfo" ) ) ) 478 { 479 ////////////////////////////////////////////////////////////////// 480 // getCommandInfo 481 ////////////////////////////////////////////////////////////////// 482 483 // Note: Implemented by base class. 484 aRet <<= getCommandInfo( Environment, sal_False ); 485 } 486 else if ( aCommand.Name.equalsAsciiL( 487 RTL_CONSTASCII_STRINGPARAM( "open" ) ) ) 488 { 489 ////////////////////////////////////////////////////////////////// 490 // open 491 ////////////////////////////////////////////////////////////////// 492 493 ucb::OpenCommandArgument2 aOpenCommand; 494 if ( !( aCommand.Argument >>= aOpenCommand ) ) 495 { 496 ucbhelper::cancelCommandExecution( 497 uno::makeAny( lang::IllegalArgumentException( 498 rtl::OUString::createFromAscii( 499 "Wrong argument type!" ), 500 static_cast< cppu::OWeakObject * >( this ), 501 -1 ) ), 502 Environment ); 503 // Unreachable 504 } 505 506 aRet = open( aOpenCommand, Environment ); 507 } 508 else if ( aCommand.Name.equalsAsciiL( 509 RTL_CONSTASCII_STRINGPARAM( "insert" ) ) ) 510 { 511 ////////////////////////////////////////////////////////////////// 512 // insert 513 ////////////////////////////////////////////////////////////////// 514 515 ucb::InsertCommandArgument arg; 516 if ( !( aCommand.Argument >>= arg ) ) 517 { 518 ucbhelper::cancelCommandExecution( 519 uno::makeAny( lang::IllegalArgumentException( 520 rtl::OUString::createFromAscii( 521 "Wrong argument type!" ), 522 static_cast< cppu::OWeakObject * >( this ), 523 -1 ) ), 524 Environment ); 525 // Unreachable 526 } 527 528 insert( arg.Data, arg.ReplaceExisting, Environment ); 529 } 530 else if ( aCommand.Name.equalsAsciiL( 531 RTL_CONSTASCII_STRINGPARAM( "delete" ) ) ) 532 { 533 ////////////////////////////////////////////////////////////////// 534 // delete 535 ////////////////////////////////////////////////////////////////// 536 537 sal_Bool bDeletePhysical = sal_False; 538 aCommand.Argument >>= bDeletePhysical; 539 540 // KSO: Ignore parameter and destroy the content, if you don't support 541 // putting objects into trashcan. ( Since we do not have a trash can 542 // service yet (src603), you actually have no other choice. ) 543 // if ( bDeletePhysical ) 544 // { 545 try 546 { 547 std::auto_ptr< DAVResourceAccess > xResAccess; 548 { 549 osl::Guard< osl::Mutex > aGuard( m_aMutex ); 550 xResAccess.reset( new DAVResourceAccess( *m_xResAccess.get() ) ); 551 } 552 xResAccess->DESTROY( Environment ); 553 { 554 osl::Guard< osl::Mutex > aGuard( m_aMutex ); 555 m_xResAccess.reset( new DAVResourceAccess( *xResAccess.get() ) ); 556 } 557 } 558 catch ( DAVException const & e ) 559 { 560 cancelCommandExecution( e, Environment, sal_True ); 561 // Unreachable 562 } 563 // } 564 565 // Propagate destruction. 566 destroy( bDeletePhysical ); 567 568 // Remove own and all children's Additional Core Properties. 569 removeAdditionalPropertySet( sal_True ); 570 } 571 else if ( aCommand.Name.equalsAsciiL( 572 RTL_CONSTASCII_STRINGPARAM( "transfer" ) ) 573 && isFolder( Environment ) ) 574 { 575 ////////////////////////////////////////////////////////////////// 576 // transfer 577 // ( Not available at documents ) 578 ////////////////////////////////////////////////////////////////// 579 580 ucb::TransferInfo transferArgs; 581 if ( !( aCommand.Argument >>= transferArgs ) ) 582 { 583 ucbhelper::cancelCommandExecution( 584 uno::makeAny( lang::IllegalArgumentException( 585 rtl::OUString::createFromAscii( 586 "Wrong argument type!" ), 587 static_cast< cppu::OWeakObject * >( this ), 588 -1 ) ), 589 Environment ); 590 // Unreachable 591 } 592 593 transfer( transferArgs, Environment ); 594 } 595 else if ( aCommand.Name.equalsAsciiL( 596 RTL_CONSTASCII_STRINGPARAM( "post" ) ) ) 597 { 598 ////////////////////////////////////////////////////////////////// 599 // post 600 ////////////////////////////////////////////////////////////////// 601 602 ucb::PostCommandArgument2 aArg; 603 if ( !( aCommand.Argument >>= aArg ) ) 604 { 605 ucbhelper::cancelCommandExecution( 606 uno::makeAny( lang::IllegalArgumentException( 607 rtl::OUString::createFromAscii( 608 "Wrong argument type!" ), 609 static_cast< cppu::OWeakObject * >( this ), 610 -1 ) ), 611 Environment ); 612 // Unreachable 613 } 614 615 post( aArg, Environment ); 616 } 617 else if ( aCommand.Name.equalsAsciiL( 618 RTL_CONSTASCII_STRINGPARAM( "lock" ) ) && 619 supportsExclusiveWriteLock( Environment ) ) 620 { 621 ////////////////////////////////////////////////////////////////// 622 // lock 623 ////////////////////////////////////////////////////////////////// 624 625 lock( Environment ); 626 } 627 else if ( aCommand.Name.equalsAsciiL( 628 RTL_CONSTASCII_STRINGPARAM( "unlock" ) ) && 629 supportsExclusiveWriteLock( Environment ) ) 630 { 631 ////////////////////////////////////////////////////////////////// 632 // unlock 633 ////////////////////////////////////////////////////////////////// 634 635 unlock( Environment ); 636 } 637 else if ( aCommand.Name.equalsAsciiL( 638 RTL_CONSTASCII_STRINGPARAM( "createNewContent" ) ) && 639 isFolder( Environment ) ) 640 { 641 ////////////////////////////////////////////////////////////////// 642 // createNewContent 643 ////////////////////////////////////////////////////////////////// 644 645 ucb::ContentInfo aArg; 646 if ( !( aCommand.Argument >>= aArg ) ) 647 { 648 ucbhelper::cancelCommandExecution( 649 uno::makeAny( lang::IllegalArgumentException( 650 rtl::OUString::createFromAscii( 651 "Wrong argument type!" ), 652 static_cast< cppu::OWeakObject * >( this ), 653 -1 ) ), 654 Environment ); 655 // Unreachable 656 } 657 658 aRet = uno::makeAny( createNewContent( aArg ) ); 659 } 660 else 661 { 662 ////////////////////////////////////////////////////////////////// 663 // Unsupported command 664 ////////////////////////////////////////////////////////////////// 665 666 ucbhelper::cancelCommandExecution( 667 uno::makeAny( ucb::UnsupportedCommandException( 668 aCommand.Name, 669 static_cast< cppu::OWeakObject * >( this ) ) ), 670 Environment ); 671 // Unreachable 672 } 673 674 OSL_TRACE( "<<<<< Content::execute: end: command: %s", 675 rtl::OUStringToOString( aCommand.Name, 676 RTL_TEXTENCODING_UTF8 ).getStr() ); 677 678 return aRet; 679 } 680 681 //========================================================================= 682 // virtual 683 void SAL_CALL Content::abort( sal_Int32 /*CommandId*/ ) 684 throw( uno::RuntimeException ) 685 { 686 try 687 { 688 std::auto_ptr< DAVResourceAccess > xResAccess; 689 { 690 osl::MutexGuard aGuard( m_aMutex ); 691 xResAccess.reset( new DAVResourceAccess( *m_xResAccess.get() ) ); 692 } 693 xResAccess->abort(); 694 { 695 osl::Guard< osl::Mutex > aGuard( m_aMutex ); 696 m_xResAccess.reset( new DAVResourceAccess( *xResAccess.get() ) ); 697 } 698 } 699 catch ( DAVException const & ) 700 { 701 // abort failed! 702 } 703 } 704 705 //========================================================================= 706 // 707 // XPropertyContainer methods. 708 // 709 //========================================================================= 710 711 // virtual 712 void SAL_CALL Content::addProperty( const rtl::OUString& Name, 713 sal_Int16 Attributes, 714 const uno::Any& DefaultValue ) 715 throw( beans::PropertyExistException, 716 beans::IllegalTypeException, 717 lang::IllegalArgumentException, 718 uno::RuntimeException ) 719 { 720 // if ( m_bTransient ) 721 // @@@ ??? 722 723 if ( !Name.getLength() ) 724 throw lang::IllegalArgumentException(); 725 726 // Check property type. 727 if ( !UCBDeadPropertyValue::supportsType( DefaultValue.getValueType() ) ) 728 { 729 OSL_ENSURE( sal_False, 730 "Content::addProperty - Unsupported property type!" ); 731 throw beans::IllegalTypeException(); 732 } 733 734 ////////////////////////////////////////////////////////////////////// 735 // Make sure a property with the requested name does not already 736 // exist in dynamic and static(!) properties. 737 ////////////////////////////////////////////////////////////////////// 738 739 // @@@ Need real command environment here, but where to get it from? 740 // XPropertyContainer interface should be replaced by 741 // XCommandProcessor commands! 742 uno::Reference< ucb::XCommandEnvironment > xEnv; 743 744 // Note: This requires network access! 745 if ( getPropertySetInfo( xEnv, sal_False /* don't cache data */ ) 746 ->hasPropertyByName( Name ) ) 747 { 748 // Property does already exist. 749 throw beans::PropertyExistException(); 750 } 751 752 ////////////////////////////////////////////////////////////////////// 753 // Add a new dynamic property. 754 ////////////////////////////////////////////////////////////////////// 755 756 ProppatchValue aValue( PROPSET, Name, DefaultValue ); 757 758 std::vector< ProppatchValue > aProppatchValues; 759 aProppatchValues.push_back( aValue ); 760 761 try 762 { 763 // Set property value at server. 764 std::auto_ptr< DAVResourceAccess > xResAccess; 765 { 766 osl::Guard< osl::Mutex > aGuard( m_aMutex ); 767 xResAccess.reset( new DAVResourceAccess( *m_xResAccess.get() ) ); 768 } 769 xResAccess->PROPPATCH( aProppatchValues, xEnv ); 770 { 771 osl::Guard< osl::Mutex > aGuard( m_aMutex ); 772 m_xResAccess.reset( new DAVResourceAccess( *xResAccess.get() ) ); 773 } 774 775 // Notify propertyset info change listeners. 776 beans::PropertySetInfoChangeEvent evt( 777 static_cast< cppu::OWeakObject * >( this ), 778 Name, 779 -1, // No handle available 780 beans::PropertySetInfoChange::PROPERTY_INSERTED ); 781 notifyPropertySetInfoChange( evt ); 782 } 783 catch ( DAVException const & e ) 784 { 785 if ( e.getStatus() == SC_FORBIDDEN ) 786 { 787 // Support for setting arbitrary dead properties is optional! 788 789 // Store property locally. 790 ContentImplHelper::addProperty( 791 Name, Attributes, DefaultValue ); 792 } 793 else 794 { 795 if ( shouldAccessNetworkAfterException( e ) ) 796 { 797 try 798 { 799 const ResourceType & rType = getResourceType( xEnv ); 800 switch ( rType ) 801 { 802 case UNKNOWN: 803 case DAV: 804 throw lang::IllegalArgumentException(); 805 806 case NON_DAV: 807 // Store property locally. 808 ContentImplHelper::addProperty( Name, 809 Attributes, 810 DefaultValue ); 811 break; 812 813 default: 814 OSL_ENSURE( sal_False, 815 "Content::addProperty - " 816 "Unsupported resource type!" ); 817 break; 818 } 819 } 820 catch ( uno::Exception const & ) 821 { 822 OSL_ENSURE( sal_False, 823 "Content::addProperty - " 824 "Unable to determine resource type!" ); 825 } 826 } 827 else 828 { 829 OSL_ENSURE( sal_False, 830 "Content::addProperty - " 831 "Unable to determine resource type!" ); 832 } 833 } 834 } 835 } 836 837 //========================================================================= 838 // virtual 839 void SAL_CALL Content::removeProperty( const rtl::OUString& Name ) 840 throw( beans::UnknownPropertyException, 841 beans::NotRemoveableException, 842 uno::RuntimeException ) 843 { 844 // @@@ Need real command environment here, but where to get it from? 845 // XPropertyContainer interface should be replaced by 846 // XCommandProcessor commands! 847 uno::Reference< ucb::XCommandEnvironment > xEnv; 848 849 #if 0 850 // @@@ REMOVEABLE z.Z. nicht richtig an der PropSetInfo gesetzt!!! 851 try 852 { 853 beans::Property aProp 854 = getPropertySetInfo( xEnv, sal_False /* don't cache data */ ) 855 ->getPropertyByName( Name ); 856 857 if ( !( aProp.Attributes & beans::PropertyAttribute::REMOVEABLE ) ) 858 { 859 // Not removeable! 860 throw beans::NotRemoveableException(); 861 } 862 } 863 catch ( beans::UnknownPropertyException const & ) 864 { 865 //OSL_ENSURE( sal_False, "removeProperty - Unknown property!" ); 866 throw; 867 } 868 #endif 869 870 ////////////////////////////////////////////////////////////////////// 871 // Try to remove property from server. 872 ////////////////////////////////////////////////////////////////////// 873 874 try 875 { 876 std::vector< ProppatchValue > aProppatchValues; 877 ProppatchValue aValue( PROPREMOVE, Name, uno::Any() ); 878 aProppatchValues.push_back( aValue ); 879 880 // Remove property value from server. 881 std::auto_ptr< DAVResourceAccess > xResAccess; 882 { 883 osl::Guard< osl::Mutex > aGuard( m_aMutex ); 884 xResAccess.reset( new DAVResourceAccess( *m_xResAccess.get() ) ); 885 } 886 xResAccess->PROPPATCH( aProppatchValues, xEnv ); 887 { 888 osl::Guard< osl::Mutex > aGuard( m_aMutex ); 889 m_xResAccess.reset( new DAVResourceAccess( *xResAccess.get() ) ); 890 } 891 892 // Notify propertyset info change listeners. 893 beans::PropertySetInfoChangeEvent evt( 894 static_cast< cppu::OWeakObject * >( this ), 895 Name, 896 -1, // No handle available 897 beans::PropertySetInfoChange::PROPERTY_REMOVED ); 898 notifyPropertySetInfoChange( evt ); 899 } 900 catch ( DAVException const & e ) 901 { 902 if ( e.getStatus() == SC_FORBIDDEN ) 903 { 904 // Support for setting arbitrary dead properties is optional! 905 906 // Try to remove property from local store. 907 ContentImplHelper::removeProperty( Name ); 908 } 909 else 910 { 911 if ( shouldAccessNetworkAfterException( e ) ) 912 { 913 try 914 { 915 const ResourceType & rType = getResourceType( xEnv ); 916 switch ( rType ) 917 { 918 case UNKNOWN: 919 case DAV: 920 throw beans::UnknownPropertyException(); 921 922 case NON_DAV: 923 // Try to remove property from local store. 924 ContentImplHelper::removeProperty( Name ); 925 break; 926 927 default: 928 OSL_ENSURE( sal_False, 929 "Content::removeProperty - " 930 "Unsupported resource type!" ); 931 break; 932 } 933 } 934 catch ( uno::Exception const & ) 935 { 936 OSL_ENSURE( sal_False, 937 "Content::removeProperty - " 938 "Unable to determine resource type!" ); 939 } 940 } 941 else 942 { 943 OSL_ENSURE( sal_False, 944 "Content::removeProperty - " 945 "Unable to determine resource type!" ); 946 // throw beans::UnknownPropertyException(); 947 } 948 } 949 } 950 } 951 952 //========================================================================= 953 // 954 // XContentCreator methods. 955 // 956 //========================================================================= 957 958 // virtual 959 uno::Sequence< ucb::ContentInfo > SAL_CALL 960 Content::queryCreatableContentsInfo() 961 throw( uno::RuntimeException ) 962 { 963 osl::Guard< osl::Mutex > aGuard( m_aMutex ); 964 965 uno::Sequence< ucb::ContentInfo > aSeq( 2 ); 966 967 // document. 968 aSeq.getArray()[ 0 ].Type 969 = rtl::OUString::createFromAscii( WEBDAV_CONTENT_TYPE ); 970 aSeq.getArray()[ 0 ].Attributes 971 = ucb::ContentInfoAttribute::INSERT_WITH_INPUTSTREAM 972 | ucb::ContentInfoAttribute::KIND_DOCUMENT; 973 974 beans::Property aProp; 975 m_pProvider->getProperty( 976 rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "Title" ) ), aProp ); 977 978 uno::Sequence< beans::Property > aDocProps( 1 ); 979 aDocProps.getArray()[ 0 ] = aProp; 980 aSeq.getArray()[ 0 ].Properties = aDocProps; 981 982 // folder. 983 aSeq.getArray()[ 1 ].Type 984 = rtl::OUString::createFromAscii( WEBDAV_COLLECTION_TYPE ); 985 aSeq.getArray()[ 1 ].Attributes 986 = ucb::ContentInfoAttribute::KIND_FOLDER; 987 988 uno::Sequence< beans::Property > aFolderProps( 1 ); 989 aFolderProps.getArray()[ 0 ] = aProp; 990 aSeq.getArray()[ 1 ].Properties = aFolderProps; 991 return aSeq; 992 } 993 994 //========================================================================= 995 // virtual 996 uno::Reference< ucb::XContent > SAL_CALL 997 Content::createNewContent( const ucb::ContentInfo& Info ) 998 throw( uno::RuntimeException ) 999 { 1000 osl::Guard< osl::Mutex > aGuard( m_aMutex ); 1001 1002 if ( !Info.Type.getLength() ) 1003 return uno::Reference< ucb::XContent >(); 1004 1005 if ( ( !Info.Type.equalsAsciiL( 1006 RTL_CONSTASCII_STRINGPARAM( WEBDAV_COLLECTION_TYPE ) ) ) 1007 && 1008 ( !Info.Type.equalsAsciiL( 1009 RTL_CONSTASCII_STRINGPARAM( WEBDAV_CONTENT_TYPE ) ) ) ) 1010 return uno::Reference< ucb::XContent >(); 1011 1012 rtl::OUString aURL = m_xIdentifier->getContentIdentifier(); 1013 1014 OSL_ENSURE( aURL.getLength() > 0, 1015 "WebdavContent::createNewContent - empty identifier!" ); 1016 1017 if ( ( aURL.lastIndexOf( '/' ) + 1 ) != aURL.getLength() ) 1018 aURL += rtl::OUString::createFromAscii( "/" ); 1019 1020 sal_Bool isCollection; 1021 if ( Info.Type.equalsAsciiL( 1022 RTL_CONSTASCII_STRINGPARAM( WEBDAV_COLLECTION_TYPE ) ) ) 1023 { 1024 aURL += rtl::OUString::createFromAscii( "New_Collection" ); 1025 isCollection = sal_True; 1026 } 1027 else 1028 { 1029 aURL += rtl::OUString::createFromAscii( "New_Content" ); 1030 isCollection = sal_False; 1031 } 1032 1033 uno::Reference< ucb::XContentIdentifier > xId( 1034 new ::ucbhelper::ContentIdentifier( m_xSMgr, aURL ) ); 1035 1036 // create the local content 1037 try 1038 { 1039 return new ::http_dav_ucp::Content( m_xSMgr, 1040 m_pProvider, 1041 xId, 1042 m_xResAccess->getSessionFactory(), 1043 isCollection ); 1044 } 1045 catch ( ucb::ContentCreationException & ) 1046 { 1047 return uno::Reference< ucb::XContent >(); 1048 } 1049 } 1050 1051 //========================================================================= 1052 // virtual 1053 rtl::OUString Content::getParentURL() 1054 { 1055 // <scheme>:// -> "" 1056 // <scheme>://foo -> "" 1057 // <scheme>://foo/ -> "" 1058 // <scheme>://foo/bar -> <scheme>://foo/ 1059 // <scheme>://foo/bar/ -> <scheme>://foo/ 1060 // <scheme>://foo/bar/abc -> <scheme>://foo/bar/ 1061 1062 rtl::OUString aURL = m_xIdentifier->getContentIdentifier(); 1063 1064 sal_Int32 nPos = aURL.lastIndexOf( '/' ); 1065 if ( nPos == ( aURL.getLength() - 1 ) ) 1066 { 1067 // Trailing slash found. Skip. 1068 nPos = aURL.lastIndexOf( '/', nPos ); 1069 } 1070 1071 sal_Int32 nPos1 = aURL.lastIndexOf( '/', nPos ); 1072 if ( nPos1 != -1 ) 1073 nPos1 = aURL.lastIndexOf( '/', nPos1 ); 1074 1075 if ( nPos1 == -1 ) 1076 return rtl::OUString(); 1077 1078 return rtl::OUString( aURL.copy( 0, nPos + 1 ) ); 1079 } 1080 1081 //========================================================================= 1082 // 1083 // Non-interface methods. 1084 // 1085 //========================================================================= 1086 1087 // static 1088 uno::Reference< sdbc::XRow > Content::getPropertyValues( 1089 const uno::Reference< lang::XMultiServiceFactory >& rSMgr, 1090 const uno::Sequence< beans::Property >& rProperties, 1091 const ContentProperties& rData, 1092 const rtl::Reference< ::ucbhelper::ContentProviderImplHelper >& rProvider, 1093 const rtl::OUString& rContentId ) 1094 { 1095 // Note: Empty sequence means "get values of all supported properties". 1096 1097 rtl::Reference< ::ucbhelper::PropertyValueSet > xRow 1098 = new ::ucbhelper::PropertyValueSet( rSMgr ); 1099 1100 sal_Int32 nCount = rProperties.getLength(); 1101 if ( nCount ) 1102 { 1103 uno::Reference< beans::XPropertySet > xAdditionalPropSet; 1104 sal_Bool bTriedToGetAdditonalPropSet = sal_False; 1105 1106 const beans::Property* pProps = rProperties.getConstArray(); 1107 for ( sal_Int32 n = 0; n < nCount; ++n ) 1108 { 1109 const beans::Property& rProp = pProps[ n ]; 1110 1111 // Process standard UCB, DAV and HTTP properties. 1112 const uno::Any & rValue = rData.getValue( rProp.Name ); 1113 if ( rValue.hasValue() ) 1114 { 1115 xRow->appendObject( rProp, rValue ); 1116 } 1117 else 1118 { 1119 // Process local Additional Properties. 1120 if ( !bTriedToGetAdditonalPropSet && !xAdditionalPropSet.is() ) 1121 { 1122 xAdditionalPropSet 1123 = uno::Reference< beans::XPropertySet >( 1124 rProvider->getAdditionalPropertySet( rContentId, 1125 sal_False ), 1126 uno::UNO_QUERY ); 1127 bTriedToGetAdditonalPropSet = sal_True; 1128 } 1129 1130 if ( !xAdditionalPropSet.is() || 1131 !xRow->appendPropertySetValue( 1132 xAdditionalPropSet, rProp ) ) 1133 { 1134 // Append empty entry. 1135 xRow->appendVoid( rProp ); 1136 } 1137 } 1138 } 1139 } 1140 else 1141 { 1142 // Append all standard UCB, DAV and HTTP properties. 1143 1144 const std::auto_ptr< PropertyValueMap > & xProps = rData.getProperties(); 1145 1146 PropertyValueMap::const_iterator it = xProps->begin(); 1147 PropertyValueMap::const_iterator end = xProps->end(); 1148 1149 ContentProvider * pProvider 1150 = static_cast< ContentProvider * >( rProvider.get() ); 1151 beans::Property aProp; 1152 1153 while ( it != end ) 1154 { 1155 if ( pProvider->getProperty( (*it).first, aProp ) ) 1156 xRow->appendObject( aProp, (*it).second.value() ); 1157 1158 ++it; 1159 } 1160 1161 // Append all local Additional Properties. 1162 uno::Reference< beans::XPropertySet > xSet( 1163 rProvider->getAdditionalPropertySet( rContentId, sal_False ), 1164 uno::UNO_QUERY ); 1165 xRow->appendPropertySet( xSet ); 1166 } 1167 1168 return uno::Reference< sdbc::XRow >( xRow.get() ); 1169 } 1170 1171 //========================================================================= 1172 uno::Reference< sdbc::XRow > Content::getPropertyValues( 1173 const uno::Sequence< beans::Property >& rProperties, 1174 const uno::Reference< ucb::XCommandEnvironment >& xEnv ) 1175 throw ( uno::Exception ) 1176 { 1177 std::auto_ptr< ContentProperties > xProps; 1178 std::auto_ptr< ContentProperties > xCachedProps; 1179 std::auto_ptr< DAVResourceAccess > xResAccess; 1180 rtl::OUString aUnescapedTitle; 1181 bool bHasAll = false; 1182 uno::Reference< lang::XMultiServiceFactory > xSMgr; 1183 uno::Reference< ucb::XContentIdentifier > xIdentifier; 1184 rtl::Reference< ::ucbhelper::ContentProviderImplHelper > xProvider; 1185 1186 { 1187 osl::Guard< osl::Mutex > aGuard( m_aMutex ); 1188 1189 aUnescapedTitle = SerfUri::unescape( m_aEscapedTitle ); 1190 xSMgr.set( m_xSMgr ); 1191 xIdentifier.set( m_xIdentifier ); 1192 xProvider.set( m_xProvider.get() ); 1193 xResAccess.reset( new DAVResourceAccess( *m_xResAccess.get() ) ); 1194 1195 // First, ask cache... 1196 if ( m_xCachedProps.get() ) 1197 { 1198 xCachedProps.reset( new ContentProperties( *m_xCachedProps.get() ) ); 1199 1200 std::vector< rtl::OUString > aMissingProps; 1201 if ( xCachedProps->containsAllNames( rProperties, aMissingProps ) ) 1202 { 1203 // All properties are already in cache! No server access needed. 1204 bHasAll = true; 1205 } 1206 1207 // use the cached ContentProperties instance 1208 xProps.reset( new ContentProperties( *xCachedProps.get() ) ); 1209 } 1210 } 1211 1212 if ( !m_bTransient && !bHasAll ) 1213 { 1214 ///////////////////////////////////////////////////////////////////// 1215 // Obtain values from server... 1216 ///////////////////////////////////////////////////////////////////// 1217 1218 // First, identify whether resource is DAV or not 1219 const ResourceType & rType = getResourceType( xEnv, xResAccess ); 1220 1221 bool bNetworkAccessAllowed = true; 1222 1223 if ( DAV == rType ) 1224 { 1225 // cache lookup... getResourceType may fill the props cache via 1226 // PROPFIND! 1227 if ( m_xCachedProps.get() ) 1228 { 1229 xCachedProps.reset( 1230 new ContentProperties( *m_xCachedProps.get() ) ); 1231 1232 std::vector< rtl::OUString > aMissingProps; 1233 if ( xCachedProps->containsAllNames( 1234 rProperties, aMissingProps ) ) 1235 { 1236 // All properties are already in cache! No server access 1237 // needed. 1238 bHasAll = true; 1239 } 1240 1241 // use the cached ContentProperties instance 1242 xProps.reset( new ContentProperties( *xCachedProps.get() ) ); 1243 } 1244 1245 if ( !bHasAll ) 1246 { 1247 // Only DAV resources support PROPFIND 1248 std::vector< rtl::OUString > aPropNames; 1249 1250 uno::Sequence< beans::Property > aProperties( 1251 rProperties.getLength() ); 1252 1253 if ( m_aFailedPropNames.size() > 0 ) 1254 { 1255 sal_Int32 nProps = 0; 1256 sal_Int32 nCount = rProperties.getLength(); 1257 for ( sal_Int32 n = 0; n < nCount; ++n ) 1258 { 1259 const rtl::OUString & rName = rProperties[ n ].Name; 1260 1261 std::vector< rtl::OUString >::const_iterator it 1262 = m_aFailedPropNames.begin(); 1263 std::vector< rtl::OUString >::const_iterator end 1264 = m_aFailedPropNames.end(); 1265 1266 while ( it != end ) 1267 { 1268 if ( *it == rName ) 1269 break; 1270 1271 ++it; 1272 } 1273 1274 if ( it == end ) 1275 { 1276 aProperties[ nProps ] = rProperties[ n ]; 1277 nProps++; 1278 } 1279 } 1280 1281 aProperties.realloc( nProps ); 1282 } 1283 else 1284 { 1285 aProperties = rProperties; 1286 } 1287 1288 if ( aProperties.getLength() > 0 ) 1289 ContentProperties::UCBNamesToDAVNames( 1290 aProperties, aPropNames ); 1291 1292 if ( aPropNames.size() > 0 ) 1293 { 1294 std::vector< DAVResource > resources; 1295 try 1296 { 1297 xResAccess->PROPFIND( 1298 DAVZERO, aPropNames, resources, xEnv ); 1299 1300 if ( 1 == resources.size() ) 1301 { 1302 if ( xProps.get()) 1303 xProps->addProperties( 1304 aPropNames, 1305 ContentProperties( resources[ 0 ] )); 1306 else 1307 xProps.reset( 1308 new ContentProperties( resources[ 0 ] ) ); 1309 } 1310 } 1311 catch ( DAVException const & e ) 1312 { 1313 bNetworkAccessAllowed 1314 = shouldAccessNetworkAfterException( e ); 1315 1316 if ( !bNetworkAccessAllowed ) 1317 { 1318 cancelCommandExecution( e, xEnv ); 1319 // unreachable 1320 } 1321 } 1322 } 1323 } 1324 } 1325 1326 if ( bNetworkAccessAllowed ) 1327 { 1328 // All properties obtained already? 1329 std::vector< rtl::OUString > aMissingProps; 1330 if ( !( xProps.get() 1331 && xProps->containsAllNames( 1332 rProperties, aMissingProps ) ) 1333 && !m_bDidGetOrHead ) 1334 { 1335 // Possibly the missing props can be obtained using a HEAD 1336 // request. 1337 1338 std::vector< rtl::OUString > aHeaderNames; 1339 ContentProperties::UCBNamesToHTTPNames( 1340 rProperties, 1341 aHeaderNames, 1342 true /* bIncludeUnmatched */ ); 1343 1344 if ( aHeaderNames.size() > 0 ) 1345 { 1346 try 1347 { 1348 DAVResource resource; 1349 xResAccess->HEAD( aHeaderNames, resource, xEnv ); 1350 m_bDidGetOrHead = true; 1351 1352 if ( xProps.get() ) 1353 xProps->addProperties( 1354 aMissingProps, 1355 ContentProperties( resource ) ); 1356 else 1357 xProps.reset ( new ContentProperties( resource ) ); 1358 1359 if ( m_eResourceType == NON_DAV ) 1360 xProps->addProperties( aMissingProps, 1361 ContentProperties( 1362 aUnescapedTitle, 1363 false ) ); 1364 } 1365 catch ( DAVException const & e ) 1366 { 1367 bNetworkAccessAllowed 1368 = shouldAccessNetworkAfterException( e ); 1369 1370 if ( !bNetworkAccessAllowed ) 1371 { 1372 cancelCommandExecution( e, xEnv ); 1373 // unreachable 1374 } 1375 } 1376 } 1377 } 1378 } 1379 1380 // might trigger HTTP redirect. 1381 // Therefore, title must be updated here. 1382 SerfUri aUri( xResAccess->getURL() ); 1383 aUnescapedTitle = aUri.GetPathBaseNameUnescaped(); 1384 1385 if ( rType == UNKNOWN ) 1386 { 1387 xProps.reset( new ContentProperties( aUnescapedTitle ) ); 1388 } 1389 1390 // For DAV resources we only know the Title, for non-DAV 1391 // resources we additionally know that it is a document. 1392 1393 if ( rType == DAV ) 1394 { 1395 //xProps.reset( 1396 // new ContentProperties( aUnescapedTitle ) ); 1397 xProps->addProperty( 1398 rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "Title" ) ), 1399 uno::makeAny( aUnescapedTitle ), 1400 true ); 1401 } 1402 else 1403 { 1404 if ( !xProps.get() ) 1405 xProps.reset( new ContentProperties( aUnescapedTitle, false ) ); 1406 else 1407 xProps->addProperty( 1408 rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "Title" ) ), 1409 uno::makeAny( aUnescapedTitle ), 1410 true ); 1411 1412 xProps->addProperty( 1413 rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "IsFolder" ) ), 1414 uno::makeAny( false ), 1415 true ); 1416 xProps->addProperty( 1417 rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "IsDocument" ) ), 1418 uno::makeAny( true ), 1419 true ); 1420 } 1421 } 1422 else 1423 { 1424 // No server access for just created (not yet committed) objects. 1425 // Only a minimal set of properties supported at this stage. 1426 if (m_bTransient) 1427 xProps.reset( new ContentProperties( aUnescapedTitle, 1428 m_bCollection ) ); 1429 } 1430 1431 sal_Int32 nCount = rProperties.getLength(); 1432 for ( sal_Int32 n = 0; n < nCount; ++n ) 1433 { 1434 const rtl::OUString rName = rProperties[ n ].Name; 1435 if ( rName.equalsAsciiL( 1436 RTL_CONSTASCII_STRINGPARAM( "BaseURI" ) ) ) 1437 { 1438 // Add BaseURI property, if requested. 1439 xProps->addProperty( 1440 rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "BaseURI" ) ), 1441 uno::makeAny( getBaseURI( xResAccess ) ), 1442 true ); 1443 } 1444 else if ( rName.equalsAsciiL( 1445 RTL_CONSTASCII_STRINGPARAM( "CreatableContentsInfo" ) ) ) 1446 { 1447 // Add CreatableContentsInfo property, if requested. 1448 sal_Bool bFolder = sal_False; 1449 xProps->getValue( 1450 rtl::OUString( RTL_CONSTASCII_USTRINGPARAM( "IsFolder" ) ) ) 1451 >>= bFolder; 1452 xProps->addProperty( 1453 rtl::OUString( 1454 RTL_CONSTASCII_USTRINGPARAM( "CreatableContentsInfo" ) ), 1455 uno::makeAny( bFolder 1456 ? queryCreatableContentsInfo() 1457 : uno::Sequence< ucb::ContentInfo >() ), 1458 true ); 1459 } 1460 } 1461 1462 uno::Reference< sdbc::XRow > xResultRow 1463 = getPropertyValues( xSMgr, 1464 rProperties, 1465 *xProps, 1466 xProvider, 1467 xIdentifier->getContentIdentifier() ); 1468 1469 { 1470 osl::Guard< osl::Mutex > aGuard( m_aMutex ); 1471 1472 if ( !m_xCachedProps.get() ) 1473 m_xCachedProps.reset( new CachableContentProperties( *xProps.get() ) ); 1474 else 1475 m_xCachedProps->addProperties( *xProps.get() ); 1476 1477 m_xResAccess.reset( new DAVResourceAccess( *xResAccess.get() ) ); 1478 m_aEscapedTitle = SerfUri::escapeSegment( aUnescapedTitle ); 1479 } 1480 1481 return xResultRow; 1482 } 1483 1484 //========================================================================= 1485 uno::Sequence< uno::Any > Content::setPropertyValues( 1486 const uno::Sequence< beans::PropertyValue >& rValues, 1487 const uno::Reference< ucb::XCommandEnvironment >& xEnv ) 1488 throw ( uno::Exception ) 1489 { 1490 uno::Reference< lang::XMultiServiceFactory > xSMgr; 1491 uno::Reference< ucb::XContentIdentifier > xIdentifier; 1492 rtl::Reference< ContentProvider > xProvider; 1493 sal_Bool bTransient; 1494 std::auto_ptr< DAVResourceAccess > xResAccess; 1495 1496 { 1497 osl::Guard< osl::Mutex > aGuard( m_aMutex ); 1498 1499 xProvider.set( m_pProvider ); 1500 xIdentifier.set( m_xIdentifier ); 1501 bTransient = m_bTransient; 1502 xResAccess.reset( new DAVResourceAccess( *m_xResAccess.get() ) ); 1503 xSMgr.set( m_xSMgr ); 1504 } 1505 1506 uno::Sequence< uno::Any > aRet( rValues.getLength() ); 1507 uno::Sequence< beans::PropertyChangeEvent > aChanges( rValues.getLength() ); 1508 sal_Int32 nChanged = 0; 1509 1510 beans::PropertyChangeEvent aEvent; 1511 aEvent.Source = static_cast< cppu::OWeakObject * >( this ); 1512 aEvent.Further = sal_False; 1513 // aEvent.PropertyName = 1514 aEvent.PropertyHandle = -1; 1515 // aEvent.OldValue = 1516 // aEvent.NewValue = 1517 1518 std::vector< ProppatchValue > aProppatchValues; 1519 std::vector< sal_Int32 > aProppatchPropsPositions; 1520 1521 uno::Reference< ucb::XPersistentPropertySet > xAdditionalPropSet; 1522 sal_Bool bTriedToGetAdditonalPropSet = sal_False; 1523 1524 sal_Bool bExchange = sal_False; 1525 rtl::OUString aNewTitle; 1526 rtl::OUString aOldTitle; 1527 sal_Int32 nTitlePos = -1; 1528 1529 uno::Reference< beans::XPropertySetInfo > xInfo; 1530 1531 const beans::PropertyValue* pValues = rValues.getConstArray(); 1532 sal_Int32 nCount = rValues.getLength(); 1533 for ( sal_Int32 n = 0; n < nCount; ++n ) 1534 { 1535 const beans::PropertyValue& rValue = pValues[ n ]; 1536 const rtl::OUString & rName = rValue.Name; 1537 1538 beans::Property aTmpProp; 1539 xProvider->getProperty( rName, aTmpProp ); 1540 1541 if ( aTmpProp.Attributes & beans::PropertyAttribute::READONLY ) 1542 { 1543 // Read-only property! 1544 aRet[ n ] <<= lang::IllegalAccessException( 1545 rtl::OUString::createFromAscii( 1546 "Property is read-only!" ), 1547 static_cast< cppu::OWeakObject * >( this ) ); 1548 continue; 1549 } 1550 1551 ////////////////////////////////////////////////////////////////// 1552 // Mandatory props. 1553 ////////////////////////////////////////////////////////////////// 1554 1555 if ( rName.equalsAsciiL( RTL_CONSTASCII_STRINGPARAM( "ContentType" ) ) ) 1556 { 1557 // Read-only property! 1558 aRet[ n ] <<= lang::IllegalAccessException( 1559 rtl::OUString::createFromAscii( 1560 "Property is read-only!" ), 1561 static_cast< cppu::OWeakObject * >( this ) ); 1562 } 1563 else if ( rName.equalsAsciiL( 1564 RTL_CONSTASCII_STRINGPARAM( "IsDocument" ) ) ) 1565 { 1566 // Read-only property! 1567 aRet[ n ] <<= lang::IllegalAccessException( 1568 rtl::OUString::createFromAscii( 1569 "Property is read-only!" ), 1570 static_cast< cppu::OWeakObject * >( this ) ); 1571 } 1572 else if ( rName.equalsAsciiL( 1573 RTL_CONSTASCII_STRINGPARAM( "IsFolder" ) ) ) 1574 { 1575 // Read-only property! 1576 aRet[ n ] <<= lang::IllegalAccessException( 1577 rtl::OUString::createFromAscii( 1578 "Property is read-only!" ), 1579 static_cast< cppu::OWeakObject * >( this ) ); 1580 } 1581 else if ( rName.equalsAsciiL( RTL_CONSTASCII_STRINGPARAM( "Title" ) ) ) 1582 { 1583 rtl::OUString aNewValue; 1584 if ( rValue.Value >>= aNewValue ) 1585 { 1586 // No empty titles! 1587 if ( aNewValue.getLength() > 0 ) 1588 { 1589 try 1590 { 1591 SerfUri aURI( xIdentifier->getContentIdentifier() ); 1592 aOldTitle = aURI.GetPathBaseNameUnescaped(); 1593 1594 if ( aNewValue != aOldTitle ) 1595 { 1596 // modified title -> modified URL -> exchange ! 1597 if ( !bTransient ) 1598 bExchange = sal_True; 1599 1600 // new value will be set later... 1601 aNewTitle = aNewValue; 1602 1603 // remember position within sequence of values (for 1604 // error handling). 1605 nTitlePos = n; 1606 } 1607 } 1608 catch ( DAVException const & ) 1609 { 1610 aRet[ n ] <<= lang::IllegalArgumentException( 1611 rtl::OUString::createFromAscii( 1612 "Invalid content identifier!" ), 1613 static_cast< cppu::OWeakObject * >( this ), 1614 -1 ); 1615 } 1616 } 1617 else 1618 { 1619 aRet[ n ] <<= lang::IllegalArgumentException( 1620 rtl::OUString::createFromAscii( 1621 "Empty title not allowed!" ), 1622 static_cast< cppu::OWeakObject * >( this ), 1623 -1 ); 1624 } 1625 } 1626 else 1627 { 1628 aRet[ n ] <<= beans::IllegalTypeException( 1629 rtl::OUString::createFromAscii( 1630 "Property value has wrong type!" ), 1631 static_cast< cppu::OWeakObject * >( this ) ); 1632 } 1633 } 1634 else 1635 { 1636 ////////////////////////////////////////////////////////////// 1637 // Optional props. 1638 ////////////////////////////////////////////////////////////// 1639 1640 if ( !xInfo.is() ) 1641 xInfo = getPropertySetInfo( xEnv, 1642 sal_False /* don't cache data */ ); 1643 1644 if ( !xInfo->hasPropertyByName( rName ) ) 1645 { 1646 // Check, whether property exists. Skip otherwise. 1647 // PROPPATCH::set would add the property automatically, which 1648 // is not allowed for "setPropertyValues" command! 1649 aRet[ n ] <<= beans::UnknownPropertyException( 1650 rtl::OUString::createFromAscii( 1651 "Property is unknown!" ), 1652 static_cast< cppu::OWeakObject * >( this ) ); 1653 continue; 1654 } 1655 1656 if ( rName.equalsAsciiL( RTL_CONSTASCII_STRINGPARAM( "Size" ) ) ) 1657 { 1658 // Read-only property! 1659 aRet[ n ] <<= lang::IllegalAccessException( 1660 rtl::OUString::createFromAscii( 1661 "Property is read-only!" ), 1662 static_cast< cppu::OWeakObject * >( this ) ); 1663 } 1664 else if ( rName.equalsAsciiL( 1665 RTL_CONSTASCII_STRINGPARAM( "DateCreated" ) ) ) 1666 { 1667 // Read-only property! 1668 aRet[ n ] <<= lang::IllegalAccessException( 1669 rtl::OUString::createFromAscii( 1670 "Property is read-only!" ), 1671 static_cast< cppu::OWeakObject * >( this ) ); 1672 } 1673 else if ( rName.equalsAsciiL( 1674 RTL_CONSTASCII_STRINGPARAM( "DateModified" ) ) ) 1675 { 1676 // Read-only property! 1677 aRet[ n ] <<= lang::IllegalAccessException( 1678 rtl::OUString::createFromAscii( 1679 "Property is read-only!" ), 1680 static_cast< cppu::OWeakObject * >( this ) ); 1681 } 1682 else if ( rName.equalsAsciiL( 1683 RTL_CONSTASCII_STRINGPARAM( "MediaType" ) ) ) 1684 { 1685 // Read-only property! 1686 // (but could be writable, if 'getcontenttype' would be) 1687 aRet[ n ] <<= lang::IllegalAccessException( 1688 rtl::OUString::createFromAscii( 1689 "Property is read-only!" ), 1690 static_cast< cppu::OWeakObject * >( this ) ); 1691 } 1692 if ( rName.equalsAsciiL( 1693 RTL_CONSTASCII_STRINGPARAM( "CreatableContentsInfo" ) ) ) 1694 { 1695 // Read-only property! 1696 aRet[ n ] <<= lang::IllegalAccessException( 1697 rtl::OUString::createFromAscii( 1698 "Property is read-only!" ), 1699 static_cast< cppu::OWeakObject * >( this ) ); 1700 } 1701 else 1702 { 1703 if ( getResourceType( xEnv, xResAccess ) == DAV ) 1704 { 1705 // Property value will be set on server. 1706 ProppatchValue aValue( PROPSET, rName, rValue.Value ); 1707 aProppatchValues.push_back( aValue ); 1708 1709 // remember position within sequence of values (for 1710 // error handling). 1711 aProppatchPropsPositions.push_back( n ); 1712 } 1713 else 1714 { 1715 // Property value will be stored in local property store. 1716 if ( !bTriedToGetAdditonalPropSet && 1717 !xAdditionalPropSet.is() ) 1718 { 1719 xAdditionalPropSet 1720 = getAdditionalPropertySet( sal_False ); 1721 bTriedToGetAdditonalPropSet = sal_True; 1722 } 1723 1724 if ( xAdditionalPropSet.is() ) 1725 { 1726 try 1727 { 1728 uno::Any aOldValue 1729 = xAdditionalPropSet->getPropertyValue( rName ); 1730 if ( aOldValue != rValue.Value ) 1731 { 1732 xAdditionalPropSet->setPropertyValue( 1733 rName, rValue.Value ); 1734 1735 aEvent.PropertyName = rName; 1736 aEvent.OldValue = aOldValue; 1737 aEvent.NewValue = rValue.Value; 1738 1739 aChanges.getArray()[ nChanged ] = aEvent; 1740 nChanged++; 1741 } 1742 } 1743 catch ( beans::UnknownPropertyException const & e ) 1744 { 1745 aRet[ n ] <<= e; 1746 } 1747 catch ( lang::WrappedTargetException const & e ) 1748 { 1749 aRet[ n ] <<= e; 1750 } 1751 catch ( beans::PropertyVetoException const & e ) 1752 { 1753 aRet[ n ] <<= e; 1754 } 1755 catch ( lang::IllegalArgumentException const & e ) 1756 { 1757 aRet[ n ] <<= e; 1758 } 1759 } 1760 else 1761 { 1762 aRet[ n ] <<= uno::Exception( 1763 rtl::OUString::createFromAscii( 1764 "No property set for storing the value!" ), 1765 static_cast< cppu::OWeakObject * >( this ) ); 1766 } 1767 } 1768 } 1769 } 1770 } // for 1771 1772 if ( !bTransient && aProppatchValues.size() ) 1773 { 1774 try 1775 { 1776 // Set property values at server. 1777 xResAccess->PROPPATCH( aProppatchValues, xEnv ); 1778 1779 std::vector< ProppatchValue >::const_iterator it 1780 = aProppatchValues.begin(); 1781 std::vector< ProppatchValue >::const_iterator end 1782 = aProppatchValues.end(); 1783 1784 while ( it != end ) 1785 { 1786 aEvent.PropertyName = (*it).name; 1787 aEvent.OldValue = uno::Any(); // @@@ to expensive to obtain! 1788 aEvent.NewValue = (*it).value; 1789 1790 aChanges.getArray()[ nChanged ] = aEvent; 1791 nChanged++; 1792 1793 ++it; 1794 } 1795 } 1796 catch ( DAVException const & e ) 1797 { 1798 // OSL_ENSURE( sal_False, 1799 // "Content::setPropertyValues - PROPPATCH failed!" ); 1800 1801 #if 1 1802 cancelCommandExecution( e, xEnv ); 1803 // unreachable 1804 #else 1805 // Note: PROPPATCH either sets ALL property values OR NOTHING. 1806 1807 std::vector< sal_Int32 >::const_iterator it 1808 = aProppatchPropsPositions.begin(); 1809 std::vector< sal_Int32 >::const_iterator end 1810 = aProppatchPropsPositions.end(); 1811 1812 while ( it != end ) 1813 { 1814 // Set error. 1815 aRet[ (*it) ] <<= MapDAVException( e, sal_True ); 1816 ++it; 1817 } 1818 #endif 1819 } 1820 } 1821 1822 if ( bExchange ) 1823 { 1824 // Assemble new content identifier... 1825 1826 rtl::OUString aNewURL = getParentURL(); 1827 if ( aNewURL.lastIndexOf( '/' ) != ( aNewURL.getLength() - 1 ) ) 1828 aNewURL += rtl::OUString::createFromAscii( "/" ); 1829 1830 aNewURL += SerfUri::escapeSegment( aNewTitle ); 1831 1832 uno::Reference< ucb::XContentIdentifier > xNewId 1833 = new ::ucbhelper::ContentIdentifier( xSMgr, aNewURL ); 1834 uno::Reference< ucb::XContentIdentifier > xOldId = xIdentifier; 1835 1836 try 1837 { 1838 SerfUri sourceURI( xOldId->getContentIdentifier() ); 1839 SerfUri targetURI( xNewId->getContentIdentifier() ); 1840 targetURI.SetScheme( sourceURI.GetScheme() ); 1841 1842 xResAccess->MOVE( 1843 sourceURI.GetPath(), targetURI.GetURI(), sal_False, xEnv ); 1844 // @@@ Should check for resources that could not be moved 1845 // (due to source access or target overwrite) and send 1846 // this information through the interaction handler. 1847 1848 // @@@ Existing content should be checked to see if it needs 1849 // to be deleted at the source 1850 1851 // @@@ Existing content should be checked to see if it has 1852 // been overwritten at the target 1853 1854 if ( exchangeIdentity( xNewId ) ) 1855 { 1856 xResAccess->setURL( aNewURL ); 1857 1858 // DAV resources store all additional props on server! 1859 // // Adapt Additional Core Properties. 1860 // renameAdditionalPropertySet( xOldId->getContentIdentifier(), 1861 // xNewId->getContentIdentifier(), 1862 // sal_True ); 1863 } 1864 else 1865 { 1866 // Do not set new title! 1867 aNewTitle = rtl::OUString(); 1868 1869 // Set error . 1870 aRet[ nTitlePos ] <<= uno::Exception( 1871 rtl::OUString::createFromAscii( "Exchange failed!" ), 1872 static_cast< cppu::OWeakObject * >( this ) ); 1873 } 1874 } 1875 catch ( DAVException const & e ) 1876 { 1877 // Do not set new title! 1878 aNewTitle = rtl::OUString(); 1879 1880 // Set error . 1881 aRet[ nTitlePos ] <<= MapDAVException( e, sal_True ); 1882 } 1883 } 1884 1885 if ( aNewTitle.getLength() ) 1886 { 1887 osl::Guard< osl::Mutex > aGuard( m_aMutex ); 1888 1889 aEvent.PropertyName = rtl::OUString::createFromAscii( "Title" ); 1890 aEvent.OldValue = uno::makeAny( aOldTitle ); 1891 aEvent.NewValue = uno::makeAny( aNewTitle ); 1892 1893 m_aEscapedTitle = SerfUri::escapeSegment( aNewTitle ); 1894 1895 aChanges.getArray()[ nChanged ] = aEvent; 1896 nChanged++; 1897 } 1898 1899 if ( nChanged > 0 ) 1900 { 1901 aChanges.realloc( nChanged ); 1902 notifyPropertiesChange( aChanges ); 1903 } 1904 1905 { 1906 osl::Guard< osl::Mutex > aGuard( m_aMutex ); 1907 m_xResAccess.reset( new DAVResourceAccess( *xResAccess.get() ) ); 1908 } 1909 1910 return aRet; 1911 } 1912 1913 //========================================================================= 1914 uno::Any Content::open( 1915 const ucb::OpenCommandArgument2 & rArg, 1916 const uno::Reference< ucb::XCommandEnvironment > & xEnv ) 1917 throw( uno::Exception ) 1918 { 1919 uno::Any aRet; 1920 1921 sal_Bool bOpenFolder = ( ( rArg.Mode == ucb::OpenMode::ALL ) || 1922 ( rArg.Mode == ucb::OpenMode::FOLDERS ) || 1923 ( rArg.Mode == ucb::OpenMode::DOCUMENTS ) ); 1924 if ( bOpenFolder ) 1925 { 1926 if ( isFolder( xEnv ) ) 1927 { 1928 // Open collection. 1929 1930 uno::Reference< ucb::XDynamicResultSet > xSet 1931 = new DynamicResultSet( m_xSMgr, this, rArg, xEnv ); 1932 aRet <<= xSet; 1933 } 1934 else 1935 { 1936 // Error: Not a folder! 1937 1938 rtl::OUStringBuffer aMsg; 1939 aMsg.appendAscii( "Non-folder resource cannot be " 1940 "opened as folder! Wrong Open Mode!" ); 1941 1942 ucbhelper::cancelCommandExecution( 1943 uno::makeAny( 1944 lang::IllegalArgumentException( 1945 aMsg.makeStringAndClear(), 1946 static_cast< cppu::OWeakObject * >( this ), 1947 -1 ) ), 1948 xEnv ); 1949 // Unreachable 1950 } 1951 } 1952 1953 if ( rArg.Sink.is() ) 1954 { 1955 // Open document. 1956 1957 if ( ( rArg.Mode == ucb::OpenMode::DOCUMENT_SHARE_DENY_NONE ) || 1958 ( rArg.Mode == ucb::OpenMode::DOCUMENT_SHARE_DENY_WRITE ) ) 1959 { 1960 // Currently(?) unsupported. 1961 ucbhelper::cancelCommandExecution( 1962 uno::makeAny( 1963 ucb::UnsupportedOpenModeException( 1964 rtl::OUString(), 1965 static_cast< cppu::OWeakObject * >( this ), 1966 sal_Int16( rArg.Mode ) ) ), 1967 xEnv ); 1968 // Unreachable 1969 } 1970 1971 rtl::OUString aURL = m_xIdentifier->getContentIdentifier(); 1972 uno::Reference< io::XOutputStream > xOut 1973 = uno::Reference< io::XOutputStream >( rArg.Sink, uno::UNO_QUERY ); 1974 if ( xOut.is() ) 1975 { 1976 // PUSH: write data 1977 try 1978 { 1979 std::auto_ptr< DAVResourceAccess > xResAccess; 1980 1981 { 1982 osl::MutexGuard aGuard( m_aMutex ); 1983 1984 xResAccess.reset( 1985 new DAVResourceAccess( *m_xResAccess.get() ) ); 1986 } 1987 1988 DAVResource aResource; 1989 std::vector< rtl::OUString > aHeaders; 1990 1991 xResAccess->GET( xOut, aHeaders, aResource, xEnv ); 1992 m_bDidGetOrHead = true; 1993 1994 { 1995 osl::MutexGuard aGuard( m_aMutex ); 1996 1997 // cache headers. 1998 if ( !m_xCachedProps.get()) 1999 m_xCachedProps.reset( 2000 new CachableContentProperties( aResource ) ); 2001 else 2002 m_xCachedProps->addProperties( aResource ); 2003 2004 m_xResAccess.reset( 2005 new DAVResourceAccess( *xResAccess.get() ) ); 2006 } 2007 } 2008 catch ( DAVException const & e ) 2009 { 2010 cancelCommandExecution( e, xEnv ); 2011 // Unreachable 2012 } 2013 } 2014 else 2015 { 2016 uno::Reference< io::XActiveDataSink > xDataSink 2017 = uno::Reference< io::XActiveDataSink >( rArg.Sink, 2018 uno::UNO_QUERY ); 2019 if ( xDataSink.is() ) 2020 { 2021 // PULL: wait for client read 2022 try 2023 { 2024 std::auto_ptr< DAVResourceAccess > xResAccess; 2025 { 2026 osl::MutexGuard aGuard( m_aMutex ); 2027 2028 xResAccess.reset( 2029 new DAVResourceAccess( *m_xResAccess.get() ) ); 2030 } 2031 2032 // fill inputsream sync; return if all data present 2033 DAVResource aResource; 2034 std::vector< rtl::OUString > aHeaders; 2035 2036 uno::Reference< io::XInputStream > xIn 2037 = xResAccess->GET( aHeaders, aResource, xEnv ); 2038 m_bDidGetOrHead = true; 2039 2040 { 2041 osl::MutexGuard aGuard( m_aMutex ); 2042 2043 // cache headers. 2044 if ( !m_xCachedProps.get()) 2045 m_xCachedProps.reset( 2046 new CachableContentProperties( aResource ) ); 2047 else 2048 m_xCachedProps->addProperties( 2049 aResource.properties ); 2050 2051 m_xResAccess.reset( 2052 new DAVResourceAccess( *xResAccess.get() ) ); 2053 } 2054 2055 xDataSink->setInputStream( xIn ); 2056 } 2057 catch ( DAVException const & e ) 2058 { 2059 cancelCommandExecution( e, xEnv ); 2060 // Unreachable 2061 } 2062 } 2063 else 2064 { 2065 // Note: aOpenCommand.Sink may contain an XStream 2066 // implementation. Support for this type of 2067 // sink is optional... 2068 ucbhelper::cancelCommandExecution( 2069 uno::makeAny( 2070 ucb::UnsupportedDataSinkException( 2071 rtl::OUString(), 2072 static_cast< cppu::OWeakObject * >( this ), 2073 rArg.Sink ) ), 2074 xEnv ); 2075 // Unreachable 2076 } 2077 } 2078 } 2079 2080 return aRet; 2081 } 2082 2083 //========================================================================= 2084 void Content::post( 2085 const ucb::PostCommandArgument2 & rArg, 2086 const uno::Reference< ucb::XCommandEnvironment > & xEnv ) 2087 throw( uno::Exception ) 2088 { 2089 uno::Reference< io::XActiveDataSink > xSink( rArg.Sink, uno::UNO_QUERY ); 2090 if ( xSink.is() ) 2091 { 2092 try 2093 { 2094 std::auto_ptr< DAVResourceAccess > xResAccess; 2095 { 2096 osl::MutexGuard aGuard( m_aMutex ); 2097 xResAccess.reset( 2098 new DAVResourceAccess( *m_xResAccess.get() ) ); 2099 } 2100 2101 uno::Reference< io::XInputStream > xResult 2102 = xResAccess->POST( rArg.MediaType, 2103 rArg.Referer, 2104 rArg.Source, 2105 xEnv ); 2106 2107 { 2108 osl::MutexGuard aGuard( m_aMutex ); 2109 m_xResAccess.reset( 2110 new DAVResourceAccess( *xResAccess.get() ) ); 2111 } 2112 2113 xSink->setInputStream( xResult ); 2114 } 2115 catch ( DAVException const & e ) 2116 { 2117 cancelCommandExecution( e, xEnv, sal_True ); 2118 // Unreachable 2119 } 2120 } 2121 else 2122 { 2123 uno::Reference< io::XOutputStream > xResult( rArg.Sink, uno::UNO_QUERY ); 2124 if ( xResult.is() ) 2125 { 2126 try 2127 { 2128 std::auto_ptr< DAVResourceAccess > xResAccess; 2129 { 2130 osl::MutexGuard aGuard( m_aMutex ); 2131 xResAccess.reset( 2132 new DAVResourceAccess( *m_xResAccess.get() ) ); 2133 } 2134 2135 xResAccess->POST( rArg.MediaType, 2136 rArg.Referer, 2137 rArg.Source, 2138 xResult, 2139 xEnv ); 2140 2141 { 2142 osl::MutexGuard aGuard( m_aMutex ); 2143 m_xResAccess.reset( 2144 new DAVResourceAccess( *xResAccess.get() ) ); 2145 } 2146 } 2147 catch ( DAVException const & e ) 2148 { 2149 cancelCommandExecution( e, xEnv, sal_True ); 2150 // Unreachable 2151 } 2152 } 2153 else 2154 { 2155 ucbhelper::cancelCommandExecution( 2156 uno::makeAny( 2157 ucb::UnsupportedDataSinkException( 2158 rtl::OUString(), 2159 static_cast< cppu::OWeakObject * >( this ), 2160 rArg.Sink ) ), 2161 xEnv ); 2162 // Unreachable 2163 } 2164 } 2165 } 2166 2167 //========================================================================= 2168 void Content::queryChildren( ContentRefList& rChildren ) 2169 { 2170 // Obtain a list with a snapshot of all currently instanciated contents 2171 // from provider and extract the contents which are direct children 2172 // of this content. 2173 2174 ::ucbhelper::ContentRefList aAllContents; 2175 m_xProvider->queryExistingContents( aAllContents ); 2176 2177 rtl::OUString aURL = m_xIdentifier->getContentIdentifier(); 2178 sal_Int32 nURLPos = aURL.lastIndexOf( '/' ); 2179 2180 if ( nURLPos != ( aURL.getLength() - 1 ) ) 2181 { 2182 // No trailing slash found. Append. 2183 aURL += rtl::OUString::createFromAscii( "/" ); 2184 } 2185 2186 sal_Int32 nLen = aURL.getLength(); 2187 2188 ::ucbhelper::ContentRefList::const_iterator it = aAllContents.begin(); 2189 ::ucbhelper::ContentRefList::const_iterator end = aAllContents.end(); 2190 2191 while ( it != end ) 2192 { 2193 ::ucbhelper::ContentImplHelperRef xChild = (*it); 2194 rtl::OUString aChildURL 2195 = xChild->getIdentifier()->getContentIdentifier(); 2196 2197 // Is aURL a prefix of aChildURL? 2198 if ( ( aChildURL.getLength() > nLen ) && 2199 ( aChildURL.compareTo( aURL, nLen ) == 0 ) ) 2200 { 2201 sal_Int32 nPos = nLen; 2202 nPos = aChildURL.indexOf( '/', nPos ); 2203 2204 if ( ( nPos == -1 ) || 2205 ( nPos == ( aChildURL.getLength() - 1 ) ) ) 2206 { 2207 // No further slashes / only a final slash. It's a child! 2208 rChildren.push_back( 2209 ::http_dav_ucp::Content::ContentRef( 2210 static_cast< ::http_dav_ucp::Content * >( 2211 xChild.get() ) ) ); 2212 } 2213 } 2214 ++it; 2215 } 2216 } 2217 2218 //========================================================================= 2219 void Content::insert( 2220 const uno::Reference< io::XInputStream > & xInputStream, 2221 sal_Bool bReplaceExisting, 2222 const uno::Reference< ucb::XCommandEnvironment >& Environment ) 2223 throw( uno::Exception ) 2224 { 2225 sal_Bool bTransient, bCollection; 2226 rtl::OUString aEscapedTitle; 2227 std::auto_ptr< DAVResourceAccess > xResAccess; 2228 2229 { 2230 osl::Guard< osl::Mutex > aGuard( m_aMutex ); 2231 2232 bTransient = m_bTransient; 2233 bCollection = m_bCollection; 2234 aEscapedTitle = m_aEscapedTitle; 2235 xResAccess.reset( new DAVResourceAccess( *m_xResAccess.get() ) ); 2236 } 2237 2238 // Check, if all required properties are present. 2239 2240 if ( aEscapedTitle.getLength() == 0 ) 2241 { 2242 OSL_ENSURE( sal_False, "Content::insert - Title missing!" ); 2243 2244 uno::Sequence< rtl::OUString > aProps( 1 ); 2245 aProps[ 0 ] = rtl::OUString::createFromAscii( "Title" ); 2246 ucbhelper::cancelCommandExecution( 2247 uno::makeAny( ucb::MissingPropertiesException( 2248 rtl::OUString(), 2249 static_cast< cppu::OWeakObject * >( this ), 2250 aProps ) ), 2251 Environment ); 2252 // Unreachable 2253 } 2254 2255 if ( !bReplaceExisting ) 2256 { 2257 /* [RFC 2616] - HTTP 2258 2259 The PUT method requests that the enclosed entity be stored under the 2260 supplied Request-URI. If the Request-URI refers to an already 2261 existing resource, the enclosed entity SHOULD be considered as a 2262 modified version of the one residing on the origin server. 2263 */ 2264 2265 /* [RFC 2518] - WebDAV 2266 2267 MKCOL creates a new collection resource at the location specified by 2268 the Request-URI. If the resource identified by the Request-URI is 2269 non-null then the MKCOL MUST fail. 2270 */ 2271 2272 // ==> Complain on PUT, continue on MKCOL. 2273 if ( !bTransient || ( bTransient && !bCollection ) ) 2274 { 2275 #undef ERROR 2276 ucb::UnsupportedNameClashException aEx( 2277 rtl::OUString::createFromAscii( "Unable to write without overwrite!" ), 2278 static_cast< cppu::OWeakObject * >( this ), 2279 ucb::NameClash::ERROR ); 2280 2281 uno::Reference< task::XInteractionHandler > xIH; 2282 2283 if ( Environment.is() ) 2284 xIH = Environment->getInteractionHandler(); 2285 2286 if ( xIH.is() ) 2287 { 2288 uno::Any aExAsAny( uno::makeAny( aEx ) ); 2289 2290 rtl::Reference< ucbhelper::SimpleInteractionRequest > xRequest 2291 = new ucbhelper::SimpleInteractionRequest( 2292 aExAsAny, 2293 ucbhelper::CONTINUATION_APPROVE 2294 | ucbhelper::CONTINUATION_DISAPPROVE ); 2295 xIH->handle( xRequest.get() ); 2296 2297 const sal_Int32 nResp = xRequest->getResponse(); 2298 2299 switch ( nResp ) 2300 { 2301 case ucbhelper::CONTINUATION_UNKNOWN: 2302 // Not handled; throw. 2303 throw aEx; 2304 // break; 2305 2306 case ucbhelper::CONTINUATION_APPROVE: 2307 // Continue -> Overwrite. 2308 bReplaceExisting = sal_True; 2309 break; 2310 2311 case ucbhelper::CONTINUATION_DISAPPROVE: 2312 // Abort. 2313 throw ucb::CommandFailedException( 2314 rtl::OUString(), 2315 uno::Reference< uno::XInterface >(), 2316 aExAsAny ); 2317 // break; 2318 2319 default: 2320 OSL_ENSURE( sal_False, 2321 "Content::insert - " 2322 "Unknown interaction selection!" ); 2323 throw ucb::CommandFailedException( 2324 rtl::OUString::createFromAscii( 2325 "Unknown interaction selection!" ), 2326 uno::Reference< uno::XInterface >(), 2327 aExAsAny ); 2328 // break; 2329 } 2330 } 2331 else 2332 { 2333 // No IH; throw. 2334 throw aEx; 2335 } 2336 } 2337 } 2338 2339 if ( bTransient ) 2340 { 2341 // Assemble new content identifier... 2342 rtl::OUString aURL = getParentURL(); 2343 if ( aURL.lastIndexOf( '/' ) != ( aURL.getLength() - 1 ) ) 2344 aURL += rtl::OUString::createFromAscii( "/" ); 2345 2346 aURL += aEscapedTitle; 2347 2348 try 2349 { 2350 xResAccess->setURL( aURL ); 2351 2352 if ( bCollection ) 2353 xResAccess->MKCOL( Environment ); 2354 else 2355 xResAccess->PUT( xInputStream, Environment ); 2356 } 2357 catch ( DAVException const & except ) 2358 { 2359 if ( bCollection ) 2360 { 2361 if ( except.getStatus() == SC_METHOD_NOT_ALLOWED ) 2362 { 2363 // [RFC 2518] - WebDAV 2364 // 405 (Method Not Allowed) - MKCOL can only be 2365 // executed on a deleted/non-existent resource. 2366 2367 if ( bReplaceExisting ) 2368 { 2369 // Destroy old resource. 2370 try 2371 { 2372 xResAccess->DESTROY( Environment ); 2373 } 2374 catch ( DAVException const & e ) 2375 { 2376 cancelCommandExecution( e, Environment, sal_True ); 2377 // Unreachable 2378 } 2379 2380 // Insert (recursion!). 2381 insert( xInputStream, bReplaceExisting, Environment ); 2382 2383 { 2384 osl::Guard< osl::Mutex > aGuard( m_aMutex ); 2385 m_xResAccess.reset( 2386 new DAVResourceAccess( *xResAccess.get() ) ); 2387 } 2388 2389 // Success! 2390 return; 2391 } 2392 else 2393 { 2394 rtl::OUString aTitle; 2395 try 2396 { 2397 SerfUri aURI( aURL ); 2398 aTitle = aURI.GetPathBaseNameUnescaped(); 2399 } 2400 catch ( DAVException const & ) 2401 { 2402 } 2403 2404 ucbhelper::cancelCommandExecution( 2405 uno::makeAny( 2406 ucb::NameClashException( 2407 rtl::OUString(), 2408 static_cast< cppu::OWeakObject * >( this ), 2409 task::InteractionClassification_ERROR, 2410 aTitle ) ), 2411 Environment ); 2412 // Unreachable 2413 } 2414 } 2415 } 2416 2417 cancelCommandExecution( except, Environment, sal_True ); 2418 // Unreachable 2419 } 2420 2421 { 2422 osl::Guard< osl::Mutex > aGuard( m_aMutex ); 2423 m_xIdentifier 2424 = new ::ucbhelper::ContentIdentifier( m_xSMgr, aURL ); 2425 } 2426 2427 inserted(); 2428 2429 { 2430 osl::Guard< osl::Mutex > aGuard( m_aMutex ); 2431 m_bTransient = sal_False; 2432 } 2433 } 2434 else 2435 { 2436 if ( !xInputStream.is() ) 2437 { 2438 ucbhelper::cancelCommandExecution( 2439 uno::makeAny( 2440 ucb::MissingInputStreamException( 2441 rtl::OUString(), 2442 static_cast< cppu::OWeakObject * >( this ) ) ), 2443 Environment ); 2444 // Unreachable 2445 } 2446 2447 try 2448 { 2449 xResAccess->PUT( xInputStream, Environment ); 2450 } 2451 catch ( DAVException const & e ) 2452 { 2453 cancelCommandExecution( e, Environment, sal_True ); 2454 // Unreachable 2455 } 2456 } 2457 2458 { 2459 osl::Guard< osl::Mutex > aGuard( m_aMutex ); 2460 m_xResAccess.reset( new DAVResourceAccess( *xResAccess.get() ) ); 2461 } 2462 } 2463 2464 //========================================================================= 2465 void Content::transfer( 2466 const ucb::TransferInfo & rArgs, 2467 const uno::Reference< ucb::XCommandEnvironment >& Environment ) 2468 throw( uno::Exception ) 2469 { 2470 uno::Reference< lang::XMultiServiceFactory > xSMgr; 2471 uno::Reference< ucb::XContentIdentifier > xIdentifier; 2472 uno::Reference< ucb::XContentProvider > xProvider; 2473 std::auto_ptr< DAVResourceAccess > xResAccess; 2474 2475 { 2476 osl::Guard< osl::Mutex > aGuard( m_aMutex ); 2477 2478 xSMgr.set( m_xSMgr ); 2479 xIdentifier.set( m_xIdentifier ); 2480 xProvider.set( m_xProvider.get() ); 2481 xResAccess.reset( new DAVResourceAccess( *m_xResAccess.get() ) ); 2482 } 2483 2484 rtl::OUString aTargetURI; 2485 try 2486 { 2487 SerfUri sourceURI( rArgs.SourceURL ); 2488 SerfUri targetURI( xIdentifier->getContentIdentifier() ); 2489 aTargetURI = targetURI.GetPathBaseNameUnescaped(); 2490 2491 // Check source's and target's URL scheme 2492 // 2493 const rtl::OUString aScheme = sourceURI.GetScheme().toAsciiLowerCase(); 2494 if ( aScheme.equalsAsciiL( 2495 RTL_CONSTASCII_STRINGPARAM( WEBDAV_URL_SCHEME ) ) ) 2496 { 2497 sourceURI.SetScheme( 2498 rtl::OUString::createFromAscii( HTTP_URL_SCHEME ) ); 2499 } 2500 else if ( aScheme.equalsAsciiL( 2501 RTL_CONSTASCII_STRINGPARAM( DAV_URL_SCHEME ) ) ) 2502 { 2503 sourceURI.SetScheme( 2504 rtl::OUString::createFromAscii( HTTP_URL_SCHEME ) ); 2505 } 2506 else if ( aScheme.equalsAsciiL( 2507 RTL_CONSTASCII_STRINGPARAM( DAVS_URL_SCHEME ) ) ) 2508 { 2509 sourceURI.SetScheme( 2510 rtl::OUString::createFromAscii( HTTPS_URL_SCHEME ) ); 2511 } 2512 else 2513 { 2514 if ( !aScheme.equalsAsciiL( 2515 RTL_CONSTASCII_STRINGPARAM( HTTP_URL_SCHEME ) ) && 2516 !aScheme.equalsAsciiL( 2517 RTL_CONSTASCII_STRINGPARAM( HTTPS_URL_SCHEME ) ) ) 2518 { 2519 ucbhelper::cancelCommandExecution( 2520 uno::makeAny( 2521 ucb::InteractiveBadTransferURLException( 2522 rtl::OUString::createFromAscii( 2523 "Unsupported URL scheme!" ), 2524 static_cast< cppu::OWeakObject * >( this ) ) ), 2525 Environment ); 2526 // Unreachable 2527 } 2528 } 2529 2530 if ( targetURI.GetScheme().toAsciiLowerCase().equalsAsciiL( 2531 RTL_CONSTASCII_STRINGPARAM( WEBDAV_URL_SCHEME ) ) ) 2532 targetURI.SetScheme( 2533 rtl::OUString::createFromAscii( HTTP_URL_SCHEME ) ); 2534 else if ( targetURI.GetScheme().toAsciiLowerCase().equalsAsciiL( 2535 RTL_CONSTASCII_STRINGPARAM( DAV_URL_SCHEME ) ) ) 2536 targetURI.SetScheme( 2537 rtl::OUString::createFromAscii( HTTP_URL_SCHEME ) ); 2538 2539 // @@@ This implementation of 'transfer' only works 2540 // if the source and target are located at same host. 2541 // (Neon does not support cross-server copy/move) 2542 2543 // Check for same host 2544 // 2545 if ( sourceURI.GetHost().getLength() && 2546 ( sourceURI.GetHost() != targetURI.GetHost() ) ) 2547 { 2548 ucbhelper::cancelCommandExecution( 2549 uno::makeAny( ucb::InteractiveBadTransferURLException( 2550 rtl::OUString::createFromAscii( 2551 "Different hosts!" ), 2552 static_cast< cppu::OWeakObject * >( this ) ) ), 2553 Environment ); 2554 // Unreachable 2555 } 2556 2557 rtl::OUString aTitle = rArgs.NewTitle; 2558 2559 if ( !aTitle.getLength() ) 2560 aTitle = sourceURI.GetPathBaseNameUnescaped(); 2561 2562 if ( aTitle.equalsAsciiL( RTL_CONSTASCII_STRINGPARAM( "/" ) ) ) 2563 { 2564 // kso: ??? 2565 aTitle = rtl::OUString(); 2566 } 2567 2568 targetURI.AppendPath( aTitle ); 2569 2570 rtl::OUString aTargetURL = xIdentifier->getContentIdentifier(); 2571 if ( ( aTargetURL.lastIndexOf( '/' ) + 1 ) 2572 != aTargetURL.getLength() ) 2573 aTargetURL += rtl::OUString::createFromAscii( "/" ); 2574 2575 aTargetURL += aTitle; 2576 2577 uno::Reference< ucb::XContentIdentifier > xTargetId 2578 = new ::ucbhelper::ContentIdentifier( xSMgr, aTargetURL ); 2579 2580 DAVResourceAccess aSourceAccess( xSMgr, 2581 xResAccess->getSessionFactory(), 2582 sourceURI.GetURI() ); 2583 2584 if ( rArgs.MoveData == sal_True ) 2585 { 2586 uno::Reference< ucb::XContentIdentifier > xId 2587 = new ::ucbhelper::ContentIdentifier( xSMgr, rArgs.SourceURL ); 2588 2589 // Note: The static cast is okay here, because its sure that 2590 // xProvider is always the WebDAVContentProvider. 2591 rtl::Reference< Content > xSource 2592 = static_cast< Content * >( 2593 xProvider->queryContent( xId ).get() ); 2594 2595 // [RFC 2518] - WebDAV 2596 // If a resource exists at the destination and the Overwrite 2597 // header is "T" then prior to performing the move the server 2598 // MUST perform a DELETE with "Depth: infinity" on the 2599 // destination resource. If the Overwrite header is set to 2600 // "F" then the operation will fail. 2601 2602 aSourceAccess.MOVE( sourceURI.GetPath(), 2603 targetURI.GetURI(), 2604 rArgs.NameClash 2605 == ucb::NameClash::OVERWRITE, 2606 Environment ); 2607 2608 if ( xSource.is() ) 2609 { 2610 // Propagate destruction to listeners. 2611 xSource->destroy( sal_True ); 2612 } 2613 2614 // DAV resources store all additional props on server! 2615 // // Rename own and all children's Additional Core Properties. 2616 // renameAdditionalPropertySet( xId->getContentIdentifier(), 2617 // xTargetId->getContentIdentifier(), 2618 // sal_True ); 2619 } 2620 else 2621 { 2622 // [RFC 2518] - WebDAV 2623 // If a resource exists at the destination and the Overwrite 2624 // header is "T" then prior to performing the copy the server 2625 // MUST perform a DELETE with "Depth: infinity" on the 2626 // destination resource. If the Overwrite header is set to 2627 // "F" then the operation will fail. 2628 2629 aSourceAccess.COPY( sourceURI.GetPath(), 2630 targetURI.GetURI(), 2631 rArgs.NameClash 2632 == ucb::NameClash::OVERWRITE, 2633 Environment ); 2634 2635 // DAV resources store all additional props on server! 2636 // // Copy own and all children's Additional Core Properties. 2637 // copyAdditionalPropertySet( xId->getContentIdentifier(), 2638 // xTargetId->getContentIdentifier(), 2639 // sal_True ); 2640 } 2641 2642 // Note: The static cast is okay here, because its sure that 2643 // xProvider is always the WebDAVContentProvider. 2644 rtl::Reference< Content > xTarget 2645 = static_cast< Content * >( 2646 xProvider->queryContent( xTargetId ).get() ); 2647 2648 // Announce transfered content in its new folder. 2649 xTarget->inserted(); 2650 } 2651 catch ( ucb::IllegalIdentifierException const & ) 2652 { 2653 // queryContent 2654 } 2655 catch ( DAVException const & e ) 2656 { 2657 // [RFC 2518] - WebDAV 2658 // 412 (Precondition Failed) - The server was unable to maintain 2659 // the liveness of the properties listed in the propertybehavior 2660 // XML element or the Overwrite header is "F" and the state of 2661 // the destination resource is non-null. 2662 2663 if ( e.getStatus() == SC_PRECONDITION_FAILED ) 2664 { 2665 switch ( rArgs.NameClash ) 2666 { 2667 case 0/*ucb::NameClash::ERROR*/: 2668 { 2669 ucbhelper::cancelCommandExecution( 2670 uno::makeAny( 2671 ucb::NameClashException( 2672 rtl::OUString(), 2673 static_cast< cppu::OWeakObject * >( this ), 2674 task::InteractionClassification_ERROR, 2675 aTargetURI ) ), 2676 Environment ); 2677 // Unreachable 2678 } 2679 2680 case ucb::NameClash::OVERWRITE: 2681 break; 2682 2683 case ucb::NameClash::KEEP: // deprecated 2684 case ucb::NameClash::RENAME: 2685 case ucb::NameClash::ASK: 2686 default: 2687 { 2688 ucbhelper::cancelCommandExecution( 2689 uno::makeAny( 2690 ucb::UnsupportedNameClashException( 2691 rtl::OUString(), 2692 static_cast< cppu::OWeakObject * >( this ), 2693 rArgs.NameClash ) ), 2694 Environment ); 2695 // Unreachable 2696 } 2697 } 2698 } 2699 2700 cancelCommandExecution( e, Environment, sal_True ); 2701 // Unreachable 2702 } 2703 2704 { 2705 osl::Guard< osl::Mutex > aGuard( m_aMutex ); 2706 m_xResAccess.reset( new DAVResourceAccess( *xResAccess.get() ) ); 2707 } 2708 } 2709 2710 //========================================================================= 2711 void Content::destroy( sal_Bool bDeletePhysical ) 2712 throw( uno::Exception ) 2713 { 2714 // @@@ take care about bDeletePhysical -> trashcan support 2715 rtl::OUString aURL = m_xIdentifier->getContentIdentifier(); 2716 2717 uno::Reference< ucb::XContent > xThis = this; 2718 2719 deleted(); 2720 2721 osl::Guard< osl::Mutex > aGuard( m_aMutex ); 2722 2723 // Process instanciated children... 2724 2725 ::http_dav_ucp::Content::ContentRefList aChildren; 2726 queryChildren( aChildren ); 2727 2728 ContentRefList::const_iterator it = aChildren.begin(); 2729 ContentRefList::const_iterator end = aChildren.end(); 2730 2731 while ( it != end ) 2732 { 2733 (*it)->destroy( bDeletePhysical ); 2734 ++it; 2735 } 2736 } 2737 2738 //========================================================================= 2739 bool Content::supportsExclusiveWriteLock( 2740 const uno::Reference< ucb::XCommandEnvironment >& Environment ) 2741 { 2742 if ( getResourceType( Environment ) == DAV ) 2743 { 2744 if ( m_xCachedProps.get() ) 2745 { 2746 uno::Sequence< ucb::LockEntry > aSupportedLocks; 2747 if ( m_xCachedProps->getValue( DAVProperties::SUPPORTEDLOCK ) 2748 >>= aSupportedLocks ) 2749 { 2750 for ( sal_Int32 n = 0; n < aSupportedLocks.getLength(); ++n ) 2751 { 2752 if ( aSupportedLocks[ n ].Scope 2753 == ucb::LockScope_EXCLUSIVE && 2754 aSupportedLocks[ n ].Type 2755 == ucb::LockType_WRITE ) 2756 return true; 2757 } 2758 } 2759 } 2760 } 2761 return false; 2762 } 2763 2764 //========================================================================= 2765 void Content::lock( 2766 const uno::Reference< ucb::XCommandEnvironment >& Environment ) 2767 throw( uno::Exception ) 2768 { 2769 try 2770 { 2771 std::auto_ptr< DAVResourceAccess > xResAccess; 2772 { 2773 osl::Guard< osl::Mutex > aGuard( m_aMutex ); 2774 xResAccess.reset( new DAVResourceAccess( *m_xResAccess.get() ) ); 2775 } 2776 2777 uno::Any aOwnerAny; 2778 aOwnerAny 2779 <<= rtl::OUString::createFromAscii( "http://ucb.openoffice.org" ); 2780 2781 ucb::Lock aLock( 2782 ucb::LockScope_EXCLUSIVE, 2783 ucb::LockType_WRITE, 2784 ucb::LockDepth_ZERO, 2785 aOwnerAny, 2786 180, // lock timeout in secs 2787 //-1, // infinite lock 2788 uno::Sequence< ::rtl::OUString >() ); 2789 2790 xResAccess->LOCK( aLock, Environment ); 2791 2792 { 2793 osl::Guard< osl::Mutex > aGuard( m_aMutex ); 2794 m_xResAccess.reset( new DAVResourceAccess( *xResAccess.get() ) ); 2795 } 2796 } 2797 catch ( DAVException const & e ) 2798 { 2799 cancelCommandExecution( e, Environment, sal_False ); 2800 // Unreachable 2801 } 2802 } 2803 2804 //========================================================================= 2805 void Content::unlock( 2806 const uno::Reference< ucb::XCommandEnvironment >& Environment ) 2807 throw( uno::Exception ) 2808 { 2809 try 2810 { 2811 std::auto_ptr< DAVResourceAccess > xResAccess; 2812 { 2813 osl::Guard< osl::Mutex > aGuard( m_aMutex ); 2814 xResAccess.reset( new DAVResourceAccess( *m_xResAccess.get() ) ); 2815 } 2816 2817 xResAccess->UNLOCK( Environment ); 2818 2819 { 2820 osl::Guard< osl::Mutex > aGuard( m_aMutex ); 2821 m_xResAccess.reset( new DAVResourceAccess( *xResAccess.get() ) ); 2822 } 2823 } 2824 catch ( DAVException const & e ) 2825 { 2826 cancelCommandExecution( e, Environment, sal_False ); 2827 // Unreachable 2828 } 2829 } 2830 2831 //========================================================================= 2832 sal_Bool Content::exchangeIdentity( 2833 const uno::Reference< ucb::XContentIdentifier >& xNewId ) 2834 { 2835 if ( !xNewId.is() ) 2836 return sal_False; 2837 2838 osl::ClearableGuard< osl::Mutex > aGuard( m_aMutex ); 2839 2840 uno::Reference< ucb::XContent > xThis = this; 2841 2842 // Already persistent? 2843 if ( m_bTransient ) 2844 { 2845 OSL_ENSURE( sal_False, "Content::exchangeIdentity - Not persistent!" ); 2846 return sal_False; 2847 } 2848 2849 // Exchange own identitity. 2850 2851 // Fail, if a content with given id already exists. 2852 // if ( !hasData( xNewId ) ) 2853 { 2854 rtl::OUString aOldURL = m_xIdentifier->getContentIdentifier(); 2855 2856 aGuard.clear(); 2857 if ( exchange( xNewId ) ) 2858 { 2859 // Process instanciated children... 2860 2861 ContentRefList aChildren; 2862 queryChildren( aChildren ); 2863 2864 ContentRefList::const_iterator it = aChildren.begin(); 2865 ContentRefList::const_iterator end = aChildren.end(); 2866 2867 while ( it != end ) 2868 { 2869 ContentRef xChild = (*it); 2870 2871 // Create new content identifier for the child... 2872 uno::Reference< ucb::XContentIdentifier > 2873 xOldChildId = xChild->getIdentifier(); 2874 rtl::OUString aOldChildURL 2875 = xOldChildId->getContentIdentifier(); 2876 rtl::OUString aNewChildURL 2877 = aOldChildURL.replaceAt( 2878 0, 2879 aOldURL.getLength(), 2880 xNewId->getContentIdentifier() ); 2881 uno::Reference< ucb::XContentIdentifier > xNewChildId 2882 = new ::ucbhelper::ContentIdentifier( 2883 m_xSMgr, aNewChildURL ); 2884 2885 if ( !xChild->exchangeIdentity( xNewChildId ) ) 2886 return sal_False; 2887 2888 ++it; 2889 } 2890 return sal_True; 2891 } 2892 } 2893 2894 OSL_ENSURE( sal_False, 2895 "Content::exchangeIdentity - " 2896 "Panic! Cannot exchange identity!" ); 2897 return sal_False; 2898 } 2899 2900 //========================================================================= 2901 sal_Bool Content::isFolder( 2902 const uno::Reference< ucb::XCommandEnvironment >& xEnv ) 2903 throw( uno::Exception ) 2904 { 2905 { 2906 osl::MutexGuard aGuard( m_aMutex ); 2907 2908 if ( m_bTransient ) 2909 return m_bCollection; 2910 } 2911 2912 uno::Sequence< beans::Property > aProperties( 1 ); 2913 aProperties[ 0 ].Name = rtl::OUString::createFromAscii( "IsFolder" ); 2914 aProperties[ 0 ].Handle = -1; 2915 uno::Reference< sdbc::XRow > xRow( getPropertyValues( aProperties, xEnv ) ); 2916 if ( xRow.is() ) 2917 { 2918 try 2919 { 2920 return xRow->getBoolean( 1 ); 2921 } 2922 catch ( sdbc::SQLException const & ) 2923 { 2924 } 2925 } 2926 2927 return sal_False; 2928 } 2929 2930 //========================================================================= 2931 uno::Any Content::MapDAVException( const DAVException & e, sal_Bool bWrite ) 2932 { 2933 // Map DAVException... 2934 uno::Any aException; 2935 2936 rtl::OUString aURL; 2937 if ( m_bTransient ) 2938 { 2939 aURL = getParentURL(); 2940 if ( aURL.lastIndexOf( '/' ) != ( aURL.getLength() - 1 ) ) 2941 aURL += rtl::OUString::createFromAscii( "/" ); 2942 2943 aURL += m_aEscapedTitle; 2944 } 2945 else 2946 { 2947 aURL = m_xIdentifier->getContentIdentifier(); 2948 } 2949 2950 switch ( e.getStatus() ) 2951 { 2952 case SC_NOT_FOUND: 2953 { 2954 uno::Sequence< uno::Any > aArgs( 1 ); 2955 aArgs[ 0 ] <<= beans::PropertyValue( 2956 rtl::OUString::createFromAscii("Uri"), -1, 2957 uno::makeAny(aURL), 2958 beans::PropertyState_DIRECT_VALUE); 2959 2960 aException <<= 2961 ucb::InteractiveAugmentedIOException( 2962 rtl::OUString::createFromAscii( "Not found!" ), 2963 static_cast< cppu::OWeakObject * >( this ), 2964 task::InteractionClassification_ERROR, 2965 ucb::IOErrorCode_NOT_EXISTING, 2966 aArgs ); 2967 return aException; 2968 } 2969 default: 2970 break; 2971 } 2972 2973 switch ( e.getError() ) 2974 { 2975 case DAVException::DAV_HTTP_ERROR: 2976 { 2977 if ( bWrite ) 2978 aException <<= 2979 ucb::InteractiveNetworkWriteException( 2980 e.getData(), 2981 static_cast< cppu::OWeakObject * >( this ), 2982 task::InteractionClassification_ERROR, 2983 e.getData() ); 2984 else 2985 aException <<= 2986 ucb::InteractiveNetworkReadException( 2987 e.getData(), 2988 static_cast< cppu::OWeakObject * >( this ), 2989 task::InteractionClassification_ERROR, 2990 e.getData() ); 2991 break; 2992 } 2993 2994 case DAVException::DAV_HTTP_LOOKUP: 2995 aException <<= 2996 ucb::InteractiveNetworkResolveNameException( 2997 rtl::OUString(), 2998 static_cast< cppu::OWeakObject * >( this ), 2999 task::InteractionClassification_ERROR, 3000 e.getData() ); 3001 break; 3002 3003 // @@@ No matching InteractiveNetwork*Exception 3004 // case DAVException::DAV_HTTP_AUTH: 3005 // break; 3006 3007 // @@@ No matching InteractiveNetwork*Exception 3008 // case DAVException::DAV_HTTP_AUTHPROXY: 3009 // break; 3010 3011 case DAVException::DAV_HTTP_CONNECT: 3012 aException <<= 3013 ucb::InteractiveNetworkConnectException( 3014 rtl::OUString(), 3015 static_cast< cppu::OWeakObject * >( this ), 3016 task::InteractionClassification_ERROR, 3017 e.getData() ); 3018 break; 3019 3020 // @@@ No matching InteractiveNetwork*Exception 3021 // case DAVException::DAV_HTTP_TIMEOUT: 3022 // break; 3023 3024 // @@@ No matching InteractiveNetwork*Exception 3025 // case DAVException::DAV_HTTP_REDIRECT: 3026 // break; 3027 3028 // @@@ No matching InteractiveNetwork*Exception 3029 // case DAVException::DAV_SESSION_CREATE: 3030 // break; 3031 3032 case DAVException::DAV_INVALID_ARG: 3033 aException <<= 3034 lang::IllegalArgumentException( 3035 rtl::OUString(), 3036 static_cast< cppu::OWeakObject * >( this ), 3037 -1 ); 3038 break; 3039 3040 case DAVException::DAV_LOCKED: 3041 #if 1 3042 aException <<= 3043 ucb::InteractiveLockingLockedException( 3044 rtl::OUString::createFromAscii( "Locked!" ), 3045 static_cast< cppu::OWeakObject * >( this ), 3046 task::InteractionClassification_ERROR, 3047 aURL, 3048 sal_False ); // not SelfOwned 3049 #else 3050 { 3051 uno::Sequence< uno::Any > aArgs( 1 ); 3052 aArgs[ 0 ] <<= beans::PropertyValue( 3053 rtl::OUString::createFromAscii("Uri"), -1, 3054 uno::makeAny(aURL), 3055 beans::PropertyState_DIRECT_VALUE); 3056 3057 aException <<= 3058 ucb::InteractiveAugmentedIOException( 3059 rtl::OUString::createFromAscii( "Locked!" ), 3060 static_cast< cppu::OWeakObject * >( this ), 3061 task::InteractionClassification_ERROR, 3062 ucb::IOErrorCode_LOCKING_VIOLATION, 3063 aArgs ); 3064 } 3065 #endif 3066 break; 3067 3068 case DAVException::DAV_LOCKED_SELF: 3069 aException <<= 3070 ucb::InteractiveLockingLockedException( 3071 rtl::OUString::createFromAscii( "Locked (self)!" ), 3072 static_cast< cppu::OWeakObject * >( this ), 3073 task::InteractionClassification_ERROR, 3074 aURL, 3075 sal_True ); // SelfOwned 3076 break; 3077 3078 case DAVException::DAV_NOT_LOCKED: 3079 aException <<= 3080 ucb::InteractiveLockingNotLockedException( 3081 rtl::OUString::createFromAscii( "Not locked!" ), 3082 static_cast< cppu::OWeakObject * >( this ), 3083 task::InteractionClassification_ERROR, 3084 aURL ); 3085 break; 3086 3087 case DAVException::DAV_LOCK_EXPIRED: 3088 aException <<= 3089 ucb::InteractiveLockingLockExpiredException( 3090 rtl::OUString::createFromAscii( "Lock expired!" ), 3091 static_cast< cppu::OWeakObject * >( this ), 3092 task::InteractionClassification_ERROR, 3093 aURL ); 3094 break; 3095 3096 default: 3097 aException <<= 3098 ucb::InteractiveNetworkGeneralException( 3099 rtl::OUString(), 3100 static_cast< cppu::OWeakObject * >( this ), 3101 task::InteractionClassification_ERROR ); 3102 break; 3103 } 3104 3105 return aException; 3106 } 3107 3108 //========================================================================= 3109 // static 3110 bool Content::shouldAccessNetworkAfterException( const DAVException & e ) 3111 { 3112 if ( ( e.getStatus() == SC_NOT_FOUND ) || 3113 ( e.getError() == DAVException::DAV_HTTP_LOOKUP ) || 3114 ( e.getError() == DAVException::DAV_HTTP_CONNECT ) || 3115 ( e.getError() == DAVException::DAV_HTTP_AUTH ) || 3116 ( e.getError() == DAVException::DAV_HTTP_AUTHPROXY ) ) 3117 return false; 3118 3119 return true; 3120 } 3121 3122 //========================================================================= 3123 void Content::cancelCommandExecution( 3124 const DAVException & e, 3125 const uno::Reference< ucb::XCommandEnvironment > & xEnv, 3126 sal_Bool bWrite /* = sal_False */ ) 3127 throw ( uno::Exception ) 3128 { 3129 ucbhelper::cancelCommandExecution( MapDAVException( e, bWrite ), xEnv ); 3130 // Unreachable 3131 } 3132 3133 //========================================================================= 3134 const rtl::OUString 3135 Content::getBaseURI( const std::auto_ptr< DAVResourceAccess > & rResAccess ) 3136 { 3137 osl::Guard< osl::Mutex > aGuard( m_aMutex ); 3138 3139 // First, try to obtain value of response header "Content-Location". 3140 if ( m_xCachedProps.get() ) 3141 { 3142 rtl::OUString aLocation; 3143 m_xCachedProps->getValue( rtl::OUString( 3144 RTL_CONSTASCII_USTRINGPARAM( 3145 "Content-Location" ) ) ) >>= aLocation; 3146 if ( aLocation.getLength() ) 3147 { 3148 try 3149 { 3150 // Do not use m_xIdentifier->getContentIdentifier() because it 3151 // for example does not reflect redirects applied to requests 3152 // done using the original URI but m_xResAccess' URI does. 3153 return rtl::Uri::convertRelToAbs( rResAccess->getURL(), 3154 aLocation ); 3155 } 3156 catch ( rtl::MalformedUriException const & ) 3157 { 3158 } 3159 } 3160 } 3161 3162 return rtl::OUString( rResAccess->getURL() ); 3163 } 3164 3165 //========================================================================= 3166 const Content::ResourceType & Content::getResourceType( 3167 const uno::Reference< ucb::XCommandEnvironment >& xEnv, 3168 const std::auto_ptr< DAVResourceAccess > & rResAccess ) 3169 throw ( uno::Exception ) 3170 { 3171 if ( m_eResourceType == UNKNOWN ) 3172 { 3173 osl::Guard< osl::Mutex > aGuard( m_aMutex ); 3174 3175 ResourceType eResourceType; 3176 eResourceType = m_eResourceType; 3177 3178 const rtl::OUString & rURL = rResAccess->getURL(); 3179 const rtl::OUString aScheme( 3180 rURL.copy( 0, rURL.indexOf( ':' ) ).toAsciiLowerCase() ); 3181 3182 try 3183 { 3184 // Try to fetch some frequently used property value, e.g. those 3185 // used when loading documents... along with identifying whether 3186 // this is a DAV resource. 3187 std::vector< DAVResource > resources; 3188 std::vector< rtl::OUString > aPropNames; 3189 uno::Sequence< beans::Property > aProperties( 5 ); 3190 aProperties[ 0 ].Name 3191 = rtl::OUString::createFromAscii( "IsFolder" ); 3192 aProperties[ 1 ].Name 3193 = rtl::OUString::createFromAscii( "IsDocument" ); 3194 aProperties[ 2 ].Name 3195 = rtl::OUString::createFromAscii( "IsReadOnly" ); 3196 aProperties[ 3 ].Name 3197 = rtl::OUString::createFromAscii( "MediaType" ); 3198 aProperties[ 4 ].Name 3199 = DAVProperties::SUPPORTEDLOCK; 3200 3201 ContentProperties::UCBNamesToDAVNames( 3202 aProperties, aPropNames ); 3203 3204 rResAccess->PROPFIND( 3205 DAVZERO, aPropNames, resources, xEnv ); 3206 3207 // TODO - is this really only one? 3208 if ( resources.size() == 1 ) 3209 { 3210 m_xCachedProps.reset( 3211 new CachableContentProperties( resources[ 0 ] ) ); 3212 m_xCachedProps->containsAllNames( 3213 aProperties, m_aFailedPropNames ); 3214 } 3215 3216 eResourceType = DAV; 3217 } 3218 catch ( DAVException const & e ) 3219 { 3220 rResAccess->resetUri(); 3221 3222 if ( e.getStatus() == SC_METHOD_NOT_ALLOWED ) 3223 { 3224 // Status SC_METHOD_NOT_ALLOWED is a safe indicator that the 3225 // resource is NON_DAV 3226 eResourceType = NON_DAV; 3227 } 3228 } 3229 m_eResourceType = eResourceType; 3230 } 3231 return m_eResourceType; 3232 } 3233 3234 //========================================================================= 3235 const Content::ResourceType & Content::getResourceType( 3236 const uno::Reference< ucb::XCommandEnvironment >& xEnv ) 3237 throw ( uno::Exception ) 3238 { 3239 return getResourceType( xEnv, m_xResAccess ); 3240 } 3241