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