mirror of
https://we.phorge.it/source/phorge.git
synced 2025-01-15 17:21:10 +01:00
Protect file data with a one-time-token
Test Plan: currently untested work in progress Reviewers: #blessed_reviewers, epriestley Subscribers: rush898, aklapper, Korvin, epriestley Projects: #wikimedia Maniphest Tasks: T5685 Differential Revision: https://secure.phabricator.com/D10054
This commit is contained in:
parent
c0919be0ec
commit
25ae4c458d
4 changed files with 129 additions and 27 deletions
20
resources/sql/autopatches/20140731.cancdn.php
Normal file
20
resources/sql/autopatches/20140731.cancdn.php
Normal file
|
@ -0,0 +1,20 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
$table = new PhabricatorFile();
|
||||||
|
$conn_w = $table->establishConnection('w');
|
||||||
|
foreach (new LiskMigrationIterator($table) as $file) {
|
||||||
|
$id = $file->getID();
|
||||||
|
echo "Updating flags for file {$id}...\n";
|
||||||
|
$meta = $file->getMetadata();
|
||||||
|
if (!idx($meta, 'canCDN')) {
|
||||||
|
|
||||||
|
$meta['canCDN'] = true;
|
||||||
|
|
||||||
|
queryfx(
|
||||||
|
$conn_w,
|
||||||
|
'UPDATE %T SET metadata = %s WHERE id = %d',
|
||||||
|
$table->getTableName(),
|
||||||
|
json_encode($meta),
|
||||||
|
$id);
|
||||||
|
}
|
||||||
|
}
|
|
@ -52,6 +52,8 @@ final class PhabricatorFilesApplication extends PhabricatorApplication {
|
||||||
'delete/(?P<id>[1-9]\d*)/' => 'PhabricatorFileDeleteController',
|
'delete/(?P<id>[1-9]\d*)/' => 'PhabricatorFileDeleteController',
|
||||||
'edit/(?P<id>[1-9]\d*)/' => 'PhabricatorFileEditController',
|
'edit/(?P<id>[1-9]\d*)/' => 'PhabricatorFileEditController',
|
||||||
'info/(?P<phid>[^/]+)/' => 'PhabricatorFileInfoController',
|
'info/(?P<phid>[^/]+)/' => 'PhabricatorFileInfoController',
|
||||||
|
'data/(?P<key>[^/]+)/(?P<phid>[^/]+)/(?P<token>[^/]+)/.*'
|
||||||
|
=> 'PhabricatorFileDataController',
|
||||||
'data/(?P<key>[^/]+)/(?P<phid>[^/]+)/.*'
|
'data/(?P<key>[^/]+)/(?P<phid>[^/]+)/.*'
|
||||||
=> 'PhabricatorFileDataController',
|
=> 'PhabricatorFileDataController',
|
||||||
'proxy/' => 'PhabricatorFileProxyController',
|
'proxy/' => 'PhabricatorFileProxyController',
|
||||||
|
|
|
@ -4,35 +4,19 @@ final class PhabricatorFileDataController extends PhabricatorFileController {
|
||||||
|
|
||||||
private $phid;
|
private $phid;
|
||||||
private $key;
|
private $key;
|
||||||
|
private $token;
|
||||||
|
|
||||||
public function willProcessRequest(array $data) {
|
public function willProcessRequest(array $data) {
|
||||||
$this->phid = $data['phid'];
|
$this->phid = $data['phid'];
|
||||||
$this->key = $data['key'];
|
$this->key = $data['key'];
|
||||||
|
$this->token = idx($data, 'token');
|
||||||
}
|
}
|
||||||
|
|
||||||
public function shouldRequireLogin() {
|
public function shouldRequireLogin() {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function processRequest() {
|
protected function checkFileAndToken($file) {
|
||||||
$request = $this->getRequest();
|
|
||||||
|
|
||||||
$alt = PhabricatorEnv::getEnvConfig('security.alternate-file-domain');
|
|
||||||
$uri = new PhutilURI($alt);
|
|
||||||
$alt_domain = $uri->getDomain();
|
|
||||||
if ($alt_domain && ($alt_domain != $request->getHost())) {
|
|
||||||
return id(new AphrontRedirectResponse())
|
|
||||||
->setURI($uri->setPath($request->getPath()));
|
|
||||||
}
|
|
||||||
|
|
||||||
// NOTE: This endpoint will ideally be accessed via CDN or otherwise on
|
|
||||||
// a non-credentialed domain. Knowing the file's secret key gives you
|
|
||||||
// access, regardless of authentication on the request itself.
|
|
||||||
|
|
||||||
$file = id(new PhabricatorFileQuery())
|
|
||||||
->setViewer(PhabricatorUser::getOmnipotentUser())
|
|
||||||
->withPHIDs(array($this->phid))
|
|
||||||
->executeOne();
|
|
||||||
if (!$file) {
|
if (!$file) {
|
||||||
return new Aphront404Response();
|
return new Aphront404Response();
|
||||||
}
|
}
|
||||||
|
@ -41,10 +25,97 @@ final class PhabricatorFileDataController extends PhabricatorFileController {
|
||||||
return new Aphront403Response();
|
return new Aphront403Response();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
public function processRequest() {
|
||||||
|
$request = $this->getRequest();
|
||||||
|
|
||||||
|
$alt = PhabricatorEnv::getEnvConfig('security.alternate-file-domain');
|
||||||
|
$base_uri = PhabricatorEnv::getEnvConfig('phabricator.base-uri');
|
||||||
|
$alt_uri = new PhutilURI($alt);
|
||||||
|
$alt_domain = $alt_uri->getDomain();
|
||||||
|
$req_domain = $request->getHost();
|
||||||
|
$main_domain = id(new PhutilURI($base_uri))->getDomain();
|
||||||
|
|
||||||
|
$cache_response = true;
|
||||||
|
|
||||||
|
if (empty($alt) || $main_domain == $alt_domain) {
|
||||||
|
// Alternate files domain isn't configured or it's set
|
||||||
|
// to the same as the default domain
|
||||||
|
|
||||||
|
// load the file with permissions checks;
|
||||||
|
$file = id(new PhabricatorFileQuery())
|
||||||
|
->setViewer($request->getUser())
|
||||||
|
->withPHIDs(array($this->phid))
|
||||||
|
->executeOne();
|
||||||
|
|
||||||
|
$error_response = $this->checkFileAndToken($file);
|
||||||
|
if ($error_response) {
|
||||||
|
return $error_response;
|
||||||
|
}
|
||||||
|
|
||||||
|
// when the file is not CDNable, don't allow cache
|
||||||
|
$cache_response = $file->getCanCDN();
|
||||||
|
} else if ($req_domain != $alt_domain) {
|
||||||
|
// Alternate domain is configured but this request isn't using it
|
||||||
|
|
||||||
|
// load the file with permissions checks;
|
||||||
|
$file = id(new PhabricatorFileQuery())
|
||||||
|
->setViewer($request->getUser())
|
||||||
|
->withPHIDs(array($this->phid))
|
||||||
|
->executeOne();
|
||||||
|
|
||||||
|
$error_response = $this->checkFileAndToken($file);
|
||||||
|
if ($error_response) {
|
||||||
|
return $error_response;
|
||||||
|
}
|
||||||
|
|
||||||
|
// if the user can see the file, generate a token;
|
||||||
|
// redirect to the alt domain with the token;
|
||||||
|
return id(new AphrontRedirectResponse())
|
||||||
|
->setURI($file->getCDNURIWithToken());
|
||||||
|
|
||||||
|
} else {
|
||||||
|
// We are using the alternate domain
|
||||||
|
|
||||||
|
// load the file, bypassing permission checks;
|
||||||
|
$file = id(new PhabricatorFileQuery())
|
||||||
|
->setViewer(PhabricatorUser::getOmnipotentUser())
|
||||||
|
->withPHIDs(array($this->phid))
|
||||||
|
->executeOne();
|
||||||
|
|
||||||
|
$error_response = $this->checkFileAndToken($file);
|
||||||
|
if ($error_response) {
|
||||||
|
return $error_response;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($this->token) {
|
||||||
|
// validate the token, if it is valid, continue
|
||||||
|
$validated_token = $file->validateOneTimeToken($this->token);
|
||||||
|
|
||||||
|
if (!$validated_token) {
|
||||||
|
return new Aphront403Response();
|
||||||
|
}
|
||||||
|
// return the file data without cache headers
|
||||||
|
$cache_response = false;
|
||||||
|
} else if (!$file->getCanCDN()) {
|
||||||
|
// file cannot be served via cdn, and no token given
|
||||||
|
// redirect to the main domain to aquire a token
|
||||||
|
$file_uri = id(new PhutilURI($file->getViewURI()))
|
||||||
|
->setDomain($main_domain);
|
||||||
|
|
||||||
|
return id(new AphrontRedirectResponse())
|
||||||
|
->setURI($file_uri);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
$data = $file->loadFileData();
|
$data = $file->loadFileData();
|
||||||
$response = new AphrontFileResponse();
|
$response = new AphrontFileResponse();
|
||||||
$response->setContent($data);
|
$response->setContent($data);
|
||||||
$response->setCacheDurationInSeconds(60 * 60 * 24 * 30);
|
if ($cache_response) {
|
||||||
|
$response->setCacheDurationInSeconds(60 * 60 * 24 * 30);
|
||||||
|
}
|
||||||
|
|
||||||
// NOTE: It's important to accept "Range" requests when playing audio.
|
// NOTE: It's important to accept "Range" requests when playing audio.
|
||||||
// If we don't, Safari has difficulty figuring out how long sounds are
|
// If we don't, Safari has difficulty figuring out how long sounds are
|
||||||
|
@ -58,6 +129,11 @@ final class PhabricatorFileDataController extends PhabricatorFileController {
|
||||||
$response->setHTTPResponseCode(206);
|
$response->setHTTPResponseCode(206);
|
||||||
$response->setRange((int)$matches[1], (int)$matches[2]);
|
$response->setRange((int)$matches[1], (int)$matches[2]);
|
||||||
}
|
}
|
||||||
|
} else if (isset($validated_token)) {
|
||||||
|
// consume the one-time token if we have one.
|
||||||
|
$unguarded = AphrontWriteGuard::beginScopedUnguardedWrites();
|
||||||
|
$validated_token->delete();
|
||||||
|
unset($unguarded);
|
||||||
}
|
}
|
||||||
|
|
||||||
$is_viewable = $file->isViewableInBrowser();
|
$is_viewable = $file->isViewableInBrowser();
|
||||||
|
|
|
@ -873,12 +873,16 @@ final class PhabricatorFile extends PhabricatorFileDAO
|
||||||
$key = Filesystem::readRandomCharacters(16);
|
$key = Filesystem::readRandomCharacters(16);
|
||||||
|
|
||||||
// Save the new secret.
|
// Save the new secret.
|
||||||
return id(new PhabricatorAuthTemporaryToken())
|
$unguarded = AphrontWriteGuard::beginScopedUnguardedWrites();
|
||||||
->setObjectPHID($this->getPHID())
|
$token = id(new PhabricatorAuthTemporaryToken())
|
||||||
->setTokenType(self::ONETIME_TEMPORARY_TOKEN_TYPE)
|
->setObjectPHID($this->getPHID())
|
||||||
->setTokenExpires(time() + phutil_units('1 hour in seconds'))
|
->setTokenType(self::ONETIME_TEMPORARY_TOKEN_TYPE)
|
||||||
->setTokenCode(PhabricatorHash::digest($key))
|
->setTokenExpires(time() + phutil_units('1 hour in seconds'))
|
||||||
->save();
|
->setTokenCode(PhabricatorHash::digest($key))
|
||||||
|
->save();
|
||||||
|
unset($unguarded);
|
||||||
|
|
||||||
|
return $key;
|
||||||
}
|
}
|
||||||
|
|
||||||
public function validateOneTimeToken($token_code) {
|
public function validateOneTimeToken($token_code) {
|
||||||
|
@ -887,7 +891,7 @@ final class PhabricatorFile extends PhabricatorFileDAO
|
||||||
->withObjectPHIDs(array($this->getPHID()))
|
->withObjectPHIDs(array($this->getPHID()))
|
||||||
->withTokenTypes(array(self::ONETIME_TEMPORARY_TOKEN_TYPE))
|
->withTokenTypes(array(self::ONETIME_TEMPORARY_TOKEN_TYPE))
|
||||||
->withExpired(false)
|
->withExpired(false)
|
||||||
->withTokenCodes(array($token_code))
|
->withTokenCodes(array(PhabricatorHash::digest($token_code)))
|
||||||
->executeOne();
|
->executeOne();
|
||||||
|
|
||||||
return $token;
|
return $token;
|
||||||
|
|
Loading…
Reference in a new issue