/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * This file is part of the LibreOffice project. * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. * * This file incorporates work covered by the following license notice: * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed * with this work for additional information regarding copyright * ownership. The ASF licenses this file to you under the Apache * License, Version 2.0 (the "License"); you may not use this file * except in compliance with the License. You may obtain a copy of * the License at http://www.apache.org/licenses/LICENSE-2.0 . */ #include "ocompinstream.hxx" #include #include #include #include #include #include "owriteablestream.hxx" #include "xstorage.hxx" using namespace ::com::sun::star; OInputCompStream::OInputCompStream( OWriteStream_Impl& aImpl, uno::Reference < io::XInputStream > const & xStream, const uno::Sequence< beans::PropertyValue >& aProps, sal_Int32 nStorageType ) : m_pImpl( &aImpl ) , m_rMutexRef( m_pImpl->m_rMutexRef ) , m_xStream( xStream ) , m_pInterfaceContainer( nullptr ) , m_aProperties( aProps ) , m_bDisposed( false ) , m_nStorageType( nStorageType ) { OSL_ENSURE( m_pImpl->m_rMutexRef.is(), "No mutex is provided!" ); if ( !m_pImpl->m_rMutexRef.is() ) throw uno::RuntimeException(); // just a disaster assert(m_xStream.is()); } OInputCompStream::OInputCompStream( uno::Reference < io::XInputStream > const & xStream, const uno::Sequence< beans::PropertyValue >& aProps, sal_Int32 nStorageType ) : m_pImpl( nullptr ) , m_rMutexRef( new SotMutexHolder ) , m_xStream( xStream ) , m_pInterfaceContainer( nullptr ) , m_aProperties( aProps ) , m_bDisposed( false ) , m_nStorageType( nStorageType ) { assert(m_xStream.is()); } OInputCompStream::~OInputCompStream() { ::osl::MutexGuard aGuard( m_rMutexRef->GetMutex() ); if ( !m_bDisposed ) { m_refCount++; dispose(); } delete m_pInterfaceContainer; } uno::Any SAL_CALL OInputCompStream::queryInterface( const uno::Type& rType ) { uno::Any aReturn; // common interfaces aReturn = ::cppu::queryInterface ( rType , static_cast ( this ) , static_cast ( this ) , static_cast ( this ) , static_cast ( this ) , static_cast ( this ) ); if ( aReturn.hasValue() ) return aReturn ; if ( m_nStorageType == embed::StorageFormats::OFOPXML ) { aReturn = ::cppu::queryInterface ( rType , static_cast ( this ) ); if ( aReturn.hasValue() ) return aReturn ; } return OWeakObject::queryInterface( rType ); } sal_Int32 SAL_CALL OInputCompStream::readBytes( uno::Sequence< sal_Int8 >& aData, sal_Int32 nBytesToRead ) { ::osl::MutexGuard aGuard( m_rMutexRef->GetMutex() ); if ( m_bDisposed ) { SAL_INFO("package.xstor", "Disposed!"); throw lang::DisposedException(); } return m_xStream->readBytes( aData, nBytesToRead ); } sal_Int32 SAL_CALL OInputCompStream::readSomeBytes( uno::Sequence< sal_Int8 >& aData, sal_Int32 nMaxBytesToRead ) { ::osl::MutexGuard aGuard( m_rMutexRef->GetMutex() ); if ( m_bDisposed ) { SAL_INFO("package.xstor", "Disposed!"); throw lang::DisposedException(); } return m_xStream->readSomeBytes( aData, nMaxBytesToRead ); } void SAL_CALL OInputCompStream::skipBytes( sal_Int32 nBytesToSkip ) { ::osl::MutexGuard aGuard( m_rMutexRef->GetMutex() ); if ( m_bDisposed ) { SAL_INFO("package.xstor", "Disposed!"); throw lang::DisposedException(); } m_xStream->skipBytes( nBytesToSkip ); } sal_Int32 SAL_CALL OInputCompStream::available( ) { ::osl::MutexGuard aGuard( m_rMutexRef->GetMutex() ); if ( m_bDisposed ) { SAL_INFO("package.xstor", "Disposed!"); throw lang::DisposedException(); } return m_xStream->available(); } void SAL_CALL OInputCompStream::closeInput( ) { dispose(); } uno::Reference< io::XInputStream > SAL_CALL OInputCompStream::getInputStream() { ::osl::MutexGuard aGuard( m_rMutexRef->GetMutex() ); if ( m_bDisposed ) { SAL_INFO("package.xstor", "Disposed!"); throw lang::DisposedException(); } return uno::Reference< io::XInputStream >( static_cast< io::XInputStream* >( this ), uno::UNO_QUERY ); } uno::Reference< io::XOutputStream > SAL_CALL OInputCompStream::getOutputStream() { ::osl::MutexGuard aGuard( m_rMutexRef->GetMutex() ); if ( m_bDisposed ) { SAL_INFO("package.xstor", "Disposed!"); throw lang::DisposedException(); } return uno::Reference< io::XOutputStream >(); } void OInputCompStream::InternalDispose() { // can be called only by OWriteStream_Impl ::osl::MutexGuard aGuard( m_rMutexRef->GetMutex() ); if ( m_bDisposed ) { SAL_INFO("package.xstor", "Disposed!"); throw lang::DisposedException(); } // the source object is also a kind of locker for the current object // since the listeners could dispose the object while being notified lang::EventObject aSource( static_cast< ::cppu::OWeakObject*>( this ) ); if ( m_pInterfaceContainer ) m_pInterfaceContainer->disposeAndClear( aSource ); try { m_xStream->closeInput(); } catch( uno::Exception& ) {} m_pImpl = nullptr; m_bDisposed = true; } void SAL_CALL OInputCompStream::dispose( ) { ::osl::MutexGuard aGuard( m_rMutexRef->GetMutex() ); if ( m_bDisposed ) { SAL_INFO("package.xstor", "Disposed!"); throw lang::DisposedException(); } if ( m_pInterfaceContainer ) { lang::EventObject aSource( static_cast< ::cppu::OWeakObject*>( this ) ); m_pInterfaceContainer->disposeAndClear( aSource ); } m_xStream->closeInput(); if ( m_pImpl ) { m_pImpl->InputStreamDisposed( this ); m_pImpl = nullptr; } m_bDisposed = true; } void SAL_CALL OInputCompStream::addEventListener( const uno::Reference< lang::XEventListener >& xListener ) { ::osl::MutexGuard aGuard( m_rMutexRef->GetMutex() ); if ( m_bDisposed ) { SAL_INFO("package.xstor", "Disposed!"); throw lang::DisposedException(); } if ( !m_pInterfaceContainer ) m_pInterfaceContainer = new ::comphelper::OInterfaceContainerHelper2( m_rMutexRef->GetMutex() ); m_pInterfaceContainer->addInterface( xListener ); } void SAL_CALL OInputCompStream::removeEventListener( const uno::Reference< lang::XEventListener >& xListener ) { ::osl::MutexGuard aGuard( m_rMutexRef->GetMutex() ); if ( m_bDisposed ) { SAL_INFO("package.xstor", "Disposed!"); throw lang::DisposedException(); } if ( m_pInterfaceContainer ) m_pInterfaceContainer->removeInterface( xListener ); } sal_Bool SAL_CALL OInputCompStream::hasByID( const OUString& sID ) { ::osl::MutexGuard aGuard( m_rMutexRef->GetMutex() ); if ( m_bDisposed ) { SAL_INFO("package.xstor", "Disposed!"); throw lang::DisposedException(); } if ( m_nStorageType != embed::StorageFormats::OFOPXML ) throw uno::RuntimeException(); try { getRelationshipByID( sID ); return true; } catch( container::NoSuchElementException& ) {} return false; } OUString SAL_CALL OInputCompStream::getTargetByID( const OUString& sID ) { ::osl::MutexGuard aGuard( m_rMutexRef->GetMutex() ); if ( m_bDisposed ) { SAL_INFO("package.xstor", "Disposed!"); throw lang::DisposedException(); } if ( m_nStorageType != embed::StorageFormats::OFOPXML ) throw uno::RuntimeException(); uno::Sequence< beans::StringPair > aSeq = getRelationshipByID( sID ); for ( sal_Int32 nInd = 0; nInd < aSeq.getLength(); nInd++ ) if ( aSeq[nInd].First == "Target" ) return aSeq[nInd].Second; return OUString(); } OUString SAL_CALL OInputCompStream::getTypeByID( const OUString& sID ) { ::osl::MutexGuard aGuard( m_rMutexRef->GetMutex() ); if ( m_bDisposed ) { SAL_INFO("package.xstor", "Disposed!"); throw lang::DisposedException(); } if ( m_nStorageType != embed::StorageFormats::OFOPXML ) throw uno::RuntimeException(); uno::Sequence< beans::StringPair > aSeq = getRelationshipByID( sID ); for ( sal_Int32 nInd = 0; nInd < aSeq.getLength(); nInd++ ) if ( aSeq[nInd].First == "Type" ) return aSeq[nInd].Second; return OUString(); } uno::Sequence< beans::StringPair > SAL_CALL OInputCompStream::getRelationshipByID( const OUString& sID ) { ::osl::MutexGuard aGuard( m_rMutexRef->GetMutex() ); if ( m_bDisposed ) { SAL_INFO("package.xstor", "Disposed!"); throw lang::DisposedException(); } if ( m_nStorageType != embed::StorageFormats::OFOPXML ) throw uno::RuntimeException(); // TODO/LATER: in future the unification of the ID could be checked uno::Sequence< uno::Sequence< beans::StringPair > > aSeq = getAllRelationships(); for ( sal_Int32 nInd1 = 0; nInd1 < aSeq.getLength(); nInd1++ ) for ( sal_Int32 nInd2 = 0; nInd2 < aSeq[nInd1].getLength(); nInd2++ ) if ( aSeq[nInd1][nInd2].First == "Id" ) { if ( aSeq[nInd1][nInd2].Second.equals( sID ) ) return aSeq[nInd1]; break; } throw container::NoSuchElementException(); } uno::Sequence< uno::Sequence< beans::StringPair > > SAL_CALL OInputCompStream::getRelationshipsByType( const OUString& sType ) { ::osl::MutexGuard aGuard( m_rMutexRef->GetMutex() ); if ( m_bDisposed ) { SAL_INFO("package.xstor", "Disposed!"); throw lang::DisposedException(); } if ( m_nStorageType != embed::StorageFormats::OFOPXML ) throw uno::RuntimeException(); uno::Sequence< uno::Sequence< beans::StringPair > > aResult; sal_Int32 nEntriesNum = 0; // TODO/LATER: in future the unification of the ID could be checked uno::Sequence< uno::Sequence< beans::StringPair > > aSeq = getAllRelationships(); for ( sal_Int32 nInd1 = 0; nInd1 < aSeq.getLength(); nInd1++ ) for ( sal_Int32 nInd2 = 0; nInd2 < aSeq[nInd1].getLength(); nInd2++ ) if ( aSeq[nInd1][nInd2].First == "Type" ) { if ( aSeq[nInd1][nInd2].Second.equals( sType ) ) { aResult.realloc( nEntriesNum ); aResult[nEntriesNum-1] = aSeq[nInd1]; } break; } return aResult; } uno::Sequence< uno::Sequence< beans::StringPair > > SAL_CALL OInputCompStream::getAllRelationships() { ::osl::MutexGuard aGuard( m_rMutexRef->GetMutex() ); if ( m_bDisposed ) { SAL_INFO("package.xstor", "Disposed!"); throw lang::DisposedException(); } if ( m_nStorageType != embed::StorageFormats::OFOPXML ) throw uno::RuntimeException(); // TODO/LATER: in future the information could be taken directly from m_pImpl when possible uno::Sequence< uno::Sequence< beans::StringPair > > aResult; for ( sal_Int32 aInd = 0; aInd < m_aProperties.getLength(); aInd++ ) if ( m_aProperties[aInd].Name == "RelationsInfo" ) { if ( m_aProperties[aInd].Value >>= aResult ) return aResult; break; } throw io::IOException(); // the relations info could not be read } void SAL_CALL OInputCompStream::insertRelationshipByID( const OUString& /*sID*/, const uno::Sequence< beans::StringPair >& /*aEntry*/, sal_Bool /*bReplace*/ ) { ::osl::MutexGuard aGuard( m_rMutexRef->GetMutex() ); if ( m_bDisposed ) { SAL_INFO("package.xstor", "Disposed!"); throw lang::DisposedException(); } if ( m_nStorageType != embed::StorageFormats::OFOPXML ) throw uno::RuntimeException(); throw io::IOException(); // TODO: Access denied } void SAL_CALL OInputCompStream::removeRelationshipByID( const OUString& /*sID*/ ) { ::osl::MutexGuard aGuard( m_rMutexRef->GetMutex() ); if ( m_bDisposed ) { SAL_INFO("package.xstor", "Disposed!"); throw lang::DisposedException(); } if ( m_nStorageType != embed::StorageFormats::OFOPXML ) throw uno::RuntimeException(); throw io::IOException(); // TODO: Access denied } void SAL_CALL OInputCompStream::insertRelationships( const uno::Sequence< uno::Sequence< beans::StringPair > >& /*aEntries*/, sal_Bool /*bReplace*/ ) { ::osl::MutexGuard aGuard( m_rMutexRef->GetMutex() ); if ( m_bDisposed ) { SAL_INFO("package.xstor", "Disposed!"); throw lang::DisposedException(); } if ( m_nStorageType != embed::StorageFormats::OFOPXML ) throw uno::RuntimeException(); throw io::IOException(); // TODO: Access denied } void SAL_CALL OInputCompStream::clearRelationships() { ::osl::MutexGuard aGuard( m_rMutexRef->GetMutex() ); if ( m_bDisposed ) { SAL_INFO("package.xstor", "Disposed!"); throw lang::DisposedException(); } if ( m_nStorageType != embed::StorageFormats::OFOPXML ) throw uno::RuntimeException(); throw io::IOException(); // TODO: Access denied } uno::Reference< beans::XPropertySetInfo > SAL_CALL OInputCompStream::getPropertySetInfo() { ::osl::MutexGuard aGuard( m_rMutexRef->GetMutex() ); if ( m_bDisposed ) { SAL_INFO("package.xstor", "Disposed!"); throw lang::DisposedException(); } //TODO: return uno::Reference< beans::XPropertySetInfo >(); } void SAL_CALL OInputCompStream::setPropertyValue( const OUString& aPropertyName, const uno::Any& /*aValue*/ ) { ::osl::MutexGuard aGuard( m_rMutexRef->GetMutex() ); if ( m_bDisposed ) { SAL_INFO("package.xstor", "Disposed!"); throw lang::DisposedException(); } // all the provided properties are accessible for ( sal_Int32 aInd = 0; aInd < m_aProperties.getLength(); aInd++ ) { if ( m_aProperties[aInd].Name.equals( aPropertyName ) ) { throw beans::PropertyVetoException(); // TODO } } throw beans::UnknownPropertyException(); // TODO } uno::Any SAL_CALL OInputCompStream::getPropertyValue( const OUString& aProp ) { ::osl::MutexGuard aGuard( m_rMutexRef->GetMutex() ); if ( m_bDisposed ) { SAL_INFO("package.xstor", "Disposed!"); throw lang::DisposedException(); } OUString aPropertyName; if ( aProp == "IsEncrypted" ) aPropertyName = "Encrypted"; else aPropertyName = aProp; if ( aPropertyName == "RelationsInfo" ) throw beans::UnknownPropertyException(); // TODO // all the provided properties are accessible for ( sal_Int32 aInd = 0; aInd < m_aProperties.getLength(); aInd++ ) { if ( m_aProperties[aInd].Name.equals( aPropertyName ) ) { return m_aProperties[aInd].Value; } } throw beans::UnknownPropertyException(); // TODO } void SAL_CALL OInputCompStream::addPropertyChangeListener( const OUString& /*aPropertyName*/, const uno::Reference< beans::XPropertyChangeListener >& /*xListener*/ ) { ::osl::MutexGuard aGuard( m_rMutexRef->GetMutex() ); if ( m_bDisposed ) { SAL_INFO("package.xstor", "Disposed!"); throw lang::DisposedException(); } //TODO: } void SAL_CALL OInputCompStream::removePropertyChangeListener( const OUString& /*aPropertyName*/, const uno::Reference< beans::XPropertyChangeListener >& /*aListener*/ ) { ::osl::MutexGuard aGuard( m_rMutexRef->GetMutex() ); if ( m_bDisposed ) { SAL_INFO("package.xstor", "Disposed!"); throw lang::DisposedException(); } //TODO: } void SAL_CALL OInputCompStream::addVetoableChangeListener( const OUString& /*PropertyName*/, const uno::Reference< beans::XVetoableChangeListener >& /*aListener*/ ) { ::osl::MutexGuard aGuard( m_rMutexRef->GetMutex() ); if ( m_bDisposed ) { SAL_INFO("package.xstor", "Disposed!"); throw lang::DisposedException(); } //TODO: } void SAL_CALL OInputCompStream::removeVetoableChangeListener( const OUString& /*PropertyName*/, const uno::Reference< beans::XVetoableChangeListener >& /*aListener*/ ) { ::osl::MutexGuard aGuard( m_rMutexRef->GetMutex() ); if ( m_bDisposed ) { SAL_INFO("package.xstor", "Disposed!"); throw lang::DisposedException(); } //TODO: } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */