Fix QUrl::matches for when removing authority parts (other than host)

QUrl::RemoveAuthority is RemoveUserInfo | RemovePort | 0x10, so the
condition

    if (options & QUrl::RemoveAuthority)

would match if any of the other bits for the username, password or port
were set, which meant we would skip the host comparison. Ditto for
username and RemovePassword.

[ChangeLog][QtCore][QUrl] Fixed a bug that caused QUrl::matches to
incorrectly compare two URLs with different hostnames or different
usernames as equal, if certain QUrl::RemoveXxx options were passed.

Change-Id: I015970a03b874898bba7fffd155856ab9d6cb1be
Fixes: QTBUG-70774
Reviewed-by: Mårten Nordheim <marten.nordheim@qt.io>
This commit is contained in:
Thiago Macieira 2018-09-27 11:56:07 -07:00
parent 94884246d4
commit caa598c843
2 changed files with 20 additions and 8 deletions

View File

@ -3722,37 +3722,37 @@ bool QUrl::matches(const QUrl &url, FormattingOptions options) const
if (isLocalFile()) if (isLocalFile())
mask &= ~QUrlPrivate::Host; mask &= ~QUrlPrivate::Host;
if (options & QUrl::RemoveScheme) if (options.testFlag(QUrl::RemoveScheme))
mask &= ~QUrlPrivate::Scheme; mask &= ~QUrlPrivate::Scheme;
else if (d->scheme != url.d->scheme) else if (d->scheme != url.d->scheme)
return false; return false;
if (options & QUrl::RemovePassword) if (options.testFlag(QUrl::RemovePassword))
mask &= ~QUrlPrivate::Password; mask &= ~QUrlPrivate::Password;
else if (d->password != url.d->password) else if (d->password != url.d->password)
return false; return false;
if (options & QUrl::RemoveUserInfo) if (options.testFlag(QUrl::RemoveUserInfo))
mask &= ~QUrlPrivate::UserName; mask &= ~QUrlPrivate::UserName;
else if (d->userName != url.d->userName) else if (d->userName != url.d->userName)
return false; return false;
if (options & QUrl::RemovePort) if (options.testFlag(QUrl::RemovePort))
mask &= ~QUrlPrivate::Port; mask &= ~QUrlPrivate::Port;
else if (d->port != url.d->port) else if (d->port != url.d->port)
return false; return false;
if (options & QUrl::RemoveAuthority) if (options.testFlag(QUrl::RemoveAuthority))
mask &= ~QUrlPrivate::Host; mask &= ~QUrlPrivate::Host;
else if (d->host != url.d->host) else if (d->host != url.d->host)
return false; return false;
if (options & QUrl::RemoveQuery) if (options.testFlag(QUrl::RemoveQuery))
mask &= ~QUrlPrivate::Query; mask &= ~QUrlPrivate::Query;
else if (d->query != url.d->query) else if (d->query != url.d->query)
return false; return false;
if (options & QUrl::RemoveFragment) if (options.testFlag(QUrl::RemoveFragment))
mask &= ~QUrlPrivate::Fragment; mask &= ~QUrlPrivate::Fragment;
else if (d->fragment != url.d->fragment) else if (d->fragment != url.d->fragment)
return false; return false;
@ -3760,7 +3760,7 @@ bool QUrl::matches(const QUrl &url, FormattingOptions options) const
if ((d->sectionIsPresent & mask) != (url.d->sectionIsPresent & mask)) if ((d->sectionIsPresent & mask) != (url.d->sectionIsPresent & mask))
return false; return false;
if (options & QUrl::RemovePath) if (options.testFlag(QUrl::RemovePath))
return true; return true;
// Compare paths, after applying path-related options // Compare paths, after applying path-related options

View File

@ -4123,6 +4123,18 @@ void tst_QUrl::matches_data()
<< "http://user:pass@www.website.com/directory" << "http://user:pass@www.website.com/directory"
<< "http://otheruser:otherpass@www.website.com/directory" << "http://otheruser:otherpass@www.website.com/directory"
<< uint(QUrl::RemovePath | QUrl::RemoveAuthority) << true; << uint(QUrl::RemovePath | QUrl::RemoveAuthority) << true;
QTest::newRow("matchingHost-removePort") << "http://example.com" << "http://example.com"
<< uint(QUrl::RemovePort) << true;
QTest::newRow("nonMatchingHost-removePort") << "http://example.com" << "http://example.net"
<< uint(QUrl::RemovePort) << false;
QTest::newRow("matchingHost-removePassword") << "http://example.com" << "http://example.com"
<< uint(QUrl::RemovePassword) << true;
QTest::newRow("nonMatchingHost-removePassword") << "http://example.com" << "http://example.net"
<< uint(QUrl::RemovePassword) << false;
QTest::newRow("matchingUserName-removePassword") << "http://user@example.com" << "http://user@example.com"
<< uint(QUrl::RemovePassword) << true;
QTest::newRow("nonMatchingUserName-removePassword") << "http://user@example.com" << "http://user2@example.com"
<< uint(QUrl::RemovePassword) << false;
} }
void tst_QUrl::matches() void tst_QUrl::matches()