office-gobmx/l10ntools/source/merge.cxx
Noel Grandin b9c806b9e3 address potential find/rfind failures
in the conversion in
    commit 74957c7d2f
    Author: Noel Grandin <noel.grandin@collabora.co.uk>
    Date:   Wed Apr 13 13:54:22 2022 +0200
    use more string_view in l10ntools

where the prior code might have been relying on the -1 returned by
indexOf and lastIndexOd

Change-Id: Ief5dedccbaf4e14e5f59aa3c2f7481ff0bb7e2e8
Reviewed-on: https://gerrit.libreoffice.org/c/core/+/133027
Tested-by: Jenkins
Reviewed-by: Stephan Bergmann <sbergman@redhat.com>
2022-04-15 14:06:06 +02:00

343 lines
10 KiB
C++

/* -*- 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 <sal/config.h>
#include <sal/log.hxx>
#include <algorithm>
#include <cstdlib>
#include <fstream>
#include <iostream>
#include <string>
#include <vector>
#include <export.hxx>
#include <po.hxx>
namespace
{
OString lcl_NormalizeFilename(std::string_view rFilename)
{
size_t idx1 = rFilename.rfind( '\\' );
size_t idx2 = rFilename.rfind( '/' );
if (idx1 == std::string_view::npos && idx2 == std::string_view::npos)
return OString(rFilename);
if (idx1 == std::string_view::npos)
idx1 = 0;
if (idx2 == std::string_view::npos)
idx2 = 0;
return OString(rFilename.substr(std::max(idx1, idx2)+1));
};
bool lcl_ReadPoChecked(
PoEntry& o_rPoEntry, PoIfstream& rPoFile,
const OString& rFileName)
{
try
{
rPoFile.readEntry( o_rPoEntry );
}
catch (const PoIfstream::Exception&)
{
SAL_WARN("l10ntools", rFileName << " contains invalid entry");
return false;
}
return true;
}
}
ResData::ResData( const OString &rGId )
:
sGId( rGId )
{
sGId = sGId.replaceAll("\r", OString());
}
ResData::ResData( const OString &rGId, const OString &rFilename)
:
sGId( rGId ),
sFilename( rFilename )
{
sGId = sGId.replaceAll("\r", OString());
}
bool MergeEntrys::GetText( OString &rReturn,
const OString &nLangIndex, bool bDel )
{
bool bReturn = true;
rReturn = sText[ nLangIndex ];
if ( bDel )
sText[ nLangIndex ] = "";
bReturn = bTextFirst[ nLangIndex ];
bTextFirst[ nLangIndex ] = false;
return bReturn;
}
namespace
{
OString GetDoubleBars()
{
//DOUBLE VERTICAL LINE instead of || because the translations make their
//way into action_names under gtk3 where || is illegal
return u8"\u2016";
}
}
OString MergeEntrys::GetQTZText(const ResData& rResData, std::string_view rOrigText)
{
const OString sFilename = rResData.sFilename.copy(rResData.sFilename.lastIndexOf('/')+1);
const OString sKey =
PoEntry::genKeyId(sFilename + rResData.sGId + rResData.sId + rResData.sResTyp + rOrigText);
return sKey + GetDoubleBars() + rOrigText;
}
MergeDataFile::MergeDataFile(
const OString &rFileName, std::string_view rFile,
bool bCaseSensitive, bool bWithQtz )
{
auto const env = getenv("ENABLE_RELEASE_BUILD");
OString sEnableReleaseBuild(env == nullptr ? "" : env);
std::ifstream aInputStream( rFileName.getStr() );
if ( !aInputStream.is_open() )
{
SAL_WARN("l10ntools", "Can't open po path container file for " << rFileName);
return;
}
std::string sPoFile;
aInputStream >> sPoFile;
bool bFirstLang = true;
while( !aInputStream.eof() )
{
bool bSkipCurrentPOFile = false;
const OString sFileName( lcl_NormalizeFilename(rFile) );
const bool bReadAll = sFileName.isEmpty();
// coverity[tainted_data] - this is a build time tool
const OString sPoFileName(sPoFile.data(), static_cast<sal_Int32>(sPoFile.length()));
PoIfstream aPoInput;
aPoInput.open( sPoFileName );
if ( !aPoInput.isOpen() )
{
SAL_WARN("l10ntools", "Can't open file: " << sPoFileName);
return;
}
OString sLang;
//Get language id from path
{
const OString sTransSource("translations/source/");
const sal_Int32 nStart =
sPoFileName.indexOf(sTransSource)+sTransSource.getLength();
const sal_Int32 nCount =
sPoFileName.indexOf('/',nStart) - nStart;
sLang = sPoFileName.copy(nStart,nCount);
}
aLanguageSet.insert( sLang );
PoEntry aNextPo;
do
{
if( !lcl_ReadPoChecked(aNextPo, aPoInput, sPoFileName) )
{
bSkipCurrentPOFile = true;
break;
}
} while( !aPoInput.eof() && aNextPo.getSourceFile() != sFileName && !bReadAll );
while( !aPoInput.eof() && (aNextPo.getSourceFile() == sFileName || bReadAll ) && !bSkipCurrentPOFile )
{
PoEntry aActPo( aNextPo );
bool bInSameComp = false;
OString sText;
OString sQHText;
OString sTitle;
OString sExText;
OString sExQHText;
OString sExTitle;
do
{
if( bInSameComp )
aActPo = aNextPo;
OString sTemp = aActPo.getMsgStr();
if( aActPo.isFuzzy() || sTemp.isEmpty() )
sTemp = aActPo.getMsgId();
switch( aActPo.getType() )
{
case PoEntry::TTEXT:
sText = sTemp;
sExText = aActPo.getMsgId();
break;
case PoEntry::TQUICKHELPTEXT:
sQHText = sTemp;
sExQHText = aActPo.getMsgId();
break;
case PoEntry::TTITLE:
sTitle = sTemp;
sExTitle = aActPo.getMsgId();
break;
}
if( !lcl_ReadPoChecked(aNextPo, aPoInput, sPoFileName) )
{
bSkipCurrentPOFile = true;
break;
}
if (aPoInput.eof())
break;
bInSameComp = PoEntry::IsInSameComp(aActPo, aNextPo);
} while( bInSameComp );
InsertEntry(
aActPo.getResourceType(), aActPo.getGroupId(),
aActPo.getLocalId(), sLang, sText,
sQHText, sTitle, aActPo.getSourceFile(),
bFirstLang, bCaseSensitive );
if( bFirstLang && bWithQtz &&
sEnableReleaseBuild != "TRUE" )
{
aLanguageSet.insert("qtz");
InsertEntry(
aActPo.getResourceType(), aActPo.getGroupId(),
aActPo.getLocalId(), "qtz",
sExText, sExQHText,
sExTitle, aActPo.getSourceFile(),
false, bCaseSensitive );
}
}
aPoInput.close();
aInputStream >> sPoFile;
bFirstLang = false;
}
aInputStream.close();
}
MergeDataFile::~MergeDataFile()
{
}
std::vector<OString> MergeDataFile::GetLanguages() const
{
return std::vector<OString>(aLanguageSet.begin(),aLanguageSet.end());
}
MergeEntrys *MergeDataFile::GetMergeData( ResData *pResData , bool bCaseSensitive )
{
OString sOldG = pResData->sGId;
OString sOldL = pResData->sId;
OString sGID = pResData->sGId;
OString sLID;
if (sGID.isEmpty())
sGID = pResData->sId;
else
sLID = pResData->sId;
pResData->sGId = sGID;
pResData->sId = sLID;
OString sKey = CreateKey( pResData->sResTyp , pResData->sGId , pResData->sId , pResData->sFilename , bCaseSensitive );
auto mit = aMap.find( sKey );
if(mit != aMap.end())
{
pResData->sGId = sOldG;
pResData->sId = sOldL;
return mit->second.get();
}
pResData->sGId = sOldG;
pResData->sId = sOldL;
return nullptr;
}
MergeEntrys *MergeDataFile::GetMergeEntrys( ResData *pResData )
{
// search for requested MergeEntrys
return GetMergeData( pResData );
}
MergeEntrys *MergeDataFile::GetMergeEntrysCaseSensitive( ResData *pResData )
{
// search for requested MergeEntrys
return GetMergeData( pResData , true );
}
void MergeDataFile::InsertEntry(
std::string_view rTYP, std::string_view rGID,
std::string_view rLID, const OString &nLANG,
const OString &rTEXT, const OString &rQHTEXT,
const OString &rTITLE, std::string_view rInFilename,
bool bFirstLang, bool bCaseSensitive )
{
MergeEntrys *pMergeEntrys = nullptr;
// search for MergeData
OString sKey = CreateKey(rTYP , rGID , rLID , rInFilename , bCaseSensitive);
if( !bFirstLang )
{
auto mit = aMap.find( sKey );
if(mit != aMap.end())
pMergeEntrys = mit->second.get();
}
if( !pMergeEntrys )
{
pMergeEntrys = new MergeEntrys;
if (!aMap.emplace( sKey, std::unique_ptr<MergeEntrys>(pMergeEntrys) ).second)
{
std::cerr << "Duplicate entry " << sKey << "\n";
std::exit(EXIT_FAILURE);
}
}
// insert the cur string
if( nLANG =="qtz" )
{
const OString sTemp = OString::Concat(rInFilename) + rGID + rLID + rTYP;
pMergeEntrys->InsertEntry(
nLANG,
rTEXT.isEmpty()? rTEXT : PoEntry::genKeyId(sTemp + rTEXT) + GetDoubleBars() + rTEXT,
rQHTEXT.isEmpty()? rQHTEXT : PoEntry::genKeyId(sTemp + rQHTEXT) + GetDoubleBars() + rQHTEXT,
rTITLE.isEmpty()? rTITLE : PoEntry::genKeyId(sTemp + rTITLE) + GetDoubleBars() + rTITLE );
}
else
{
pMergeEntrys->InsertEntry( nLANG , rTEXT, rQHTEXT, rTITLE );
}
}
OString MergeDataFile::CreateKey(std::string_view rTYP, std::string_view rGID,
std::string_view rLID, std::string_view rFilename, bool bCaseSensitive)
{
static const char sStroke[] = "-";
OString sKey = OString::Concat(rTYP) + sStroke + rGID + sStroke + rLID + sStroke +
lcl_NormalizeFilename(rFilename);
if(bCaseSensitive)
return sKey; // officecfg case sensitive identifier
return sKey.toAsciiUpperCase();
}
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */