mirror of
https://we.phorge.it/source/phorge.git
synced 2025-01-20 19:51:08 +01:00
Don't try to set anonymous session cookie on CDN/file domain
Summary: Ref T2380. If an install has a CDN domain configured, but does not list it as an alternate domain (which is standard/correct, but not incredibly common, see T2380), we'll currently try to set anonymous cookies on it. These will correctly fail security rules. Instead, don't try to set these cookies. I missed this in testing yesterday because I have a file domain, but I also have it configured as an alternate domain, which allows cookies to be set. Generally, domain management is due for some refactoring. Test Plan: Set file domain but not as an alternate, logged out, nuked file domain cookies, reloaded page. No error after patch. Reviewers: btrahan, csilvers Reviewed By: btrahan CC: aran Maniphest Tasks: T2380 Differential Revision: https://secure.phabricator.com/D8057
This commit is contained in:
parent
2735229e33
commit
11786fb1cc
2 changed files with 72 additions and 53 deletions
|
@ -1,10 +1,9 @@
|
||||||
<?php
|
<?php
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
* @task data Accessing Request Data
|
||||||
|
* @task cookie Managing Cookies
|
||||||
*
|
*
|
||||||
* @task data Accessing Request Data
|
|
||||||
*
|
|
||||||
* @group aphront
|
|
||||||
*/
|
*/
|
||||||
final class AphrontRequest {
|
final class AphrontRequest {
|
||||||
|
|
||||||
|
@ -297,62 +296,76 @@ final class AphrontRequest {
|
||||||
unset($_COOKIE[$name]);
|
unset($_COOKIE[$name]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the domain which cookies should be set on for this request, or null
|
||||||
|
* if the request does not correspond to a valid cookie domain.
|
||||||
|
*
|
||||||
|
* @return PhutilURI|null Domain URI, or null if no valid domain exists.
|
||||||
|
*
|
||||||
|
* @task cookie
|
||||||
|
*/
|
||||||
|
private function getCookieDomainURI() {
|
||||||
|
$host = $this->getHost();
|
||||||
|
|
||||||
|
// If there's no base domain configured, just use whatever the request
|
||||||
|
// domain is. This makes setup easier, and we'll tell administrators to
|
||||||
|
// configure a base domain during the setup process.
|
||||||
|
$base_uri = PhabricatorEnv::getEnvConfig('phabricator.base-uri');
|
||||||
|
if (!strlen($base_uri)) {
|
||||||
|
return new PhutilURI('http://'.$host.'/');
|
||||||
|
}
|
||||||
|
|
||||||
|
$alternates = PhabricatorEnv::getEnvConfig('phabricator.allowed-uris');
|
||||||
|
$allowed_uris = array_merge(
|
||||||
|
array($base_uri),
|
||||||
|
$alternates);
|
||||||
|
|
||||||
|
foreach ($allowed_uris as $allowed_uri) {
|
||||||
|
$uri = new PhutilURI($allowed_uri);
|
||||||
|
if ($uri->getDomain() == $host) {
|
||||||
|
return $uri;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Determine if security policy rules will allow cookies to be set when
|
||||||
|
* responding to the request.
|
||||||
|
*
|
||||||
|
* @return bool True if setCookie() will succeed. If this method returns
|
||||||
|
* false, setCookie() will throw.
|
||||||
|
*
|
||||||
|
* @task cookie
|
||||||
|
*/
|
||||||
|
final public function canSetCookies() {
|
||||||
|
return (bool)$this->getCookieDomainURI();
|
||||||
|
}
|
||||||
|
|
||||||
final public function setCookie($name, $value, $expire = null) {
|
final public function setCookie($name, $value, $expire = null) {
|
||||||
|
|
||||||
$is_secure = false;
|
$is_secure = false;
|
||||||
|
|
||||||
// If a base URI has been configured, ensure cookies are only set on that
|
$base_domain_uri = $this->getCookieDomainURI();
|
||||||
// domain. Also, use the URI protocol to control SSL-only cookies.
|
if (!$base_domain_uri) {
|
||||||
$base_uri = PhabricatorEnv::getEnvConfig('phabricator.base-uri');
|
$configured_as = PhabricatorEnv::getEnvConfig('phabricator.base-uri');
|
||||||
if ($base_uri) {
|
$accessed_as = $this->getHost();
|
||||||
$alternates = PhabricatorEnv::getEnvConfig('phabricator.allowed-uris');
|
|
||||||
$allowed_uris = array_merge(
|
|
||||||
array($base_uri),
|
|
||||||
$alternates);
|
|
||||||
|
|
||||||
$host = $this->getHost();
|
throw new Exception(
|
||||||
|
pht(
|
||||||
$match = null;
|
'This Phabricator install is configured as "%s", but you are '.
|
||||||
foreach ($allowed_uris as $allowed_uri) {
|
'using the domain name "%s" to access a page which is trying to '.
|
||||||
$uri = new PhutilURI($allowed_uri);
|
'set a cookie. Acccess Phabricator on the configured primary '.
|
||||||
$domain = $uri->getDomain();
|
'domain or a configured alternate domain. Phabricator will not '.
|
||||||
if ($host == $domain) {
|
'set cookies on other domains for security reasons.',
|
||||||
$match = $uri;
|
$configured_as,
|
||||||
break;
|
$accessed_as));
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($match === null) {
|
|
||||||
if (count($allowed_uris) > 1) {
|
|
||||||
throw new Exception(
|
|
||||||
pht(
|
|
||||||
'This Phabricator install is configured as "%s", but you are '.
|
|
||||||
'accessing it via "%s". Access Phabricator via the primary '.
|
|
||||||
'configured domain, or one of the permitted alternate '.
|
|
||||||
'domains: %s. Phabricator will not set cookies on other domains '.
|
|
||||||
'for security reasons.',
|
|
||||||
$base_uri,
|
|
||||||
$host,
|
|
||||||
implode(', ', $alternates)));
|
|
||||||
} else {
|
|
||||||
throw new Exception(
|
|
||||||
pht(
|
|
||||||
'This Phabricator install is configured as "%s", but you are '.
|
|
||||||
'accessing it via "%s". Acccess Phabricator via the primary '.
|
|
||||||
'configured domain. Phabricator will not set cookies on other '.
|
|
||||||
'domains for security reasons.',
|
|
||||||
$base_uri,
|
|
||||||
$host));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
$base_domain = $match->getDomain();
|
|
||||||
$is_secure = ($match->getProtocol() == 'https');
|
|
||||||
} else {
|
|
||||||
$base_uri = new PhutilURI(PhabricatorEnv::getRequestBaseURI());
|
|
||||||
$base_domain = $base_uri->getDomain();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$base_domain = $base_domain_uri->getDomain();
|
||||||
|
$is_secure = ($base_domain_uri->getProtocol() == 'https');
|
||||||
|
|
||||||
if ($expire === null) {
|
if ($expire === null) {
|
||||||
$expire = time() + (60 * 60 * 24 * 365 * 5);
|
$expire = time() + (60 * 60 * 24 * 365 * 5);
|
||||||
}
|
}
|
||||||
|
|
|
@ -50,9 +50,15 @@ abstract class PhabricatorController extends AphrontController {
|
||||||
$phsid = $session_engine->establishSession(
|
$phsid = $session_engine->establishSession(
|
||||||
PhabricatorAuthSession::TYPE_WEB,
|
PhabricatorAuthSession::TYPE_WEB,
|
||||||
null);
|
null);
|
||||||
$request->setCookie(PhabricatorCookies::COOKIE_SESSION, $phsid);
|
|
||||||
|
// This may be a resource request, in which case we just don't set
|
||||||
|
// the cookie.
|
||||||
|
if ($request->canSetCookies()) {
|
||||||
|
$request->setCookie(PhabricatorCookies::COOKIE_SESSION, $phsid);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
if (!$user->isLoggedIn()) {
|
if (!$user->isLoggedIn()) {
|
||||||
$user->attachAlternateCSRFString(PhabricatorHash::digest($phsid));
|
$user->attachAlternateCSRFString(PhabricatorHash::digest($phsid));
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue