1
0
Fork 0
mirror of https://we.phorge.it/source/phorge.git synced 2025-01-07 05:11:05 +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:
epriestley 2014-01-24 12:29:03 -08:00
parent 2735229e33
commit 11786fb1cc
2 changed files with 72 additions and 53 deletions

View file

@ -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);
} }

View file

@ -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));
} }