Skip to content

Commit

Permalink
WIP
Browse files Browse the repository at this point in the history
Repo/Service file: Full support for using repo variables in URLs (bsc#1217192)

With a libzypp providing 'libzypp(repovarexpand) >= 1.2' the use of
repo variables in URLs is no longer restricted to the host and path
component of an URL, nor must the not-expanded string form a valid URL.
  • Loading branch information
mlandres committed Nov 29, 2023
1 parent 466f22f commit 6b31a68
Show file tree
Hide file tree
Showing 10 changed files with 145 additions and 25 deletions.
2 changes: 2 additions & 0 deletions doc/autoinclude/FeatureTest.doc
Original file line number Diff line number Diff line change
Expand Up @@ -79,6 +79,8 @@ Packages requiring a feature may use the corresponding \c Requires: in their .sp
<DD>Also support user defined repo variables in /etc/zypp/vars.d.</DD>
<DT>version 1.1</DT>
<DD>Support repo variables in an URIs \c host and \c port component.</DD>
<DT>version 1.2</DT>
<DD>Support repo variables forming valid URIs after expansion - but not before.</DD>
</DL></DD>
</DL>

Expand Down
5 changes: 3 additions & 2 deletions doc/autoinclude/RepoVariables.doc
Original file line number Diff line number Diff line change
Expand Up @@ -15,7 +15,7 @@ The variable expander also supports shell like definition of default and alterna
\see \ref zypp::repo::RepoVarExpand Variable expander


\subsection zypp-repoars-builtin Builtin repository variables
\subsection zypp-repovars-builtin Builtin repository variables

\li \c $arch -
The system's CPU architecture.
Expand All @@ -31,10 +31,11 @@ The variable expander also supports shell like definition of default and alterna
\c $releasever_major will be set to the leading portion up to (but not including) the 1st dot; \c $releasever_minor to the trailing portion after the 1st dot. If there's no dot in \c $releasever, \c $releasever_major is the same as \c $releasever and \c $releasever_minor is empty.


\subsection zypp-repoars-userdefined User defined repository variables [requires 'libzypp(repovarexpand) >= 1']
\subsection zypp-repovars-userdefined User defined repository variables [requires 'libzypp(repovarexpand) >= 1']

A custom repository variable is defined by creating a file in \c /etc/zypp/vars.d. The variable name equals the file name. The files first line (up to but not including the newline character) defines the variables value. Valid variable(file) names consist of alphanumeric chars and '_' only.

Variable substitution within an URIs authority [requires 'libzypp(repovarexpand) >= 1.1'] is limited to \c host and \c port. Bash style definition of default and alternate values is not supported. No variables can be used in an URIs \c scheme, \c user and \c password.

The use of arbitrary strings which form a valid url after variable substitution - but not before - requires 'libzypp(repovarexpand) >= 1.2'. The \c zypp-rawurl: schema is used to wrap the unexpanded strings into a valid \ref Url until they get expanded (\see \ref zypp::RawUrl).
*/
2 changes: 1 addition & 1 deletion libzypp.spec.cmake
Original file line number Diff line number Diff line change
Expand Up @@ -62,7 +62,7 @@ Provides: libzypp(plugin:services) = 1
Provides: libzypp(plugin:system) = 1
Provides: libzypp(plugin:urlresolver) = 0
Provides: libzypp(plugin:repoverification) = 0
Provides: libzypp(repovarexpand) = 1.1
Provides: libzypp(repovarexpand) = 1.2

%if 0%{?suse_version}
Recommends: logrotate
Expand Down
92 changes: 92 additions & 0 deletions tests/parser/RepoFileReader_test.cc
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,23 @@ using std::stringstream;
using std::string;
using namespace zypp;

// This is kind of ugly for testing: Whenever a zypp lock is
// acquired the repo variables are cleared and re-read from disk,
// which is ok (by ZYppFactory).
// The VarReplacer itself however acquires a lock to resolve the
// Target variables (like $releasever), which is ok as well.
// For testing however - if we want to manipulate the variable
// definitions - we must hold a lock ourselves. Otherwise the
// VarReplacer will acquired one and so discard our custom
// settings.
#include <zypp/ZYppFactory.h>
auto guard { getZYpp() };
namespace zyppintern {
std::map<std::string,std::string> repoVariablesGet();
void repoVariablesSwap( std::map<std::string,std::string> & val_r );
}
// ---

static std::string suse_repo = "[factory-oss]\n"
"name=factory-oss $releasever - $basearch\n"
"failovermethod=priority\n"
Expand Down Expand Up @@ -56,3 +73,78 @@ BOOST_AUTO_TEST_CASE(read_repo_file)
BOOST_CHECK_EQUAL( Url("http://serv.er/loc1"), repo.mirrorListUrl() );
}
}

BOOST_AUTO_TEST_CASE(rawurl2repoinfo)
{
// Set up repo variables...
std::map<std::string,std::string> vardef { ::zyppintern::repoVariablesGet() };
vardef["releasever"] = "myversion";
vardef["basearch"] = "myarch";
vardef["OPENSUSE_DISTURL"] = "https://cdn.opensuse.org/repositories/";
::zyppintern::repoVariablesSwap( vardef );

{
std::stringstream input(
"[leap-repo]\n"
"name=leap-repo $releasever - $basearch\n"
"baseurl= ${OPENSUSE_DISTURL}leap/repo\n"
"gpgkey= ${OPENSUSE_DISTURL}leap/repo\n"
"mirrorlist= ${OPENSUSE_DISTURL}leap/repo\n"
"metalink= ${OPENSUSE_DISTURL}leap/repo\n"
);
RepoCollector collector;
parser::RepoFileReader parser( input, bind( &RepoCollector::collect, &collector, _1 ) );
BOOST_CHECK_EQUAL(1, collector.repos.size());

const RepoInfo & repo( collector.repos.front() );
BOOST_CHECK_EQUAL( repo.name(), "leap-repo myversion - myarch" );

BOOST_CHECK_EQUAL( repo.url().asCompleteString(), "https://cdn.opensuse.org/repositories/leap/repo" );
BOOST_CHECK_EQUAL( repo.gpgKeyUrl().asCompleteString(), "https://cdn.opensuse.org/repositories/leap/repo" );
BOOST_CHECK_EQUAL( repo.mirrorListUrl().asCompleteString(), "https://cdn.opensuse.org/repositories/leap/repo" );

BOOST_CHECK_EQUAL( repo.rawUrl().asCompleteString(), "zypp-rawurl:#${OPENSUSE_DISTURL}leap/repo" );
BOOST_CHECK_EQUAL( repo.rawGpgKeyUrl().asCompleteString(), "zypp-rawurl:#${OPENSUSE_DISTURL}leap/repo" );
BOOST_CHECK_EQUAL( repo.rawMirrorListUrl().asCompleteString(), "zypp-rawurl:#${OPENSUSE_DISTURL}leap/repo" );
}

// RepoFileReader unfortunately encodes "proxy=" into the URLs
// query part (by now just for the baseurl). For RawUrls the
// VarReplacer needs to take this into account.
{
std::stringstream input(
"[leap-repo]\n"
"name=leap-repo $releasever - $basearch\n"
"proxy=myproxy.host:1234\n"
"baseurl=${OPENSUSE_DISTURL}leap/repo\n"
"baseurl=https://cdn.opensuse.org/repositories/leap/repo\n"
"baseurl=https://cdn.opensuse.org/repositories/leap/repo?proxy=otherproxy.host\n"
"gpgkey=${OPENSUSE_DISTURL}leap/repo\n"
"mirrorlist=${OPENSUSE_DISTURL}leap/repo\n"
"metalink=${OPENSUSE_DISTURL}leap/repo\n"
);
RepoCollector collector;
parser::RepoFileReader parser( input, bind( &RepoCollector::collect, &collector, _1 ) );
BOOST_CHECK_EQUAL(1, collector.repos.size());

const RepoInfo & repo( collector.repos.front() );
BOOST_CHECK_EQUAL( repo.name(), "leap-repo myversion - myarch" );

BOOST_CHECK_EQUAL( repo.baseUrlsSize(), 3 );
const auto & baseurls = repo.baseUrls();
auto iter = baseurls.begin();
BOOST_CHECK_EQUAL( (iter++)->asCompleteString(), "https://cdn.opensuse.org/repositories/leap/repo?proxy=myproxy.host&proxyport=1234" );
BOOST_CHECK_EQUAL( (iter++)->asCompleteString(), "https://cdn.opensuse.org/repositories/leap/repo?proxy=myproxy.host&proxyport=1234" );
BOOST_CHECK_EQUAL( (iter++)->asCompleteString(), "https://cdn.opensuse.org/repositories/leap/repo?proxy=otherproxy.host" );
BOOST_CHECK_EQUAL( repo.gpgKeyUrl().asCompleteString(), "https://cdn.opensuse.org/repositories/leap/repo" );
BOOST_CHECK_EQUAL( repo.mirrorListUrl().asCompleteString(), "https://cdn.opensuse.org/repositories/leap/repo" );

const auto & rawbaseurls = repo.rawBaseUrls();
iter = rawbaseurls.begin();
BOOST_CHECK_EQUAL( (iter++)->asCompleteString(), "zypp-rawurl:?proxy=myproxy.host&proxyport=1234#${OPENSUSE_DISTURL}leap/repo" );
BOOST_CHECK_EQUAL( (iter++)->asCompleteString(), "https://cdn.opensuse.org/repositories/leap/repo?proxy=myproxy.host&proxyport=1234" );
BOOST_CHECK_EQUAL( (iter++)->asCompleteString(), "https://cdn.opensuse.org/repositories/leap/repo?proxy=otherproxy.host" );
BOOST_CHECK_EQUAL( repo.rawGpgKeyUrl().asCompleteString(), "zypp-rawurl:#${OPENSUSE_DISTURL}leap/repo" );
BOOST_CHECK_EQUAL( repo.rawMirrorListUrl().asCompleteString(), "zypp-rawurl:#${OPENSUSE_DISTURL}leap/repo" );
}
}
16 changes: 10 additions & 6 deletions zypp/RepoInfo.cc
Original file line number Diff line number Diff line change
Expand Up @@ -884,10 +884,12 @@ namespace zypp

std::ostream & RepoInfo::dumpOn( std::ostream & str ) const
{
// Don't dump raw Urls while the fragment is hidden in the default view
// (because it may otherwise expose embedded passwords).
RepoInfoBase::dumpOn(str);
if ( _pimpl->baseurl2dump() )
{
for ( const auto & url : _pimpl->baseUrls().raw() )
for ( const auto & url : _pimpl->baseUrls().transformed() ) // not .raw()!
{
str << "- url : " << url << std::endl;
}
Expand All @@ -899,7 +901,7 @@ namespace zypp
str << tag_r << value_r << std::endl;
});

strif( (_pimpl->_mirrorListForceMetalink ? "- metalink : " : "- mirrorlist : "), rawMirrorListUrl().asString() );
strif( (_pimpl->_mirrorListForceMetalink ? "- metalink : " : "- mirrorlist : "), mirrorListUrl().asString() ); // not rawMirrorListUrl()
strif( "- path : ", path().asString() );
str << "- type : " << type() << std::endl;
str << "- priority : " << priority() << std::endl;
Expand All @@ -913,7 +915,7 @@ namespace zypp
<< std::endl;
#undef OUTS

for ( const auto & url : _pimpl->gpgKeyUrls().raw() )
for ( const auto & url : _pimpl->gpgKeyUrls().transformed() ) // not .raw()!
{
str << "- gpgkey : " << url << std::endl;
}
Expand All @@ -932,6 +934,8 @@ namespace zypp

std::ostream & RepoInfo::dumpAsIniOn( std::ostream & str ) const
{
// NOTE: hotfix1050625::asString provides the Url asRawSring,
// which is what we want when writing a .repo file.
RepoInfoBase::dumpAsIniOn(str);

if ( _pimpl->baseurl2dump() )
Expand All @@ -948,8 +952,8 @@ namespace zypp
if ( ! _pimpl->path.empty() )
str << "path="<< path() << endl;

if ( ! (rawMirrorListUrl().asString().empty()) )
str << (_pimpl->_mirrorListForceMetalink ? "metalink=" : "mirrorlist=") << hotfix1050625::asString( rawMirrorListUrl() ) << endl;
if ( _pimpl->_mirrorListUrl.raw().isValid() )
str << (_pimpl->_mirrorListForceMetalink ? "metalink=" : "mirrorlist=") << hotfix1050625::asString( _pimpl->_mirrorListUrl.raw() ) << endl;

if ( type() != repo::RepoType::NONE )
str << "type=" << type().asString() << endl;
Expand All @@ -970,7 +974,7 @@ namespace zypp
std::string indent( "gpgkey=");
for ( const auto & url : _pimpl->gpgKeyUrls().raw() )
{
str << indent << url << endl;
str << indent << hotfix1050625::asString( url ) << endl;
if ( indent[0] != ' ' )
indent = " ";
}
Expand Down
25 changes: 16 additions & 9 deletions zypp/RepoInfo.h
Original file line number Diff line number Diff line change
Expand Up @@ -65,8 +65,8 @@ namespace zypp
* \note A RepoInfo is a hint about how
* to create a Repository.
*
* \note Name, baseUrls and mirrorUrl are subject to repo variable replacement
* (\see \ref RepoVariablesStringReplacer).
* \note Name, baseUrls, gpgKeyUrl and mirrorUrl are subject to repo variable
* replacement(\see \ref RepoVariablesStringReplacer and \ref RawUrl).
*/
class RepoInfo : public repo::RepoInfoBase
{
Expand Down Expand Up @@ -132,8 +132,9 @@ namespace zypp
{ return( baseUrlsEmpty() ? Url() : *baseUrlsBegin()); }
/**
* Pars pro toto: The first repository raw url (no variables replaced)
* \deprecated \see \ref RawUrl
*/
Url rawUrl() const;
Url rawUrl() const ZYPP_DEPRECATED;

/**
* The complete set of repository urls
Expand All @@ -144,8 +145,9 @@ namespace zypp
url_set baseUrls() const;
/**
* The complete set of raw repository urls (no variables replaced)
* \deprecated \see \ref RawUrl
*/
url_set rawBaseUrls() const;
url_set rawBaseUrls() const ZYPP_DEPRECATED;

/**
* Add a base url. \see baseUrls
Expand Down Expand Up @@ -197,8 +199,9 @@ namespace zypp
Url mirrorListUrl() const;
/**
* The raw mirrorListUrl (no variables replaced).
* \deprecated \see \ref RawUrl
*/
Url rawMirrorListUrl() const;
Url rawMirrorListUrl() const ZYPP_DEPRECATED;
/**
* Set mirror list url. \see mirrorListUrl
* \param url The base url for the list
Expand Down Expand Up @@ -391,15 +394,19 @@ namespace zypp

/** The list of gpgkey URLs defined for this repo */
url_set gpgKeyUrls() const;
/** The list of raw gpgkey URLs defined for this repo (no variables replaced) */
url_set rawGpgKeyUrls() const;
/** The list of raw gpgkey URLs defined for this repo (no variables replaced)
* \deprecated \see \ref RawUrl
*/
url_set rawGpgKeyUrls() const ZYPP_DEPRECATED;
/** Set a list of gpgkey URLs defined for this repo */
void setGpgKeyUrls( url_set urls );

/** (leagcy API) The 1st gpgkey URL defined for this repo */
Url gpgKeyUrl() const;
/** (leagcy API) The 1st raw gpgkey URL defined for this repo (no variables replaced) */
Url rawGpgKeyUrl() const;
/** (leagcy API) The 1st raw gpgkey URL defined for this repo (no variables replaced)
* \deprecated \see \ref RawUrl
*/
Url rawGpgKeyUrl() const ZYPP_DEPRECATED;
/** (leagcy API) Set the gpgkey URL defined for this repo */
void setGpgKeyUrl( const Url &gpgkey );

Expand Down
4 changes: 3 additions & 1 deletion zypp/ServiceInfo.cc
Original file line number Diff line number Diff line change
Expand Up @@ -172,8 +172,10 @@ namespace zypp

std::ostream & ServiceInfo::dumpAsIniOn( std::ostream & str ) const
{
// NOTE: hotfix1050625::asString provides the Url asRawSring,
// which is what we want when writing a .service file.
RepoInfoBase::dumpAsIniOn(str)
<< "url = " << hotfix1050625::asString( rawUrl() ) << endl
<< "url = " << hotfix1050625::asString( _pimpl->_url.raw() ) << endl
<< "type = " << type() << endl;

if ( ttl() )
Expand Down
12 changes: 7 additions & 5 deletions zypp/ServiceInfo.h
Original file line number Diff line number Diff line change
Expand Up @@ -65,8 +65,10 @@ namespace zypp
/** The service url */
Url url() const;

/** The service raw url (no variables replaced) */
Url rawUrl() const;
/** The service raw url (no variables replaced)
* \deprecated \see \ref RawUrl
*/
Url rawUrl() const ZYPP_DEPRECATED;

/** Set the service url (raw value) */
void setUrl( const Url& url );
Expand All @@ -85,16 +87,16 @@ namespace zypp
* You don't want to use the setters unless you are a \ref RepoManager.
*/
//@{
/** Sugested TTL between two metadata auto-refreshs.
/** Suggested TTL between two metadata auto-refreshes.
* The value (in seconds) may be provided in repoindex.xml:xpath:/repoindex@ttl.
* Default is \a 0 - perform each auto-refresh request.
*/
Date::Duration ttl() const;

/** Set sugested TTL. */
/** Set suggested TTL. */
void setTtl( Date::Duration ttl_r );

/** Lazy init sugested TTL. */
/** Lazy init suggested TTL. */
void setProbedTtl( Date::Duration ttl_r ) const;

/** Date of last refresh (if known). */
Expand Down
2 changes: 1 addition & 1 deletion zypp/parser/RepoFileReader.cc
Original file line number Diff line number Diff line change
Expand Up @@ -115,7 +115,7 @@ namespace zypp
// #285: Fedora/dnf allows WS separated urls (and an optional comma)
strv::splitRx( line_r, "[,[:blank:]]*[[:blank:]][,[:blank:]]*", [&store_r]( std::string_view w ) {
if ( ! w.empty() )
store_r.push_back( Url(std::string(w)) );
store_r.push_back( RawUrl(std::string(w)) ); // RawUrl! to support repo var replacement
});
}

Expand Down
10 changes: 10 additions & 0 deletions zypp/repo/RepoVariables.cc
Original file line number Diff line number Diff line change
Expand Up @@ -567,6 +567,16 @@ namespace zypp
if ( value.schemeIsRawUrl() ) {
const std::string & replaced { RepoVarExpand()( value.getFragment(), RepoVarsMap::lookup ) };
newurl = Url( replaced );
// Fixup some legacy issue: RepoFileReader encodes a 'proxy=' in .repo in
// the Url's Query part. We must restore it from there unless the original Url
// defined proxy in it's query part.
if ( newurl.getQueryParam( "proxy" ).empty() ) {
const std::string & p { value.getQueryParam( "proxy" ) };
if ( not p.empty() ) {
newurl.setQueryParam( "proxy", p );
newurl.setQueryParam( "proxyport", value.getQueryParam( "proxyport" ) );
}
}
} else {
// The traditional
Url::ViewOptions toReplace = value.getViewOptions() - url::ViewOption::WITH_USERNAME - url::ViewOption::WITH_PASSWORD;
Expand Down

0 comments on commit 6b31a68

Please sign in to comment.