Files
libreoffice/unotools/source/config/bootstrap.cxx

859 lines
27 KiB
C++
Raw Normal View History

/*************************************************************************
*
* $RCSfile: bootstrap.cxx,v $
*
* $Revision: 1.6 $
*
* last change: $Author: jb $ $Date: 2001-09-25 10:27:16 $
*
* The Contents of this file are made available subject to the terms of
* either of the following licenses
*
* - GNU Lesser General Public License Version 2.1
* - Sun Industry Standards Source License Version 1.1
*
* Sun Microsystems Inc., October, 2000
*
* GNU Lesser General Public License Version 2.1
* =============================================
* Copyright 2000 by Sun Microsystems, Inc.
* 901 San Antonio Road, Palo Alto, CA 94303, USA
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License version 2.1, as published by the Free Software Foundation.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library; if not, write to the Free Software
* Foundation, Inc., 59 Temple Place, Suite 330, Boston,
* MA 02111-1307 USA
*
*
* Sun Industry Standards Source License Version 1.1
* =================================================
* The contents of this file are subject to the Sun Industry Standards
* Source License Version 1.1 (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.openoffice.org/license.html.
*
* Software provided under this License is provided on an "AS IS" basis,
* WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING,
* WITHOUT LIMITATION, WARRANTIES THAT THE SOFTWARE IS FREE OF DEFECTS,
* MERCHANTABLE, FIT FOR A PARTICULAR PURPOSE, OR NON-INFRINGING.
* See the License for the specific provisions governing your rights and
* obligations concerning the Software.
*
* The Initial Developer of the Original Code is: Sun Microsystems, Inc.
*
* Copyright: 2000 by Sun Microsystems, Inc.
*
* All Rights Reserved.
*
* Contributor(s): _______________________________________
*
*
************************************************************************/
#include <stdio.h>
#include "unotools/bootstrap.hxx"
// ---------------------------------------------------------------------------------------
#ifndef _RTL_USTRING_HXX_
#include <rtl/ustring.hxx>
#endif
#ifndef _RTL_USTRBUF_HXX_
#include <rtl/ustrbuf.hxx>
#endif
#ifndef _OSL_FILE_HXX_
#include <osl/file.hxx>
#endif
#ifndef _OSL_MUTEX_HXX_
#include <osl/mutex.hxx>
#endif
#ifndef _OSL_DIAGNOSE_H_
#include <osl/diagnose.h>
#endif
// ---------------------------------------------------------------------------------------
#ifndef _RTL_STRING_HXX_
#include <rtl/string.hxx>
#endif
#ifndef _OSL_PROFILE_HXX_
#include <osl/profile.hxx>
#endif
#ifndef _OSL_SECURITY_H_
#include <osl/security.h> // for osl_getConfigDir
#endif
#ifndef _OSL_PROCESS_H_
#include <osl/process.h> // for osl_getExecutableFile
#endif
#ifndef _OSL_THREAD_H_
#include <osl/thread.h> // for osl_getThreadTextEncoding
#endif
// ---------------------------------------------------------------------------------------
#define BOOTSTRAP_FILE SAL_CONFIGFILE( "bootstrap" )
#define BOOTSTRAP_SECTION "Bootstrap"
#define BOOTSTRAP_ITEM_PRODUCT_KEY "ProductKey"
#define BOOTSTRAP_ITEM_LOCATION "Location"
#define BOOTSTRAP_ITEM_SECTION "Section"
#define BOOTSTRAP_ITEM_LOGO "Logo"
#define BOOTSTRAP_VALUE_SECTION_DEFAULT "Versions"
// ---------------------------------------------------------------------------------------
typedef char const * AsciiString;
// ---------------------------------------------------------------------------------------
namespace utl
{
// ---------------------------------------------------------------------------------------
using ::rtl::OUString;
using ::rtl::OUStringBuffer;
using ::rtl::OString;
// ---------------------------------------------------------------------------------------
// Implementation class: Bootstrap::Impl
// ---------------------------------------------------------------------------------------
class Bootstrap::Impl
{
public: // struct to cache the result of a path lookup
struct PathData
{
OUString path;
PathStatus status;
PathData()
: path()
, status(DATA_UNKNOWN)
{}
};
public: // data members
// base install data
PathData aBootstrapINI_;
PathData aBaseInstall_;
OUString sExename_;
// user install data
PathData aVersionINI_;
PathData aUserInstall_;
// overall status
Status status_;
public: // construction and initialization
Impl()
{
status_ = initialize();
}
Status initialize();
// access helper - only use for pure ASCII data
OUString getBootstrapValue(AsciiString _sName, OUString const& _sDefault) const;
private: // implementation
bool initBaseInstallationData();
bool initUserInstallationData();
};
// ---------------------------------------------------------------------------------------
Bootstrap::Impl const& Bootstrap::data()
{
static Impl* s_pData = NULL;
if (!s_pData)
{
using namespace osl;
MutexGuard aGuard( Mutex::getGlobalMutex() );
static Impl s_theData;
s_pData = &s_theData;
}
return *s_pData;
}
// ---------------------------------------------------------------------------------------
// helper
// ---------------------------------------------------------------------------------------
typedef Bootstrap::PathStatus PathStatus;
sal_Unicode const cURLSeparator = '/';
// ---------------------------------------------------------------------------------------
static
inline
OUString getURLSeparator()
{
static OUString theSep(&cURLSeparator,1);
return theSep;
}
// ---------------------------------------------------------------------------------------
// path status utility function
static
PathStatus implCheckStatusOfURL(OUString const& _sURL, osl::DirectoryItem& aDirItem)
{
using namespace osl;
PathStatus eStatus = Bootstrap::DATA_UNKNOWN;
if (_sURL.getLength() != 0)
{
switch( DirectoryItem::get(_sURL, aDirItem) )
{
case DirectoryItem::E_None: // Success
eStatus = Bootstrap::PATH_EXISTS;
break;
case DirectoryItem::E_NOENT: // No such file or directory<br>
eStatus = Bootstrap::PATH_VALID;
break;
case DirectoryItem::E_INVAL: // the format of the parameters was not valid<br>
case DirectoryItem::E_NAMETOOLONG: // File name too long<br>
case DirectoryItem::E_NOTDIR: // A component of the path prefix of path is not a directory<p>
eStatus = Bootstrap::DATA_INVALID;
break;
// how to handle these ?
case DirectoryItem::E_LOOP: // Too many symbolic links encountered<br>
case DirectoryItem::E_ACCES: // permission denied<br>
// any other error - what to do ?
default:
eStatus = Bootstrap::DATA_UNKNOWN;
break;
}
}
else
eStatus = Bootstrap::DATA_MISSING;
return eStatus;
}
// ---------------------------------------------------------------------------------------
static
bool implNormalizeURL(OUString & _sURL, osl::DirectoryItem& aDirItem)
{
using namespace osl;
OSL_PRECOND(aDirItem.is(), "Opened DirItem required");
static const sal_uInt32 cFileStatusMask = FileStatusMask_FileURL;
FileStatus aFileStatus(cFileStatusMask);
if (aDirItem.getFileStatus(aFileStatus) != DirectoryItem::E_None)
return false;
OUString aNormalizedURL = aFileStatus.getFileURL();
if (aNormalizedURL.getLength() == 0)
return false;
_sURL = aNormalizedURL;
return true;
}
// ---------------------------------------------------------------------------------------
#ifdef DEBUG
static
PathStatus dbgCheckStatusOfURL(OUString const& _sURL)
{
using namespace osl;
DirectoryItem aDirItem;
return implCheckStatusOfURL(_sURL,aDirItem);
}
// ---------------------------------------------------------------------------------------
#endif
static
bool normalizeURL(OUString & _sURL)
{
using namespace osl;
DirectoryItem aDirItem;
bool bResult = DirectoryItem::get(_sURL, aDirItem) == DirectoryItem::E_None;
if ( bResult )
bResult = implNormalizeURL(_sURL,aDirItem);
return bResult;
}
// ---------------------------------------------------------------------------------------
static
PathStatus checkStatusAndNormalizeURL(OUString & _sURL)
{
using namespace osl;
DirectoryItem aDirItem;
PathStatus eStatus = implCheckStatusOfURL(_sURL,aDirItem);
if (eStatus == Bootstrap::PATH_EXISTS)
{
if (!implNormalizeURL(_sURL,aDirItem))
OSL_ENSURE(false,"Unexpected failure getting actual URL for existing object");
}
return eStatus;
}
// ----------------------------------------------------------------------------------
// helpers to build and check a nested URL
static
PathStatus getDerivedPath(OUString& _rURL, OUString const& _sRelativeURL, OUString const& _aBaseURL, PathStatus _aBaseStatus)
{
OSL_ASSERT(_sRelativeURL.getLength() != 0 && _sRelativeURL[0] != cURLSeparator);
PathStatus aStatus = _aBaseStatus;
// do we have a base path ?
if (_aBaseURL.getLength())
{
OSL_PRECOND(_aBaseURL[_aBaseURL.getLength()-1] != cURLSeparator,"Unexpected: base URL ends in slash");
_rURL = _aBaseURL + getURLSeparator() + _sRelativeURL;
// a derived (nested) URL can only exist or have a lesser status, if the parent exists
if (aStatus == Bootstrap::PATH_EXISTS)
aStatus = checkStatusAndNormalizeURL(_rURL);
else // the relative appendix must be valid
OSL_ASSERT(aStatus != Bootstrap::PATH_VALID || dbgCheckStatusOfURL(_rURL) == Bootstrap::PATH_VALID);
}
else
{
// clear the result
_rURL = _aBaseURL;
// if we have no data it can't be a valid path
OSL_ASSERT( aStatus > Bootstrap::PATH_VALID );
}
return aStatus;
}
// ----------------------------------------------------------------------------------
static
inline
PathStatus getDerivedPath(OUString& _rURL, OUString const& _sRelativeURL, Bootstrap::Impl::PathData const& _aBaseData)
{
return getDerivedPath(_rURL,_sRelativeURL,_aBaseData.path,_aBaseData.status);
}
// ---------------------------------------------------------------------------------------
static bool normalizeAndSubstitutePathVariables(OUString& _rPath)
{
OSL_PRECOND(_rPath.getLength() != 0 , "Invalid parameter: Empty path in normalizeAndSubstitutePathVariables" );
typedef sal_Bool (SAL_CALL * getSystemDirectoryFunction)(oslSecurity, rtl_uString **);
// recognized variables and methods to retrieve their substitute pathes
// Note: variables are expected to be an absolute URL and thus must be at the beginning of _rPath
struct
{
sal_Char const * pattern;
getSystemDirectoryFunction getPath;
}
aPathVariables[] =
{
{ "$(SYSUSERHOME)", ::osl_getHomeDir },
{ "$(SYSUSERCONFIG)", ::osl_getConfigDir }
};
bool bIsNormalized = false;
// check for all variables
for (sal_Int32 i=0; i < sizeof(aPathVariables)/sizeof(aPathVariables[0]); ++i)
{
OUString sVariable = OUString::createFromAscii(aPathVariables[i].pattern);
sal_Int32 nVariablePos = _rPath.indexOf(sVariable);
OSL_ENSURE( 0 >= nVariablePos, "Unexpected: System directory variable in the middle of a path");
if (0 == nVariablePos)
{
OSL_ENSURE( !bIsNormalized, "Found multiple path variables");
oslSecurity aCurrentUserSec = osl_getCurrentSecurity();
OUString sSubstitute;
aPathVariables[i].getPath(aCurrentUserSec, &sSubstitute.pData);
_rPath = _rPath.replaceAt(nVariablePos, sVariable.getLength(), sSubstitute);
bIsNormalized = true;
#ifndef DBG_UTIL
break; // in debug builds continue for additional checks
#endif // DBG_UTIL
}
}
if ( !bIsNormalized)
{
OUString sOther;
// make a normalized path
if ( osl::File::E_None == osl::File::getFileURLFromSystemPath(_rPath, sOther) )
{
_rPath = sOther;
bIsNormalized = true;
}
// check if it already was normalized
else
bIsNormalized = ( osl::File::E_None == osl::File::getSystemPathFromFileURL(_rPath, sOther) );
}
return bIsNormalized;
}
// ----------------------------------------------------------------------------------
static
void readPathFromINI(Bootstrap::Impl::PathData & _rResult, osl::Profile & _rIniFile, OString const& _sSection, OString const& _sEntry)
{
// encoding to use for converting pathes read from an INI
rtl_TextEncoding const nCvtEncoding = osl_getThreadTextEncoding();
OString sConvertable = _rIniFile.readString(_sSection, _sEntry, OString());
_rResult.path = rtl::OStringToOUString(sConvertable, nCvtEncoding);
if ( _rResult.path.getLength() == 0)
_rResult.status = Bootstrap::DATA_MISSING;
else if ( normalizeAndSubstitutePathVariables(_rResult.path) )
_rResult.status = checkStatusAndNormalizeURL(_rResult.path);
else
_rResult.status = Bootstrap::DATA_INVALID;
}
// ---------------------------------------------------------------------------------------
2001-08-09 11:22:27 +00:00
// Error reporting
2001-08-09 11:22:27 +00:00
static char const IS_MISSING[] = "is missing";
static char const IS_INVALID[] = "is corrupt";
static char const PERIOD[] = ". ";
// ---------------------------------------------------------------------------------------
static void addFileError(OUStringBuffer& _rBuf, OUString const& _aPath, AsciiString _sWhat)
{
OUString sSimpleFileName = _aPath.copy(1 +_aPath.lastIndexOf(cURLSeparator));
2001-08-09 11:22:27 +00:00
_rBuf.appendAscii("The configuration file");
_rBuf.appendAscii(" '").append(sSimpleFileName).appendAscii("' ");
_rBuf.appendAscii(_sWhat).appendAscii(PERIOD);
}
// ---------------------------------------------------------------------------------------
2001-08-09 11:22:27 +00:00
static void addMissingDirectoryError(OUStringBuffer& _rBuf, OUString const& _aPath)
{
2001-08-09 11:22:27 +00:00
_rBuf.appendAscii("The configuration directory");
_rBuf.appendAscii(" '").append(_aPath).appendAscii("' ");
_rBuf.appendAscii(IS_MISSING).appendAscii(PERIOD);
}
// ---------------------------------------------------------------------------------------
2001-08-09 11:22:27 +00:00
static void addUnexpectedError(OUStringBuffer& _rBuf, AsciiString _sExtraInfo = NULL)
{
2001-08-09 11:22:27 +00:00
if (NULL == _sExtraInfo)
_sExtraInfo = "An internal failure occurred";
_rBuf.appendAscii(_sExtraInfo).appendAscii(PERIOD);
}
// ---------------------------------------------------------------------------------------
static Bootstrap::FailureCode describeError(OUStringBuffer& _rBuf, Bootstrap::Impl const& _rData)
{
Bootstrap::FailureCode eErrCode = Bootstrap::INVALID_BOOTSTRAP_DATA;
2001-08-09 11:22:27 +00:00
_rBuf.appendAscii("The program cannot be started. ");
switch (_rData.aUserInstall_.status)
{
case Bootstrap::PATH_EXISTS:
switch (_rData.aBaseInstall_.status)
{
case Bootstrap::PATH_VALID:
2001-08-09 11:22:27 +00:00
addMissingDirectoryError(_rBuf, _rData.aBaseInstall_.path);
eErrCode = Bootstrap::MISSING_INSTALL_DIRECTORY;
break;
case Bootstrap::DATA_INVALID:
2001-08-09 11:22:27 +00:00
addUnexpectedError(_rBuf,"The installation path is invalid");
break;
case Bootstrap::DATA_MISSING:
2001-08-09 11:22:27 +00:00
addUnexpectedError(_rBuf,"The installation path is not available");
break;
case Bootstrap::PATH_EXISTS: // seems to be all fine (?)
2001-08-09 11:22:27 +00:00
addUnexpectedError(_rBuf,"");
break;
default: OSL_ASSERT(false);
2001-08-09 11:22:27 +00:00
addUnexpectedError(_rBuf);
break;
}
break;
case Bootstrap::PATH_VALID:
2001-08-09 11:22:27 +00:00
addMissingDirectoryError(_rBuf, _rData.aUserInstall_.path);
eErrCode = Bootstrap::MISSING_USER_DIRECTORY;
break;
// else fall through
case Bootstrap::DATA_INVALID:
if (_rData.aVersionINI_.status == Bootstrap::PATH_EXISTS)
{
2001-08-09 11:22:27 +00:00
addFileError(_rBuf, _rData.aVersionINI_.path, IS_INVALID);
eErrCode = Bootstrap::INVALID_VERSION_FILE_ENTRY;
break;
}
// else fall through
case Bootstrap::DATA_MISSING:
switch (_rData.aVersionINI_.status)
{
case Bootstrap::PATH_EXISTS:
2001-08-09 11:22:27 +00:00
addFileError(_rBuf, _rData.aVersionINI_.path, "does not support the current version");
eErrCode = Bootstrap::MISSING_VERSION_FILE_ENTRY;
break;
case Bootstrap::PATH_VALID:
2001-08-09 11:22:27 +00:00
addFileError(_rBuf, _rData.aVersionINI_.path, IS_MISSING);
eErrCode = Bootstrap::MISSING_VERSION_FILE;
break;
default:
switch (_rData.aBootstrapINI_.status)
{
case Bootstrap::PATH_EXISTS:
2001-08-09 11:22:27 +00:00
addFileError(_rBuf, _rData.aBootstrapINI_.path, IS_INVALID);
if (_rData.aVersionINI_.status == Bootstrap::DATA_MISSING)
eErrCode = Bootstrap::MISSING_BOOTSTRAP_FILE_ENTRY;
else
eErrCode = Bootstrap::INVALID_BOOTSTRAP_FILE_ENTRY;
break;
case Bootstrap::DATA_INVALID: OSL_ASSERT(false);
case Bootstrap::PATH_VALID:
2001-08-09 11:22:27 +00:00
addFileError(_rBuf, _rData.aBootstrapINI_.path, IS_MISSING);
eErrCode = Bootstrap::MISSING_BOOTSTRAP_FILE;
break;
default:
2001-08-09 11:22:27 +00:00
addUnexpectedError(_rBuf);
break;
}
break;
}
break;
default: OSL_ASSERT(false);
2001-08-09 11:22:27 +00:00
addUnexpectedError(_rBuf);
break;
}
return eErrCode;
}
// ---------------------------------------------------------------------------------------
// ---------------------------------------------------------------------------------------
// class Bootstrap
// ---------------------------------------------------------------------------------------
OUString Bootstrap::getProductKey()
{
Impl const& aData = data();
OUString const& sDefaultProductKey = aData.sExename_;
//OUString const sDefaultProductKey(RTL_CONSTASCII_USTRINGPARAM(BOOTSTRAP_VALUE_PRODUCT_KEY_DEFAULT));
return aData.getBootstrapValue( BOOTSTRAP_ITEM_PRODUCT_KEY, sDefaultProductKey );
}
// ---------------------------------------------------------------------------------------
OUString Bootstrap::getProductKey(OUString const& _sDefault)
{
Impl const& aData = data();
return aData.getBootstrapValue( BOOTSTRAP_ITEM_PRODUCT_KEY, _sDefault );
}
// ---------------------------------------------------------------------------------------
OUString Bootstrap::getLogoData(OUString const& _sDefault)
{
Impl const& aData = data();
return aData.getBootstrapValue( BOOTSTRAP_ITEM_LOGO, _sDefault );
}
// ---------------------------------------------------------------------------------------
Bootstrap::PathStatus Bootstrap::locateBaseInstallation(OUString& _rURL)
{
Impl::PathData const& aPathData = data().aBaseInstall_;
_rURL = aPathData.path;
return aPathData.status;
}
// ---------------------------------------------------------------------------------------
PathStatus Bootstrap::locateUserInstallation(OUString& _rURL)
{
Impl::PathData const& aPathData = data().aUserInstall_;
_rURL = aPathData.path;
return aPathData.status;
}
// ---------------------------------------------------------------------------------------
PathStatus Bootstrap::locateSharedData(OUString& _rURL)
{
OUString const aShareDir(RTL_CONSTASCII_USTRINGPARAM("share"));
return getDerivedPath(_rURL,aShareDir,data().aBaseInstall_);
}
// ---------------------------------------------------------------------------------------
PathStatus Bootstrap::locateUserData(OUString& _rURL)
{
OUString const aUserDir(RTL_CONSTASCII_USTRINGPARAM("user"));
return getDerivedPath(_rURL,aUserDir,data().aUserInstall_);
}
// ---------------------------------------------------------------------------------------
PathStatus Bootstrap::locateBootstrapFile(OUString& _rURL)
{
Impl::PathData const& aPathData = data().aBootstrapINI_;
_rURL = aPathData.path;
return aPathData.status;
}
// ---------------------------------------------------------------------------------------
PathStatus Bootstrap::locateVersionFile(OUString& _rURL)
{
Impl::PathData const& aPathData = data().aVersionINI_;
_rURL = aPathData.path;
return aPathData.status;
}
// ---------------------------------------------------------------------------------------
Bootstrap::Status Bootstrap::checkBootstrapStatus(OUString& _rDiagnosticMessage)
{
FailureCode eDummyCode(NO_FAILURE);
return checkBootstrapStatus(_rDiagnosticMessage,eDummyCode);
}
// ---------------------------------------------------------------------------------------
Bootstrap::Status Bootstrap::checkBootstrapStatus(rtl::OUString& _rDiagnosticMessage, FailureCode& _rErrCode)
{
Impl const& aData = data();
Status result = aData.status_;
// maybe do further checks here
OUStringBuffer sErrorBuffer;
if (result != DATA_OK)
_rErrCode = describeError(sErrorBuffer,aData);
else
_rErrCode = NO_FAILURE;
_rDiagnosticMessage = sErrorBuffer.makeStringAndClear();
return result;
}
// ---------------------------------------------------------------------------------------
// class Bootstrap::Impl
// ---------------------------------------------------------------------------------------
bool Bootstrap::Impl::initBaseInstallationData()
{
OUString sExecutable;
if (osl_Process_E_None != osl_getExecutableFile(&sExecutable.pData))
{
aBaseInstall_.status = DATA_MISSING;
return false;
}
// split the executable name
sal_Int32 nSepIndex = sExecutable.lastIndexOf(cURLSeparator);
sExename_ = sExecutable.copy(nSepIndex + 1);
/// ... and get the basename (strip the extension)
{
sal_Unicode cExtensionSep = '.';
sal_Int32 const nExtIndex = sExename_.lastIndexOf(cExtensionSep);
sal_Int32 const nExtLength = sExename_.getLength() - nExtIndex - 1;
if (0 < nExtIndex && nExtLength < 4)
sExename_ = sExename_.copy(0,nExtIndex);
}
OSL_ENSURE(nSepIndex > 0, "osl_getExecutableFile returns local path only");
if (nSepIndex <= 0)
{
aBaseInstall_.status = DATA_INVALID;
return false;
}
// reduce to executable directory
sExecutable = sExecutable.copy(0,nSepIndex);
// ... and split to find installation directory
nSepIndex = sExecutable.lastIndexOf(cURLSeparator);
OSL_ENSURE(nSepIndex > 0, "osl_getExecutableFile returns relative path only");
if (nSepIndex <= 0)
{
aBaseInstall_.status = DATA_INVALID;
return false;
}
aBaseInstall_.path = sExecutable.copy(0,nSepIndex);
aBaseInstall_.status = PATH_EXISTS;
OSL_ASSERT( dbgCheckStatusOfURL(sExecutable) == PATH_EXISTS );
OSL_ASSERT( dbgCheckStatusOfURL(aBaseInstall_.path) == PATH_EXISTS );
// now find the bootstrap ini
OUString const sBootstrapName(RTL_CONSTASCII_USTRINGPARAM(BOOTSTRAP_FILE));
aBootstrapINI_.status = getDerivedPath(aBootstrapINI_.path,sBootstrapName,sExecutable,PATH_EXISTS);
return true;
}
// ---------------------------------------------------------------------------------------
bool Bootstrap::Impl::initUserInstallationData()
{
OString sVersionFileSection(BOOTSTRAP_VALUE_SECTION_DEFAULT);
//OString sVersionFileProductKey(BOOTSTRAP_VALUE_PRODUCT_KEY_DEFAULT);
OString sVersionFileProductKey = rtl::OUStringToOString(
this->getBootstrapValue(BOOTSTRAP_ITEM_PRODUCT_KEY,sExename_),
RTL_TEXTENCODING_ASCII_US);
// try to find a sversion ini
if (aBootstrapINI_.status == PATH_EXISTS)
{
// read the bootstrap file
osl::Profile aBootstrapFile(aBootstrapINI_.path);
OString const sSection(BOOTSTRAP_SECTION);
// get the section/key name of the entry in the product version file
sVersionFileSection = aBootstrapFile.readString(sSection,BOOTSTRAP_ITEM_SECTION,sVersionFileSection);
sVersionFileProductKey = aBootstrapFile.readString(sSection,BOOTSTRAP_ITEM_PRODUCT_KEY,sVersionFileProductKey);
readPathFromINI(aVersionINI_, aBootstrapFile, sSection,BOOTSTRAP_ITEM_LOCATION);
}
// try to find a user installation
switch (aVersionINI_.status)
{
case PATH_EXISTS: // sversion ini found
{
// open the product version profile
osl::Profile aProductVersionFile(aVersionINI_.path);
readPathFromINI(aUserInstall_, aProductVersionFile, sVersionFileSection, sVersionFileProductKey);
}
break;
case PATH_VALID: // sversion ini location invalid or file missing
case DATA_INVALID:
aUserInstall_.status = DATA_MISSING;
break;
default: OSL_ASSERT(!"Unreachable code");
case DATA_UNKNOWN: // no bootstrap ini
case DATA_MISSING: // no sversion ini location
{
// should we do this - look for a single-user user directory ?
OUString const aUserDir(RTL_CONSTASCII_USTRINGPARAM("user"));
OUString sTestUserDir;
if ( getDerivedPath(sTestUserDir,aUserDir,aBaseInstall_) )
aUserInstall_ = aBaseInstall_;
else
// or just this - distinguishing this from
aUserInstall_.status = DATA_MISSING;
}
break;
}
return aUserInstall_.status == PATH_EXISTS;
}
// ---------------------------------------------------------------------------------------
Bootstrap::Status Bootstrap::Impl::initialize()
{
Bootstrap::Status result;
if (!initBaseInstallationData())
{
result = INVALID_BASE_INSTALL;
}
else if (!initUserInstallationData())
{
result = INVALID_USER_INSTALL;
if (aUserInstall_.status >= DATA_MISSING)
{
switch (aVersionINI_.status)
{
case PATH_EXISTS:
case PATH_VALID:
result = MISSING_USER_INSTALL;
break;
case DATA_INVALID:
case DATA_MISSING:
result = INVALID_BASE_INSTALL;
break;
}
}
}
else
{
result = DATA_OK;
}
return result;
}
// ---------------------------------------------------------------------------------------
OUString Bootstrap::Impl::getBootstrapValue(AsciiString _sName, OUString const& _sDefault) const
{
if (aBootstrapINI_.status != PATH_EXISTS) return _sDefault;
osl::Profile aBootstrapFile(aBootstrapINI_.path);
// get the path to the sversion.ini
OString sConvertable = aBootstrapFile.readString( BOOTSTRAP_SECTION, _sName, OString());
return rtl::OStringToOUString(sConvertable, RTL_TEXTENCODING_ASCII_US);
}
// ---------------------------------------------------------------------------------------
} // namespace utl