/* -*- 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 . */ #ifndef INCLUDED_PACKAGE_SOURCE_XSTOR_XSTORAGE_HXX #define INCLUDED_PACKAGE_SOURCE_XSTOR_XSTORAGE_HXX #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "ohierarchyholder.hxx" #include "disposelistener.hxx" #include #include #include #include namespace com::sun::star::uno { class XComponentContext; } #define RELINFO_NO_INIT 1 #define RELINFO_READ 2 #define RELINFO_CHANGED 3 #define RELINFO_CHANGED_STREAM 4 #define RELINFO_CHANGED_STREAM_READ 5 #define RELINFO_BROKEN 6 #define RELINFO_CHANGED_BROKEN 7 #define STOR_MESS_PRECOMMIT 1 #define STOR_MESS_COMMITTED 2 #define STOR_MESS_PREREVERT 3 #define STOR_MESS_REVERTED 4 // a common implementation for an entry struct OStorage_Impl; struct OWriteStream_Impl; struct SotElement_Impl { OUString m_aOriginalName; bool m_bIsRemoved; bool m_bIsInserted; bool m_bIsStorage; std::unique_ptr m_xStorage; std::unique_ptr> m_xStream; public: SotElement_Impl(OUString aName, bool bStor, bool bNew); }; // Main storage implementation class OStorage; struct StorageHolder_Impl { OStorage* m_pPointer; css::uno::WeakReference< css::embed::XStorage > m_xWeakRef; explicit inline StorageHolder_Impl( OStorage* pStorage ); }; class SwitchablePersistenceStream; struct OStorage_Impl { typedef std::vector StorageHoldersType; rtl::Reference m_xMutex; OStorage* m_pAntiImpl; // only valid if external references exists StorageHoldersType m_aReadOnlyWrapVector; // only valid if readonly external reference exists sal_Int32 m_nStorageMode; // open mode ( read/write/trunc/nocreate ) bool m_bIsModified; // only modified elements will be sent to the original content bool m_bBroadcastModified; // will be set if notification is required bool m_bCommited; // sending the streams is coordinated by the root storage of the package bool m_bIsRoot; // marks this storage as root storages that manages all commits and reverts bool m_bListCreated; bool m_bRepairPackage = false; /// Count of registered modification listeners oslInterlockedCount m_nModifiedListenerCount; bool HasModifiedListener() const { return m_nModifiedListenerCount > 0 && m_pAntiImpl != nullptr; } std::unordered_map> m_aChildrenMap; std::vector< SotElement_Impl* > m_aDeletedVector; css::uno::Reference< css::container::XNameContainer > m_xPackageFolder; css::uno::Reference< css::lang::XSingleServiceFactory > m_xPackage; css::uno::Reference< css::uno::XComponentContext > m_xContext; // valid only for root storage css::uno::Reference< css::io::XInputStream > m_xInputStream; // ??? may be stored in properties css::uno::Reference< css::io::XStream > m_xStream; // ??? may be stored in properties css::uno::Sequence< css::beans::PropertyValue > m_xProperties; bool m_bHasCommonEncryptionData; ::comphelper::SequenceAsHashMap m_aCommonEncryptionData; // must be empty in case of root storage OStorage_Impl* m_pParent; bool m_bControlMediaType; OUString m_aMediaType; bool m_bMTFallbackUsed; bool m_bControlVersion; OUString m_aVersion; rtl::Reference m_pSwitchStream; sal_Int32 m_nStorageType; // the mode in which the storage is used // the _rels substorage that is handled in a special way in embed::StorageFormats::OFOPXML SotElement_Impl* m_pRelStorElement; css::uno::Reference< css::embed::XStorage > m_xRelStorage; css::uno::Sequence< css::uno::Sequence< css::beans::StringPair > > m_aRelInfo; css::uno::Reference< css::io::XInputStream > m_xNewRelInfoStream; sal_Int16 m_nRelInfoStatus; // Constructors OStorage_Impl( css::uno::Reference< css::io::XInputStream > const & xInputStream, sal_Int32 nMode, const css::uno::Sequence< css::beans::PropertyValue >& xProperties, css::uno::Reference< css::uno::XComponentContext > const & xContext, sal_Int32 nStorageType ); OStorage_Impl( css::uno::Reference< css::io::XStream > const & xStream, sal_Int32 nMode, const css::uno::Sequence< css::beans::PropertyValue >& xProperties, css::uno::Reference< css::uno::XComponentContext > const & xContext, sal_Int32 nStorageType ); // constructor for a substorage OStorage_Impl( OStorage_Impl* pParent, sal_Int32 nMode, css::uno::Reference< css::container::XNameContainer > const & xPackageFolder, css::uno::Reference< css::lang::XSingleServiceFactory > xPackage, css::uno::Reference< css::uno::XComponentContext > const & xContext, sal_Int32 nStorageType ); ~OStorage_Impl(); void SetReadOnlyWrap( OStorage& aStorage ); void RemoveReadOnlyWrap( const OStorage& aStorage ); void OpenOwnPackage(); void ReadContents(); void ReadRelInfoIfNecessary(); bool HasChildren(); void GetStorageProperties(); css::uno::Sequence< css::uno::Sequence< css::beans::StringPair > > GetAllRelationshipsIfAny(); void CopyLastCommitTo( const css::uno::Reference< css::embed::XStorage >& xNewStor ); void InsertIntoPackageFolder( const OUString& aName, const css::uno::Reference< css::container::XNameContainer >& xParentPackageFolder ); void Commit(); void Revert(); /// @throws css::packages::NoEncryptionException ::comphelper::SequenceAsHashMap GetCommonRootEncryptionData(); void CopyToStorage( const css::uno::Reference< css::embed::XStorage >& xDest, bool bDirect ); void CopyStorageElement( SotElement_Impl* pElement, const css::uno::Reference< css::embed::XStorage >& xDest, const OUString& aName, bool bDirect ); SotElement_Impl* FindElement( const OUString& rName ); SotElement_Impl* InsertStream( const OUString& aName, bool bEncr ); void InsertRawStream( const OUString& aName, const css::uno::Reference< css::io::XInputStream >& xInStream ); std::unique_ptr CreateNewStorageImpl( sal_Int32 nStorageMode ); SotElement_Impl* InsertStorage( const OUString& aName, sal_Int32 nStorageMode ); SotElement_Impl* InsertElement( const OUString& aName, bool bIsStorage ); void OpenSubStorage( SotElement_Impl* pElement, sal_Int32 nStorageMode ); void OpenSubStream( SotElement_Impl* pElement ); css::uno::Sequence< OUString > GetElementNames(); void RemoveElement( OUString const & rName, SotElement_Impl* pElement ); static void ClearElement( SotElement_Impl* pElement ); /// @throws css::embed::InvalidStorageException /// @throws css::lang::IllegalArgumentException /// @throws css::packages::WrongPasswordException /// @throws css::packages::NoEncryptionException /// @throws css::container::NoSuchElementException /// @throws css::io::IOException /// @throws css::embed::StorageWrappedTargetException /// @throws css::uno::RuntimeException void CloneStreamElement( const OUString& aStreamName, bool bPassProvided, const ::comphelper::SequenceAsHashMap& aEncryptionData, css::uno::Reference< css::io::XStream >& xTargetStream ); void RemoveStreamRelInfo( std::u16string_view aOriginalName ); void CreateRelStorage(); void CommitStreamRelInfo( std::u16string_view rName, SotElement_Impl const * pStreamElement ); css::uno::Reference< css::io::XInputStream > GetRelInfoStreamForName( std::u16string_view aName ); void CommitRelInfo( const css::uno::Reference< css::container::XNameContainer >& xNewPackageFolder ); static void completeStorageStreamCopy_Impl( const css::uno::Reference< css::io::XStream >& xSource, const css::uno::Reference< css::io::XStream >& xDest, sal_Int32 nStorageType, const css::uno::Sequence< css::uno::Sequence< css::beans::StringPair > >& aRelInfo ); }; class OStorage final : public css::lang::XTypeProvider , public css::embed::XStorage2 , public css::embed::XStorageRawAccess , public css::embed::XTransactedObject , public css::embed::XTransactionBroadcaster , public css::util::XModifiable , public css::embed::XEncryptionProtectedStorage , public css::beans::XPropertySet , public css::embed::XOptimizedStorage , public css::embed::XRelationshipAccess , public css::embed::XHierarchicalStorageAccess2 , public ::cppu::OWeakObject { OStorage_Impl* m_pImpl; rtl::Reference m_xSharedMutex; comphelper::OMultiTypeInterfaceContainerHelper2 m_aListenersContainer; // list of listeners ::std::optional< ::cppu::OTypeCollection> m_oTypeCollection; bool m_bReadOnlyWrap; ::rtl::Reference m_pSubElDispListener; ::std::vector< css::uno::WeakReference< css::lang::XComponent > > m_aOpenSubComponentsVector; ::rtl::Reference< OHierarchyHolder_Impl > m_rHierarchyHolder; SotElement_Impl* OpenStreamElement_Impl( const OUString& aStreamName, sal_Int32 nOpenMode, bool bEncr ); void BroadcastModifiedIfNecessary(); void BroadcastTransaction( sal_Int8 nMessage ); void MakeLinkToSubComponent_Impl( const css::uno::Reference< css::lang::XComponent >& xComponent ); public: OStorage( css::uno::Reference< css::io::XInputStream > const & xInputStream, sal_Int32 nMode, const css::uno::Sequence< css::beans::PropertyValue >& xProperties, css::uno::Reference< css::uno::XComponentContext > const & xContext, sal_Int32 nStorageType ); OStorage( css::uno::Reference< css::io::XStream > const & xStream, sal_Int32 nMode, const css::uno::Sequence< css::beans::PropertyValue >& xProperties, css::uno::Reference< css::uno::XComponentContext > const & xContext, sal_Int32 nStorageType ); OStorage( OStorage_Impl* pImpl, bool bReadOnlyWrap ); virtual ~OStorage() override; void InternalDispose( bool bNotifyImpl ); void ChildIsDisposed( const css::uno::Reference< css::uno::XInterface >& xChild ); sal_Int32 GetRefCount_Impl() const { return m_refCount; } // XInterface virtual css::uno::Any SAL_CALL queryInterface( const css::uno::Type& rType ) override; virtual void SAL_CALL acquire() noexcept override; virtual void SAL_CALL release() noexcept override; // XTypeProvider virtual css::uno::Sequence< css::uno::Type > SAL_CALL getTypes() override; virtual css::uno::Sequence< sal_Int8 > SAL_CALL getImplementationId() override; // XStorage virtual void SAL_CALL copyToStorage( const css::uno::Reference< css::embed::XStorage >& xDest ) override; virtual css::uno::Reference< css::io::XStream > SAL_CALL openStreamElement( const OUString& aStreamName, sal_Int32 nOpenMode ) override; virtual css::uno::Reference< css::io::XStream > SAL_CALL openEncryptedStreamElement( const OUString& aStreamName, sal_Int32 nOpenMode, const OUString& aPass ) override; virtual css::uno::Reference< css::embed::XStorage > SAL_CALL openStorageElement( const OUString& aStorName, sal_Int32 nStorageMode ) override; virtual css::uno::Reference< css::io::XStream > SAL_CALL cloneStreamElement( const OUString& aStreamName ) override; virtual css::uno::Reference< css::io::XStream > SAL_CALL cloneEncryptedStreamElement( const OUString& aStreamName, const OUString& aPass ) override; virtual void SAL_CALL copyLastCommitTo( const css::uno::Reference< css::embed::XStorage >& xTargetStorage ) override; virtual void SAL_CALL copyStorageElementLastCommitTo( const OUString& aStorName, const css::uno::Reference< css::embed::XStorage >& xTargetStorage ) override; virtual sal_Bool SAL_CALL isStreamElement( const OUString& aElementName ) override; virtual sal_Bool SAL_CALL isStorageElement( const OUString& aElementName ) override; virtual void SAL_CALL removeElement( const OUString& aElementName ) override; virtual void SAL_CALL renameElement( const OUString& rEleName, const OUString& rNewName ) override; virtual void SAL_CALL copyElementTo( const OUString& aElementName, const css::uno::Reference< css::embed::XStorage >& xDest, const OUString& aNewName ) override; virtual void SAL_CALL moveElementTo( const OUString& aElementName, const css::uno::Reference< css::embed::XStorage >& xDest, const OUString& rNewName ) override; // XStorage2 virtual css::uno::Reference< css::io::XStream > SAL_CALL openEncryptedStream( const OUString& sStreamName, ::sal_Int32 nOpenMode, const css::uno::Sequence< css::beans::NamedValue >& aEncryptionData ) override; virtual css::uno::Reference< css::io::XStream > SAL_CALL cloneEncryptedStream( const OUString& sStreamName, const css::uno::Sequence< css::beans::NamedValue >& aEncryptionData ) override; // XStorageRawAccess virtual css::uno::Reference< css::io::XInputStream > SAL_CALL getPlainRawStreamElement( const OUString& sStreamName ) override; virtual css::uno::Reference< css::io::XInputStream > SAL_CALL getRawEncrStreamElement( const OUString& sStreamName ) override; virtual void SAL_CALL insertRawEncrStreamElement( const OUString& aStreamName, const css::uno::Reference< css::io::XInputStream >& xInStream ) override; // XTransactedObject virtual void SAL_CALL commit() override; virtual void SAL_CALL revert() override; // XTransactionBroadcaster virtual void SAL_CALL addTransactionListener( const css::uno::Reference< css::embed::XTransactionListener >& aListener ) override; virtual void SAL_CALL removeTransactionListener( const css::uno::Reference< css::embed::XTransactionListener >& aListener ) override; // XModifiable virtual sal_Bool SAL_CALL isModified() override; virtual void SAL_CALL setModified( sal_Bool bModified ) override; virtual void SAL_CALL addModifyListener( const css::uno::Reference< css::util::XModifyListener >& aListener ) override; virtual void SAL_CALL removeModifyListener( const css::uno::Reference< css::util::XModifyListener >& aListener ) override; // XNameAccess virtual css::uno::Any SAL_CALL getByName( const OUString& aName ) override; virtual css::uno::Sequence< OUString > SAL_CALL getElementNames() override; virtual sal_Bool SAL_CALL hasByName( const OUString& aName ) override; virtual css::uno::Type SAL_CALL getElementType() override; virtual sal_Bool SAL_CALL hasElements() override; // XComponent virtual void SAL_CALL dispose() override; virtual void SAL_CALL addEventListener( const css::uno::Reference< css::lang::XEventListener >& xListener ) override; virtual void SAL_CALL removeEventListener( const css::uno::Reference< css::lang::XEventListener >& xListener ) override; // XEncryptionProtectedSource virtual void SAL_CALL setEncryptionPassword( const OUString& aPass ) override; virtual void SAL_CALL removeEncryption() override; // XEncryptionProtectedSource2 virtual void SAL_CALL setEncryptionData( const css::uno::Sequence< css::beans::NamedValue >& aEncryptionData ) override; virtual sal_Bool SAL_CALL hasEncryptionData() override; // XEncryptionProtectedStorage virtual void SAL_CALL setEncryptionAlgorithms( const css::uno::Sequence< css::beans::NamedValue >& aAlgorithms ) override; virtual void SAL_CALL setGpgProperties( const css::uno::Sequence< css::uno::Sequence< css::beans::NamedValue > >& aCryptProps ) override; virtual css::uno::Sequence< css::beans::NamedValue > SAL_CALL getEncryptionAlgorithms() override; // XPropertySet virtual css::uno::Reference< css::beans::XPropertySetInfo > SAL_CALL getPropertySetInfo() override; virtual void SAL_CALL setPropertyValue( const OUString& aPropertyName, const css::uno::Any& aValue ) override; virtual css::uno::Any SAL_CALL getPropertyValue( const OUString& PropertyName ) override; virtual void SAL_CALL addPropertyChangeListener( const OUString& aPropertyName, const css::uno::Reference< css::beans::XPropertyChangeListener >& xListener ) override; virtual void SAL_CALL removePropertyChangeListener( const OUString& aPropertyName, const css::uno::Reference< css::beans::XPropertyChangeListener >& aListener ) override; virtual void SAL_CALL addVetoableChangeListener( const OUString& PropertyName, const css::uno::Reference< css::beans::XVetoableChangeListener >& aListener ) override; virtual void SAL_CALL removeVetoableChangeListener( const OUString& PropertyName, const css::uno::Reference< css::beans::XVetoableChangeListener >& aListener ) override; // XOptimizedStorage virtual void SAL_CALL insertRawNonEncrStreamElementDirect( const OUString& sStreamName, const css::uno::Reference< css::io::XInputStream >& xInStream ) override; virtual void SAL_CALL insertStreamElementDirect( const OUString& sStreamName, const css::uno::Reference< css::io::XInputStream >& xInStream, const css::uno::Sequence< css::beans::PropertyValue >& aProps ) override; virtual void SAL_CALL copyElementDirectlyTo( const OUString& sSourceName, const css::uno::Reference< css::embed::XOptimizedStorage >& xTargetStorage, const OUString& sTargetName ) override; virtual void SAL_CALL writeAndAttachToStream( const css::uno::Reference< css::io::XStream >& xStream ) override; virtual void SAL_CALL attachToURL( const OUString& sURL, sal_Bool bReadOnly ) override; virtual css::uno::Any SAL_CALL getElementPropertyValue( const OUString& sElementName, const OUString& sPropertyName ) override; virtual void SAL_CALL copyStreamElementData( const OUString& sStreamName, const css::uno::Reference< css::io::XStream >& xTargetStream ) override; // XRelationshipAccess virtual sal_Bool SAL_CALL hasByID( const OUString& sID ) override; virtual OUString SAL_CALL getTargetByID( const OUString& sID ) override; virtual OUString SAL_CALL getTypeByID( const OUString& sID ) override; virtual css::uno::Sequence< css::beans::StringPair > SAL_CALL getRelationshipByID( const OUString& sID ) override; virtual css::uno::Sequence< css::uno::Sequence< css::beans::StringPair > > SAL_CALL getRelationshipsByType( const OUString& sType ) override; virtual css::uno::Sequence< css::uno::Sequence< css::beans::StringPair > > SAL_CALL getAllRelationships( ) override; virtual void SAL_CALL insertRelationshipByID( const OUString& sID, const css::uno::Sequence< css::beans::StringPair >& aEntry, sal_Bool bReplace ) override; virtual void SAL_CALL removeRelationshipByID( const OUString& sID ) override; virtual void SAL_CALL insertRelationships( const css::uno::Sequence< css::uno::Sequence< css::beans::StringPair > >& aEntries, sal_Bool bReplace ) override; virtual void SAL_CALL clearRelationships( ) override; // XHierarchicalStorageAccess virtual css::uno::Reference< css::embed::XExtendedStorageStream > SAL_CALL openStreamElementByHierarchicalName( const OUString& sStreamPath, ::sal_Int32 nOpenMode ) override; virtual css::uno::Reference< css::embed::XExtendedStorageStream > SAL_CALL openEncryptedStreamElementByHierarchicalName( const OUString& sStreamName, ::sal_Int32 nOpenMode, const OUString& sPassword ) override; virtual void SAL_CALL removeStreamElementByHierarchicalName( const OUString& sElementPath ) override; // XHierarchicalStorageAccess2 virtual css::uno::Reference< css::embed::XExtendedStorageStream > SAL_CALL openEncryptedStreamByHierarchicalName( const OUString& sStreamName, ::sal_Int32 nOpenMode, const css::uno::Sequence< css::beans::NamedValue >& aEncryptionData ) override; }; StorageHolder_Impl::StorageHolder_Impl( OStorage* pStorage ) : m_pPointer( pStorage ) , m_xWeakRef( css::uno::Reference< css::embed::XStorage >( pStorage ) ) { } #endif /* vim:set shiftwidth=4 softtabstop=4 expandtab: */