mirror of
https://we.phorge.it/source/phorge.git
synced 2025-01-10 23:01:04 +01:00
When deleting a file, delete all transformations of the file
Summary: Fixes T3143. When a user deletes a file, delete all transforms of the file too. In particular, this means that deleting an image deletes all the thumbnails of the image. In most cases, this aligns with user expectations. The only sort of weird case I can come up with is that memes are transformations of the source macro image, so deleting grumpycat will delete all the hilarious grumpycat memes. This seems not-too-unreasonable, though, and desirable if someone accidentally uploads an inappropriate image which is promptly turned into a meme. Test Plan: Added a unit test which covers both inbound and outbound transformations. Uploaded a file and deleted it, verified its thumbnail was also deleted. Reviewers: chad, btrahan, joseph.kampf Reviewed By: btrahan CC: aran, joseph.kampf Maniphest Tasks: T3143 Differential Revision: https://secure.phabricator.com/D5879
This commit is contained in:
parent
dc6c1bf435
commit
402d2e2605
3 changed files with 183 additions and 16 deletions
|
@ -24,6 +24,38 @@ final class PhabricatorFileQuery
|
||||||
return $this;
|
return $this;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Select files which are transformations of some other file. For example,
|
||||||
|
* you can use this query to find previously generated thumbnails of an image
|
||||||
|
* file.
|
||||||
|
*
|
||||||
|
* As a parameter, provide a list of transformation specifications. Each
|
||||||
|
* specification is a dictionary with the keys `originalPHID` and `transform`.
|
||||||
|
* The `originalPHID` is the PHID of the original file (the file which was
|
||||||
|
* transformed) and the `transform` is the name of the transform to query
|
||||||
|
* for. If you pass `true` as the `transform`, all transformations of the
|
||||||
|
* file will be selected.
|
||||||
|
*
|
||||||
|
* For example:
|
||||||
|
*
|
||||||
|
* array(
|
||||||
|
* array(
|
||||||
|
* 'originalPHID' => 'PHID-FILE-aaaa',
|
||||||
|
* 'transform' => 'sepia',
|
||||||
|
* ),
|
||||||
|
* array(
|
||||||
|
* 'originalPHID' => 'PHID-FILE-bbbb',
|
||||||
|
* 'transform' => true,
|
||||||
|
* ),
|
||||||
|
* )
|
||||||
|
*
|
||||||
|
* This selects the `"sepia"` transformation of the file with PHID
|
||||||
|
* `PHID-FILE-aaaa` and all transformations of the file with PHID
|
||||||
|
* `PHID-FILE-bbbb`.
|
||||||
|
*
|
||||||
|
* @param list<dict> List of transform specifications, described above.
|
||||||
|
* @return this
|
||||||
|
*/
|
||||||
public function withTransforms(array $specs) {
|
public function withTransforms(array $specs) {
|
||||||
foreach ($specs as $spec) {
|
foreach ($specs as $spec) {
|
||||||
if (!is_array($spec) ||
|
if (!is_array($spec) ||
|
||||||
|
@ -50,7 +82,7 @@ final class PhabricatorFileQuery
|
||||||
|
|
||||||
$data = queryfx_all(
|
$data = queryfx_all(
|
||||||
$conn_r,
|
$conn_r,
|
||||||
'SELECT * FROM %T f %Q %Q %Q %Q',
|
'SELECT f.* FROM %T f %Q %Q %Q %Q',
|
||||||
$table->getTableName(),
|
$table->getTableName(),
|
||||||
$this->buildJoinClause($conn_r),
|
$this->buildJoinClause($conn_r),
|
||||||
$this->buildWhereClause($conn_r),
|
$this->buildWhereClause($conn_r),
|
||||||
|
@ -108,11 +140,18 @@ final class PhabricatorFileQuery
|
||||||
if ($this->transforms) {
|
if ($this->transforms) {
|
||||||
$clauses = array();
|
$clauses = array();
|
||||||
foreach ($this->transforms as $transform) {
|
foreach ($this->transforms as $transform) {
|
||||||
$clauses[] = qsprintf(
|
if ($transform['transform'] === true) {
|
||||||
$conn_r,
|
$clauses[] = qsprintf(
|
||||||
'(t.originalPHID = %s AND t.transform = %s)',
|
$conn_r,
|
||||||
$transform['originalPHID'],
|
'(t.originalPHID = %s)',
|
||||||
$transform['transform']);
|
$transform['originalPHID']);
|
||||||
|
} else {
|
||||||
|
$clauses[] = qsprintf(
|
||||||
|
$conn_r,
|
||||||
|
'(t.originalPHID = %s AND t.transform = %s)',
|
||||||
|
$transform['originalPHID'],
|
||||||
|
$transform['transform']);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
$where[] = qsprintf($conn_r, '(%Q)', implode(') OR (', $clauses));
|
$where[] = qsprintf($conn_r, '(%Q)', implode(') OR (', $clauses));
|
||||||
}
|
}
|
||||||
|
|
|
@ -384,29 +384,54 @@ final class PhabricatorFile extends PhabricatorFileDAO
|
||||||
}
|
}
|
||||||
|
|
||||||
public function delete() {
|
public function delete() {
|
||||||
// delete all records of this file in transformedfile
|
|
||||||
$trans_files = id(new PhabricatorTransformedFile())->loadAllWhere(
|
// We want to delete all the rows which mark this file as the transformation
|
||||||
'TransformedPHID = %s', $this->getPHID());
|
// of some other file (since we're getting rid of it). We also delete all
|
||||||
|
// the transformations of this file, so that a user who deletes an image
|
||||||
|
// doesn't need to separately hunt down and delete a bunch of thumbnails and
|
||||||
|
// resizes of it.
|
||||||
|
|
||||||
|
$outbound_xforms = id(new PhabricatorFileQuery())
|
||||||
|
->setViewer(PhabricatorUser::getOmnipotentUser())
|
||||||
|
->withTransforms(
|
||||||
|
array(
|
||||||
|
array(
|
||||||
|
'originalPHID' => $this->getPHID(),
|
||||||
|
'transform' => true,
|
||||||
|
),
|
||||||
|
))
|
||||||
|
->execute();
|
||||||
|
|
||||||
|
foreach ($outbound_xforms as $outbound_xform) {
|
||||||
|
$outbound_xform->delete();
|
||||||
|
}
|
||||||
|
|
||||||
|
$inbound_xforms = id(new PhabricatorTransformedFile())->loadAllWhere(
|
||||||
|
'transformedPHID = %s',
|
||||||
|
$this->getPHID());
|
||||||
|
|
||||||
$this->openTransaction();
|
$this->openTransaction();
|
||||||
foreach ($trans_files as $trans_file) {
|
foreach ($inbound_xforms as $inbound_xform) {
|
||||||
$trans_file->delete();
|
$inbound_xform->delete();
|
||||||
}
|
}
|
||||||
$ret = parent::delete();
|
$ret = parent::delete();
|
||||||
$this->saveTransaction();
|
$this->saveTransaction();
|
||||||
|
|
||||||
// Check to see if other files are using storage
|
// Check to see if other files are using storage
|
||||||
$other_file = id(new PhabricatorFile())->loadAllWhere(
|
$other_file = id(new PhabricatorFile())->loadAllWhere(
|
||||||
'storageEngine = %s AND storageHandle = %s AND
|
'storageEngine = %s AND storageHandle = %s AND
|
||||||
storageFormat = %s AND id != %d LIMIT 1', $this->getStorageEngine(),
|
storageFormat = %s AND id != %d LIMIT 1',
|
||||||
$this->getStorageHandle(), $this->getStorageFormat(),
|
$this->getStorageEngine(),
|
||||||
|
$this->getStorageHandle(),
|
||||||
|
$this->getStorageFormat(),
|
||||||
$this->getID());
|
$this->getID());
|
||||||
|
|
||||||
// If this is the only file using the storage, delete storage
|
// If this is the only file using the storage, delete storage
|
||||||
if (count($other_file) == 0) {
|
if (!$other_file) {
|
||||||
$engine = $this->instantiateStorageEngine();
|
$engine = $this->instantiateStorageEngine();
|
||||||
$engine->deleteFile($this->getStorageHandle());
|
$engine->deleteFile($this->getStorageHandle());
|
||||||
}
|
}
|
||||||
|
|
||||||
return $ret;
|
return $ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -145,4 +145,107 @@ final class PhabricatorFileTestCase extends PhabricatorTestCase {
|
||||||
$this->assertEqual($ttl, $file->getTTL());
|
$this->assertEqual($ttl, $file->getTTL());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function testFileTransformDelete() {
|
||||||
|
// We want to test that a file deletes all its inbound transformation
|
||||||
|
// records and outbound transformed derivatives when it is deleted.
|
||||||
|
|
||||||
|
// First, we create a chain of transforms, A -> B -> C.
|
||||||
|
|
||||||
|
$engine = new PhabricatorTestStorageEngine();
|
||||||
|
|
||||||
|
$params = array(
|
||||||
|
'name' => 'test.txt',
|
||||||
|
'storageEngines' => array(
|
||||||
|
$engine,
|
||||||
|
),
|
||||||
|
);
|
||||||
|
|
||||||
|
$a = PhabricatorFile::newFromFileData('a', $params);
|
||||||
|
$b = PhabricatorFile::newFromFileData('b', $params);
|
||||||
|
$c = PhabricatorFile::newFromFileData('c', $params);
|
||||||
|
|
||||||
|
id(new PhabricatorTransformedFile())
|
||||||
|
->setOriginalPHID($a->getPHID())
|
||||||
|
->setTransform('test:a->b')
|
||||||
|
->setTransformedPHID($b->getPHID())
|
||||||
|
->save();
|
||||||
|
|
||||||
|
id(new PhabricatorTransformedFile())
|
||||||
|
->setOriginalPHID($b->getPHID())
|
||||||
|
->setTransform('test:b->c')
|
||||||
|
->setTransformedPHID($c->getPHID())
|
||||||
|
->save();
|
||||||
|
|
||||||
|
// Now, verify that A -> B and B -> C exist.
|
||||||
|
|
||||||
|
$xform_a = id(new PhabricatorFileQuery())
|
||||||
|
->setViewer(PhabricatorUser::getOmnipotentUser())
|
||||||
|
->withTransforms(
|
||||||
|
array(
|
||||||
|
array(
|
||||||
|
'originalPHID' => $a->getPHID(),
|
||||||
|
'transform' => true,
|
||||||
|
),
|
||||||
|
))
|
||||||
|
->execute();
|
||||||
|
|
||||||
|
$this->assertEqual(1, count($xform_a));
|
||||||
|
$this->assertEqual($b->getPHID(), head($xform_a)->getPHID());
|
||||||
|
|
||||||
|
$xform_b = id(new PhabricatorFileQuery())
|
||||||
|
->setViewer(PhabricatorUser::getOmnipotentUser())
|
||||||
|
->withTransforms(
|
||||||
|
array(
|
||||||
|
array(
|
||||||
|
'originalPHID' => $b->getPHID(),
|
||||||
|
'transform' => true,
|
||||||
|
),
|
||||||
|
))
|
||||||
|
->execute();
|
||||||
|
|
||||||
|
$this->assertEqual(1, count($xform_b));
|
||||||
|
$this->assertEqual($c->getPHID(), head($xform_b)->getPHID());
|
||||||
|
|
||||||
|
// Delete "B".
|
||||||
|
|
||||||
|
$b->delete();
|
||||||
|
|
||||||
|
// Now, verify that the A -> B and B -> C links are gone.
|
||||||
|
|
||||||
|
$xform_a = id(new PhabricatorFileQuery())
|
||||||
|
->setViewer(PhabricatorUser::getOmnipotentUser())
|
||||||
|
->withTransforms(
|
||||||
|
array(
|
||||||
|
array(
|
||||||
|
'originalPHID' => $a->getPHID(),
|
||||||
|
'transform' => true,
|
||||||
|
),
|
||||||
|
))
|
||||||
|
->execute();
|
||||||
|
|
||||||
|
$this->assertEqual(0, count($xform_a));
|
||||||
|
|
||||||
|
$xform_b = id(new PhabricatorFileQuery())
|
||||||
|
->setViewer(PhabricatorUser::getOmnipotentUser())
|
||||||
|
->withTransforms(
|
||||||
|
array(
|
||||||
|
array(
|
||||||
|
'originalPHID' => $b->getPHID(),
|
||||||
|
'transform' => true,
|
||||||
|
),
|
||||||
|
))
|
||||||
|
->execute();
|
||||||
|
|
||||||
|
$this->assertEqual(0, count($xform_b));
|
||||||
|
|
||||||
|
// Also verify that C has been deleted.
|
||||||
|
|
||||||
|
$alternate_c = id(new PhabricatorFileQuery())
|
||||||
|
->setViewer(PhabricatorUser::getOmnipotentUser())
|
||||||
|
->withPHIDs(array($c->getPHID()))
|
||||||
|
->execute();
|
||||||
|
|
||||||
|
$this->assertEqual(array(), $alternate_c);
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue