19#include <zypp-core/base/DefaultIntegral>
23#include <zypp-common/PublicKey.h>
35#include <zypp-core/base/InputStream>
61 if (
PathInfo(path_r/
"/repodata/repomd.xml").isFile() )
63 else if (
PathInfo(path_r/
"/content").isFile() )
65 else if (
PathInfo(path_r/
"/cookie").isFile() )
68 DBG <<
"Probed cached type " << ret <<
" at " << path_r << endl;
125 std::string licenseStem(
"license" );
126 if ( !name_r.empty() )
129 licenseStem += name_r;
177 return std::string();
178 return (*
baseUrls().transformedBegin()).asString();
211 [
this](
int num_r,
const std::string& line_r )->
bool
215 std::vector<std::string> words;
216 if (
str::split( line_r, std::back_inserter(words) ) > 1
217 && words[0].length() == 12 )
219 this->
_keywords.second.insert( ++words.begin(), words.end() );
258 if (
PathInfo(gpgcheckFile).isExist() )
283 static const Pathname truePath(
"true" );
284 static const Pathname falsePath(
"false" );
285 static const Pathname indeterminatePath(
"indeterminate" );
288 static const ssize_t bufsiz = 63;
289 static char buf[bufsiz+1];
290 ssize_t ret = ::readlink( path_r.
c_str(), buf, bufsiz );
291 buf[ret == -1 ? 0 : ret] =
'\0';
296 if ( linkval == truePath )
298 else if ( linkval == falsePath )
300 else if ( linkval == indeterminatePath )
380 mutable std::pair<FalseBool, std::set<std::string> >
_keywords;
387 {
return new Impl( *
this ); }
394 return str <<
"RepoInfo::Impl";
413 {
return _pimpl->priority; }
426 {
return _pimpl->cfgGpgCheck(); }
429 {
_pimpl->rawGpgCheck( value_r ); }
441 if ( ret &&
_pimpl->internalUnsignedConfirmed() )
447 {
_pimpl->rawRepoGpgCheck( value_r ); }
457 {
_pimpl->rawPkgGpgCheck( value_r ); }
462 g_r =
_pimpl->rawGpgCheck();
463 r_r =
_pimpl->rawRepoGpgCheck();
464 p_r =
_pimpl->rawPkgGpgCheck();
476 {
_pimpl->internalSetValidRepoSignature( value_r ); }
482 {
if ( ! sameTriboolState( lhs, rhs ) ) { lhs = rhs;
return true; }
return false; }
486 bool changed =
false;
487 if ( changeGpgCheckTo( ogpg[0], g ) ) changed =
true;
488 if ( changeGpgCheckTo( ogpg[1], r ) ) changed =
true;
489 if ( changeGpgCheckTo( ogpg[2], p ) ) changed =
true;
499 bool changed =
false;
506 changed = changeGpgCheckTo( ogpg,
true,
true,
true );
509 changed = changeGpgCheckTo( ogpg,
true,
false,
false );
512 changed = changeGpgCheckTo( ogpg,
true,
false,
indeterminate );
515 changed = changeGpgCheckTo( ogpg,
true,
indeterminate,
false );
537 {
_pimpl->_mirrorListUrl.raw() = url_r;
_pimpl->_mirrorListForceMetalink =
false; }
543 {
_pimpl->_mirrorListUrl.raw() = url_r;
_pimpl->_mirrorListForceMetalink =
true; }
549 {
_pimpl->gpgKeyUrls().raw().swap( urls ); }
553 _pimpl->gpgKeyUrls().raw().clear();
554 _pimpl->gpgKeyUrls().raw().push_back( url_r );
558 {
return _pimpl->repoStatusString(); }
566 for (
const auto &
url :
_pimpl->baseUrls().raw() )
569 _pimpl->baseUrls().raw().push_back( std::move(url_r) );
574 _pimpl->baseUrls().raw().clear();
575 _pimpl->baseUrls().raw().push_back( std::move(url_r) );
579 {
_pimpl->baseUrls().raw().swap( urls ); }
588 {
_pimpl->setProbedType( t ); }
598 {
return _pimpl->predownloadPath(); }
601 {
_pimpl->keeppackages = keep; }
616 {
return _pimpl->metadataPath(); }
619 {
return _pimpl->packagesPath(); }
622 {
return _pimpl->usesAutoMetadataPaths(); }
625 {
return _pimpl->type(); }
628 {
return _pimpl->_mirrorListUrl.transformed(); }
631 {
return _pimpl->_mirrorListUrl.raw(); }
634 {
return _pimpl->gpgKeyUrls().empty(); }
637 {
return _pimpl->gpgKeyUrls().size(); }
640 {
return _pimpl->gpgKeyUrls().transformed(); }
643 {
return _pimpl->gpgKeyUrls().raw(); }
646 {
return(
_pimpl->gpgKeyUrls().empty() ?
Url() : *
_pimpl->gpgKeyUrls().transformedBegin() ); }
649 {
return(
_pimpl->gpgKeyUrls().empty() ?
Url() : *
_pimpl->gpgKeyUrls().rawBegin() ) ; }
652 {
return _pimpl->baseUrls().transformed(); }
655 {
return _pimpl->baseUrls().raw(); }
661 {
return _pimpl->service; }
664 {
return _pimpl->targetDistro; }
667 {
return(
_pimpl->baseUrls().empty() ?
Url() : *
_pimpl->baseUrls().rawBegin() ); }
670 {
return _pimpl->baseUrls().transformedBegin(); }
673 {
return _pimpl->baseUrls().transformedEnd(); }
676 {
return _pimpl->baseUrls().size(); }
679 {
return _pimpl->baseUrls().empty(); }
682 {
return _pimpl->baseurl2dump(); }
685 {
return _pimpl->contentKeywords(); }
688 {
_pimpl->addContent( keyword_r ); }
691 {
return _pimpl->hasContent(); }
694 {
return _pimpl->hasContent( keyword_r ); }
702 {
return !
_pimpl->licenseTgz( name_r ).empty(); }
711 if ( licenseTgz.
empty() )
715 cmd.push_back(
"tar" );
716 cmd.push_back(
"-t" );
717 cmd.push_back(
"-z" );
718 cmd.push_back(
"-f" );
719 cmd.push_back( licenseTgz.
asString() );
723 static const std::string noAcceptanceFile =
"no-acceptance-needed\n";
726 if ( output == noAcceptanceFile )
732 MIL <<
"License(" << name_r <<
") in " <<
name() <<
" has to be accepted: " << (accept?
"true":
"false" ) << endl;
741 {
return getLicense( std::string(), lang_r ); }
746 if ( avlocales.empty() )
747 return std::string();
750 if ( !getLang && avlocales.find(
Locale::noCode ) == avlocales.end() )
752 WAR <<
"License(" << name_r <<
") in " <<
name() <<
" contains no fallback text!" << endl;
756 getLang = *avlocales.begin();
760 static const std::string licenseFileFallback(
"license.txt" );
761 std::string licenseFile( !getLang ? licenseFileFallback
765 cmd.push_back(
"tar" );
766 cmd.push_back(
"-x" );
767 cmd.push_back(
"-z" );
768 cmd.push_back(
"-O" );
769 cmd.push_back(
"-f" );
770 cmd.push_back(
_pimpl->licenseTgz( name_r ).asString() );
771 cmd.push_back( licenseFile );
790 if ( licenseTgz.
empty() )
794 cmd.push_back(
"tar" );
795 cmd.push_back(
"-t" );
796 cmd.push_back(
"-z" );
797 cmd.push_back(
"-f" );
798 cmd.push_back( licenseTgz.
asString() );
804 static const C_Str license(
"license." );
805 static const C_Str dotTxt(
".txt\n" );
808 if ( output.size() <= license.
size() + dotTxt.
size() )
811 ret.insert(
Locale( std::string( output.c_str()+license.
size(), output.size()- license.
size() - dotTxt.
size() ) ) );
822 RepoInfoBase::dumpOn(
str);
823 if (
_pimpl->baseurl2dump() )
825 for (
const auto &
url :
_pimpl->baseUrls().raw() )
827 str <<
"- url : " <<
url << std::endl;
832 auto strif( [&] (
const std::string & tag_r,
const std::string & value_r ) {
833 if ( ! value_r.empty() )
834 str << tag_r << value_r << std::endl;
839 str <<
"- type : " <<
type() << std::endl;
843#define OUTS(T,B) ( indeterminate(T) ? (std::string("D(")+(B?"Y":"N")+")") : ((bool)T?"Y":"N") )
851 for (
const auto &
url :
_pimpl->gpgKeyUrls().raw() )
853 str <<
"- gpgkey : " <<
url << std::endl;
859 strif(
"- service : ",
service() );
870 RepoInfoBase::dumpAsIniOn(
str);
872 if (
_pimpl->baseurl2dump() )
876 for (
const auto &
url :
_pimpl->baseUrls().raw() )
879 if ( indent.empty() ) indent =
" ";
883 if ( !
_pimpl->path.empty() )
884 str <<
"path="<<
path() << endl;
896 str <<
"gpgcheck=" << (
_pimpl->rawGpgCheck() ?
"1" :
"0") << endl;
899 str <<
"repo_gpgcheck=" << (
_pimpl->rawRepoGpgCheck() ?
"1" :
"0") << endl;
902 str <<
"pkg_gpgcheck=" << (
_pimpl->rawPkgGpgCheck() ?
"1" :
"0") << endl;
905 std::string indent(
"gpgkey=");
906 for (
const auto &
url :
_pimpl->gpgKeyUrls().raw() )
908 str << indent <<
url << endl;
909 if ( indent[0] !=
' ' )
933 <<
" priority=\"" <<
priority() <<
"\""
934 <<
" enabled=\"" <<
enabled() <<
"\""
936 <<
" gpgcheck=\"" <<
gpgCheck() <<
"\""
940 str <<
" raw_gpgcheck=\"" << (
_pimpl->rawGpgCheck() ?
"1" :
"0") <<
"\"";
942 str <<
" raw_repo_gpgcheck=\"" << (
_pimpl->rawRepoGpgCheck() ?
"1" :
"0") <<
"\"";
944 str <<
" raw_pkg_gpgcheck=\"" << (
_pimpl->rawPkgGpgCheck() ?
"1" :
"0") <<
"\"";
947 str <<
" gpgkey=\"" <<
escape(tmpstr) <<
"\"";
949 str << (
_pimpl->_mirrorListForceMetalink ?
" metalink=\"" :
" mirrorlist=\"") <<
escape(tmpstr) <<
"\"";
952 if (
_pimpl->baseurl2dump() )
955 str <<
"<url>" <<
escape((*it).asString()) <<
"</url>" << endl;
958 str <<
"</repo>" << endl;
972#define OUTS( V ) case RepoInfo::V: return str << #V; break
973 OUTS( GpgCheck::On );
974 OUTS( GpgCheck::Strict );
975 OUTS( GpgCheck::AllowUnsigned );
976 OUTS( GpgCheck::AllowUnsignedRepo );
977 OUTS( GpgCheck::AllowUnsignedPackage );
978 OUTS( GpgCheck::Default );
979 OUTS( GpgCheck::Off );
980 OUTS( GpgCheck::indeterminate );
983 return str <<
"GpgCheck::UNKNOWN";
991 if ( canSkipMediaCheck ) {
993 if ( not mDataPath.empty() ) {
994 PathInfo mediafile { mDataPath/
"media.1/media" };
995 if ( mediafile.isExist() ) {
997 if ( lverifier && lverifier.totalMedia() > 1 ) {
998 canSkipMediaCheck =
false;
1003 if ( canSkipMediaCheck )
1004 DBG <<
"Can SKIP media.1/media check for status calc of repo " <<
alias() << endl;
1005 return not canSkipMediaCheck;
ZYPP_API detail::EscapedString escape(const std::string &in_r)
Escape xml special charaters (& -> &; from IoBind library).
base::ValueTransform< Url, repo::RepoVariablesUrlReplacer > RepoVariablesReplacedUrl
Helper managing repo variables replaced urls.
base::ContainerTransform< std::list< Url >, repo::RepoVariablesUrlReplacer > RepoVariablesReplacedUrlList
Helper managing repo variables replaced url lists.
Convenience char* constructible from std::string and char*, it maps (char*)0 to an empty string.
Integral type with defined initial value when default constructed.
Execute a program and give access to its io An object of this class encapsulates the execution of an ...
int close() override
Wait for the progamm to complete.
std::vector< std::string > Arguments
const char * c_str() const
'Language[_Country]' codes.
static const Locale noCode
Empty code.
static Locale bestMatch(const LocaleSet &avLocales_r, Locale requested_r=Locale())
Return the best match for Locale requested_r within the available avLocales_r.
void setPkgGpgCheck(TriBool value_r)
Set the value for pkgGpgCheck (or indeterminate to use the default).
std::ostream & dumpAsXmlOn(std::ostream &str, const std::string &content="") const override
Write an XML representation of this RepoInfo object.
void setGpgKeyUrls(url_set urls)
Set a list of gpgkey URLs defined for this repo.
void setMirrorListUrls(url_set urls)
Like setMirrorListUrl but take an url_set.
void setMetalinkUrl(const Url &url)
Like setMirrorListUrl but expect metalink format.
std::ostream & dumpOn(std::ostream &str) const override
Write a human-readable representation of this RepoInfo object into the str stream.
Pathname metadataPath() const
Path where this repo metadata was read from.
url_set::size_type urls_size_type
void setGpgKeyUrl(const Url &gpgkey)
(leagcy API) Set the gpgkey URL defined for this repo
bool usesAutoMetadataPaths() const
Whether metadataPath uses AUTO% setup.
GpgCheck
Some predefined settings.
bool baseUrlsEmpty() const
whether repository urls are available
Pathname predownloadPath() const
Path where this repo packages are predownloaded.
bool hasContent() const
Check for content keywords.
void setKeepPackages(bool keep)
Set if packaqes downloaded from this repository will be kept in local cache.
url_set gpgKeyUrls() const
The list of gpgkey URLs defined for this repo.
Url rawGpgKeyUrl() const
(leagcy API) The 1st raw gpgkey URL defined for this repo (no variables replaced)
transform_iterator< repo::RepoVariablesUrlReplacer, url_set::const_iterator > urls_const_iterator
void setMirrorListUrl(const Url &url)
Set mirror list url.
Url rawUrl() const
Pars pro toto: The first repository raw url (no variables replaced)
Pathname provideKey(const std::string &keyID_r, const Pathname &targetDirectory_r) const
downloads all configured gpg keys into the defined directory
repo::RepoType type() const
Type of repository,.
url_set rawGpgKeyUrls() const
The list of raw gpgkey URLs defined for this repo (no variables replaced)
static unsigned noPriority()
The least priority (unsigned(-1)).
urls_size_type baseUrlsSize() const
number of repository urls
bool keepPackages() const
Whether packages downloaded from this repository will be kept in local cache.
Url url() const
Pars pro toto: The first repository url.
static const RepoInfo noRepo
Represents no Repository (one with an empty alias).
void setBaseUrl(Url url)
Clears current base URL list and adds url.
const std::set< std::string > & contentKeywords() const
Content keywords defined.
urls_const_iterator baseUrlsEnd() const
iterator that points at end of repository urls
void setPackagesPath(const Pathname &path)
set the path where the local packages are stored
std::string getLicense(const Locale &lang_r=Locale()) const
Return the best license for the current (or a specified) locale.
bool baseUrlSet() const
Whether there are manualy configured repository urls.
void setService(const std::string &name)
sets service which added this repository
void setGpgCheck(TriBool value_r)
Set the value for gpgCheck (or indeterminate to use the default).
bool effectiveKeepPackages() const
keepPackages unless the package cache itself enforces keeping the packages.
Pathname path() const
Repository path.
urls_size_type gpgKeyUrlsSize() const
Number of gpgkey URLs defined.
LocaleSet getLicenseLocales() const
Return the locales the license is available for.
url_set baseUrls() const
The complete set of repository urls.
bool requireStatusWithMediaFile() const
Returns true if this repository requires the media.1/media file to be included in the metadata status...
void addBaseUrl(Url url)
Add a base url.
bool pkgGpgCheckIsMandatory() const
Mandatory check (pkgGpgCheck is not off) must ask to confirm using unsigned packages.
url_set rawBaseUrls() const
The complete set of raw repository urls (no variables replaced)
Url mirrorListUrl() const
Url of a file which contains a list of repository urls.
void setProbedType(const repo::RepoType &t) const
This allows to adjust the RepoType lazy, from NONE to some probed value, even for const objects.
void setBaseUrls(url_set urls)
Clears current base URL list and adds an url_set.
std::string service() const
Gets name of the service to which this repository belongs or empty string if it has been added manual...
void setTargetDistribution(const std::string &targetDistribution)
Sets the distribution for which is this repository meant.
std::ostream & dumpAsIniOn(std::ostream &str) const override
Write this RepoInfo object into str in a .repo file format.
unsigned priority() const
Repository priority for solver.
bool gpgCheck() const
Whether default signature checking should be performed.
void setPath(const Pathname &path)
set the product path.
Url gpgKeyUrl() const
(leagcy API) The 1st gpgkey URL defined for this repo
void setValidRepoSignature(TriBool value_r)
Set the value for validRepoSignature (or indeterminate if unsigned).
static unsigned defaultPriority()
The default priority (99).
bool needToAcceptLicense() const
Whether the repo license has to be accepted, e.g.
void setPriority(unsigned newval_r)
Set repository priority for solver.
RWCOW_pointer< Impl > _pimpl
Pointer to implementation.
urls_const_iterator baseUrlsBegin() const
iterator that points at begin of repository urls
bool hasLicense() const
Whether there is a license associated with the repo.
bool repoGpgCheckIsMandatory() const
Mandatory check (repoGpgCheck is on) must ask to confirm using unsigned repos.
void setMetadataPath(const Pathname &path)
Set the path where the local metadata is stored.
TriBool validRepoSignature() const
Whether the repo metadata are signed and successfully validated or indeterminate if unsigned.
void setRepoGpgCheck(TriBool value_r)
Set the value for repoGpgCheck (or indeterminate to use the default).
void setMetalinkUrls(url_set urls)
Like setMirrorListUrls but expect metalink format.
Pathname packagesPath() const
Path where this repo packages are cached.
void addContent(const std::string &keyword_r)
Add content keywords.
bool repoGpgCheck() const
Whether the signature of repo metadata should be checked for this repo.
std::string targetDistribution() const
Distribution for which is this repository meant.
void getRawGpgChecks(TriBool &g_r, TriBool &r_r, TriBool &p_r) const
Raw values for RepoManager.
void setType(const repo::RepoType &t)
set the repository type
bool pkgGpgCheck() const
Whether the signature of rpm packages should be checked for this repo.
std::string repoStatusString() const
A string value to track changes requiring a refresh.
Url rawMirrorListUrl() const
The raw mirrorListUrl (no variables replaced).
bool gpgKeyUrlsEmpty() const
Whether gpgkey URLs are defined.
std::string asString() const
Returns a default string representation of the Url object.
bool gpgCheck() const
Turn signature checking on/off (on)
TriBool pkgGpgCheck() const
Check rpm package signatures (indeterminate - according to gpgcheck)
static ZConfig & instance()
Singleton ctor.
TriBool repoGpgCheck() const
Check repo matadata signatures (indeterminate - according to gpgcheck)
std::string receiveLine()
Read one line from the input stream.
Find pathnames matching a pattern.
bool empty() const
Whether matches were found.
const_iterator begin() const
Iterator pointing to the first result.
int add(const Pathname &pattern_r, Flags flags_r=Flags())
Add pathnames matching pattern_r to the current result.
Wrapper class for stat/lstat.
bool isExist() const
Return whether valid stat info exists.
const char * c_str() const
String representation.
const std::string & asString() const
String representation.
bool empty() const
Test for an empty path.
Pathname filepath() const
File where this repo was read from.
bool autorefresh() const
If true, the repostory must be refreshed before creating resolvables from it.
std::string name() const
Repository name.
bool enabled() const
If enabled is false, then this repository must be ignored as if does not exists, except when checking...
std::string alias() const
unique identifier for this source.
const std::vector< Url > & getUrls() const
xmlTextReader based interface to iterate xml streams.
bool seekToEndNode(int depth_r, const std::string &name_r)
XmlString nodeText()
If the current node is not empty, advances the reader to the next node, and returns the value.
bool seekToNode(int depth_r, const std::string &name_r)
std::string asString() const
Explicit conversion to std::string.
static SyncContextRef defaultContext()
String related utilities and Regular expression matching.
int symlink(const Pathname &oldpath, const Pathname &newpath)
Like 'symlink'.
int unlink(const Pathname &path)
Like 'unlink'.
std::string asString(const Url &url_r)
int forEachLine(std::istream &str_r, const function< bool(int, std::string)> &consume_r)
Simple lineparser: Call functor consume_r for each line.
bool hasSuffix(const C_Str &str_r, const C_Str &suffix_r)
Return whether str_r has suffix suffix_r.
bool hasPrefix(const C_Str &str_r, const C_Str &prefix_r)
Return whether str_r has prefix prefix_r.
bool startsWith(const C_Str &str_r, const C_Str &prefix_r)
alias for hasPrefix
std::string form(const char *format,...) __attribute__((format(printf
Printf style construction of std::string.
unsigned split(const C_Str &line_r, TOutputIterator result_r, const C_Str &sepchars_r=" \t", const Trim trim_r=NO_TRIM)
Split line_r into words.
ZYPP_API detail::EscapedString escape(const std::string &in_r)
Escape xml special charaters (& -> &; from IoBind library).
Easy-to use interface to the ZYPP dependency resolver.
std::unordered_set< Locale > LocaleSet
std::ostream & operator<<(std::ostream &str, const SerialNumber &obj)
boost::logic::tribool TriBool
3-state boolean logic (true, false and indeterminate).
std::string asString(const Patch::Category &obj)
zypp::Pathname provideKey(SyncContextRef ctx, zypp::RepoInfo info, std::string keyID_r, zypp::Pathname targetDirectory_r)
TriBool rawPkgGpgCheck() const
TriBool _rawRepoGpgCheck
need to check repo sign.: Y/N/(ZConf(Y/N/gpgCheck))
TriBool internalValidRepoSignature() const
Signature check result needs to be stored/retrieved from _metadataPath.
std::string repoStatusString() const
bool triBoolFromPath(const Pathname &path_r, TriBool &ret_r) const
bool internalUnsignedConfirmed() const
We definitely have a symlink pointing to "indeterminate" (for repoGpgCheckIsMandatory)?
TriBool triBoolFromPath(const Pathname &path_r) const
Impl(const Impl &)=default
void packagesPath(Pathname new_r)
void rawRepoGpgCheck(TriBool val_r)
void rawPkgGpgCheck(TriBool val_r)
Pathname predownloadPath() const
Pathname licenseTgz(const std::string &name_r) const
Path to a license tarball in case it exists in the repo.
RepoVariablesReplacedUrl _mirrorListUrl
bool _mirrorListForceMetalink
Impl * clone() const
clone for RWCOW_pointer
Pathname metadataPath() const
DefaultIntegral< unsigned, defaultPriority > priority
std::ostream & operator<<(std::ostream &str, const RepoInfo::Impl &obj)
Stream output.
void setType(const repo::RepoType &t)
friend Impl * rwcowClone(const Impl *rhs)
bool hasContent(const std::string &keyword_r) const
const std::set< std::string > & contentKeywords() const
TriBool cfgPkgGpgCheck() const
bool baseurl2dump() const
const RepoVariablesReplacedUrlList & baseUrls() const
RepoVariablesReplacedUrlList _gpgKeyUrls
RepoVariablesReplacedUrlList _baseUrls
Impl & operator=(Impl &&)=delete
void rawGpgCheck(TriBool val_r)
void addContent(const std::string &keyword_r)
TriBool _rawGpgCheck
default gpgcheck behavior: Y/N/ZConf
Pathname packagesPath() const
void metadataPath(Pathname new_r)
TriBool rawGpgCheck() const
TriBool rawRepoGpgCheck() const
void internalSetValidRepoSignature(TriBool value_r)
const RepoVariablesReplacedUrlList & gpgKeyUrls() const
TriBool cfgRepoGpgCheck() const
static const unsigned defaultPriority
TriBool _rawPkgGpgCheck
need to check pkg sign.: Y/N/(ZConf(Y/N/gpgCheck))
void setProbedType(const repo::RepoType &t) const
RepoVariablesReplacedUrlList & baseUrls()
TriBool _validRepoSignature
have signed and valid repo metadata
repo::RepoType type() const
Impl & operator=(const Impl &)=delete
std::pair< FalseBool, std::set< std::string > > _keywords
RepoVariablesReplacedUrlList & gpgKeyUrls()
bool usesAutoMetadataPaths() const
static const unsigned noPriority
Repository type enumeration.
static const RepoType YAST2
const std::string & asString() const
static const RepoType RPMMD
static const RepoType NONE
static const RepoType RPMPLAINDIR
#define for_(IT, BEG, END)
Convenient for-loops using iterator.