blob: d861e0801bb74c1528d37cbddacfbdcbc54c2c16 [file] [log] [blame]
/**************************************************************
*
* 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
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*
*************************************************************/
// MARKER(update_precomp.py): autogen include statement, do not remove
#include "precompiled_svl.hxx"
#include <svl/restrictedpaths.hxx>
#include <algorithm>
#include <osl/process.h>
#include <tools/urlobj.hxx>
#include <unotools/localfilehelper.hxx>
#include <unotools/syslocale.hxx>
namespace svt
{
namespace
{
// ----------------------------------------------------------------
/** retrieves the value of an environment variable
@return <TRUE/> if and only if the retrieved string value is not empty
*/
bool lcl_getEnvironmentValue( const sal_Char* _pAsciiEnvName, ::rtl::OUString& _rValue )
{
_rValue = ::rtl::OUString();
::rtl::OUString sEnvName = ::rtl::OUString::createFromAscii( _pAsciiEnvName );
osl_getEnvironment( sEnvName.pData, &_rValue.pData );
return _rValue.getLength() != 0;
}
//-----------------------------------------------------------------
void lcl_convertStringListToUrls( const String& _rColonSeparatedList, ::std::vector< String >& _rTokens, bool _bFinalSlash )
{
const sal_Unicode s_cSeparator =
#if defined(WNT)
';'
#else
':'
#endif
;
xub_StrLen nTokens = _rColonSeparatedList.GetTokenCount( s_cSeparator );
_rTokens.resize( 0 ); _rTokens.reserve( nTokens );
for ( xub_StrLen i=0; i<nTokens; ++i )
{
// the current token in the list
String sCurrentToken = _rColonSeparatedList.GetToken( i, s_cSeparator );
if ( !sCurrentToken.Len() )
continue;
INetURLObject aCurrentURL;
String sURL;
if ( ::utl::LocalFileHelper::ConvertPhysicalNameToURL( sCurrentToken, sURL ) )
aCurrentURL = INetURLObject( sURL );
else
{
// smart URL parsing, assuming FILE protocol
aCurrentURL = INetURLObject( sCurrentToken, INET_PROT_FILE );
}
if ( _bFinalSlash )
aCurrentURL.setFinalSlash( );
else
aCurrentURL.removeFinalSlash( );
_rTokens.push_back( aCurrentURL.GetMainURL( INetURLObject::NO_DECODE ) );
}
}
}
//=====================================================================
//= CheckURLAllowed
//=====================================================================
struct CheckURLAllowed
{
protected:
#ifdef WNT
SvtSysLocale m_aSysLocale;
#endif
String m_sCheckURL; // the URL to check
bool m_bAllowParent;
public:
inline CheckURLAllowed( const String& _rCheckURL, bool bAllowParent = true )
:m_sCheckURL( _rCheckURL ), m_bAllowParent( bAllowParent )
{
#ifdef WNT
// on windows, assume that the relevant file systems are case insensitive,
// thus normalize the URL
m_sCheckURL = m_aSysLocale.GetCharClass().toLower( m_sCheckURL, 0, m_sCheckURL.Len() );
#endif
}
bool operator()( const String& _rApprovedURL )
{
#ifdef WNT
// on windows, assume that the relevant file systems are case insensitive,
// thus normalize the URL
String sApprovedURL( m_aSysLocale.GetCharClass().toLower( _rApprovedURL, 0, _rApprovedURL.Len() ) );
#else
String sApprovedURL( _rApprovedURL );
#endif
xub_StrLen nLenApproved = sApprovedURL.Len();
xub_StrLen nLenChecked = m_sCheckURL.Len();
if ( nLenApproved > nLenChecked )
{
if ( m_bAllowParent )
{
if ( sApprovedURL.Search( m_sCheckURL ) == 0 )
{
if ( ( m_sCheckURL.GetChar( nLenChecked - 1 ) == '/' )
|| ( sApprovedURL.GetChar( nLenChecked ) == '/' ) )
return true;
}
}
else
{
// just a difference in final slash?
if ( ( nLenApproved == ( nLenChecked + 1 ) ) &&
( sApprovedURL.GetChar( nLenApproved - 1 ) == '/' ) )
return true;
}
return false;
}
else if ( nLenApproved < nLenChecked )
{
if ( m_sCheckURL.Search( sApprovedURL ) == 0 )
{
if ( ( sApprovedURL.GetChar( nLenApproved - 1 ) == '/' )
|| ( m_sCheckURL.GetChar( nLenApproved ) == '/' ) )
return true;
}
return false;
}
else
{
// strings have equal length
return ( sApprovedURL == m_sCheckURL );
}
}
};
//=====================================================================
//= RestrictedPaths
//=====================================================================
//---------------------------------------------------------------------
RestrictedPaths::RestrictedPaths()
:m_bFilterIsEnabled( true )
{
::rtl::OUString sRestrictedPathList;
if ( lcl_getEnvironmentValue( "RestrictedPath", sRestrictedPathList ) )
// append a final slash. This ensures that when we later on check
// for unrestricted paths, we don't allow paths like "/home/user35" just because
// "/home/user3" is allowed - with the final slash, we make it "/home/user3/".
lcl_convertStringListToUrls( sRestrictedPathList, m_aUnrestrictedURLs, true );
}
RestrictedPaths::~RestrictedPaths() {}
// --------------------------------------------------------------------
bool RestrictedPaths::isUrlAllowed( const String& _rURL ) const
{
if ( m_aUnrestrictedURLs.empty() || !m_bFilterIsEnabled )
return true;
::std::vector< String >::const_iterator aApprovedURL = ::std::find_if(
m_aUnrestrictedURLs.begin(),
m_aUnrestrictedURLs.end(),
CheckURLAllowed( _rURL, true )
);
return ( aApprovedURL != m_aUnrestrictedURLs.end() );
}
// --------------------------------------------------------------------
bool RestrictedPaths::isUrlAllowed( const String& _rURL, bool allowParents ) const
{
if ( m_aUnrestrictedURLs.empty() || !m_bFilterIsEnabled )
return true;
::std::vector< String >::const_iterator aApprovedURL = ::std::find_if(
m_aUnrestrictedURLs.begin(),
m_aUnrestrictedURLs.end(),
CheckURLAllowed( _rURL, allowParents )
);
return ( aApprovedURL != m_aUnrestrictedURLs.end() );
}
} // namespace svt