summaryrefslogtreecommitdiffstats
path: root/apps
diff options
context:
space:
mode:
authorVincent Petry <vincent@nextcloud.com>2022-02-10 10:02:17 +0100
committerGitHub <noreply@github.com>2022-02-10 10:02:17 +0100
commitb8ddca8119f448cab29afd40cfea2c793ecdef72 (patch)
tree032b4be61e7221b9715f0e1766525f565013be16 /apps
parentbcbd42d56c0af7eb2f2f3aac9da84e3bcb271fd5 (diff)
parent41fe871111fe8fabc10004f82a6e62bf11432fc2 (diff)
downloadnextcloud-server-b8ddca8119f448cab29afd40cfea2c793ecdef72.tar.gz
nextcloud-server-b8ddca8119f448cab29afd40cfea2c793ecdef72.zip
Merge pull request #31013 from nextcloud/backport/30114/stable22
[stable22] Prevent writing invalid mtime
Diffstat (limited to 'apps')
-rw-r--r--apps/dav/lib/Connector/Sabre/Node.php5
-rw-r--r--apps/dav/tests/unit/Connector/Sabre/FileTest.php24
-rw-r--r--apps/dav/tests/unit/Connector/Sabre/NodeTest.php48
3 files changed, 63 insertions, 14 deletions
diff --git a/apps/dav/lib/Connector/Sabre/Node.php b/apps/dav/lib/Connector/Sabre/Node.php
index aa03f42dc35..3d7ffafadc2 100644
--- a/apps/dav/lib/Connector/Sabre/Node.php
+++ b/apps/dav/lib/Connector/Sabre/Node.php
@@ -412,6 +412,11 @@ abstract class Node implements \Sabre\DAV\INode {
throw new \InvalidArgumentException('X-OC-MTime header must be an integer (unix timestamp).');
}
+ // Prevent writing invalid mtime (timezone-proof)
+ if ((int)$mtimeFromRequest <= 24 * 60 * 60) {
+ throw new \InvalidArgumentException('X-OC-MTime header must be a valid positive integer');
+ }
+
return (int)$mtimeFromRequest;
}
}
diff --git a/apps/dav/tests/unit/Connector/Sabre/FileTest.php b/apps/dav/tests/unit/Connector/Sabre/FileTest.php
index 304e8abed07..3e6a47d5854 100644
--- a/apps/dav/tests/unit/Connector/Sabre/FileTest.php
+++ b/apps/dav/tests/unit/Connector/Sabre/FileTest.php
@@ -361,28 +361,28 @@ class FileTest extends TestCase {
'expected result' => null
],
"castable string (int)" => [
- 'HTTP_X_OC_MTIME' => "34",
- 'expected result' => 34
+ 'HTTP_X_OC_MTIME' => "987654321",
+ 'expected result' => 987654321
],
"castable string (float)" => [
- 'HTTP_X_OC_MTIME' => "34.56",
- 'expected result' => 34
+ 'HTTP_X_OC_MTIME' => "123456789.56",
+ 'expected result' => 123456789
],
"float" => [
- 'HTTP_X_OC_MTIME' => 34.56,
- 'expected result' => 34
+ 'HTTP_X_OC_MTIME' => 123456789.56,
+ 'expected result' => 123456789
],
"zero" => [
'HTTP_X_OC_MTIME' => 0,
- 'expected result' => 0
+ 'expected result' => null
],
"zero string" => [
'HTTP_X_OC_MTIME' => "0",
- 'expected result' => 0
+ 'expected result' => null
],
"negative zero string" => [
'HTTP_X_OC_MTIME' => "-0",
- 'expected result' => 0
+ 'expected result' => null
],
"string starting with number following by char" => [
'HTTP_X_OC_MTIME' => "2345asdf",
@@ -398,11 +398,11 @@ class FileTest extends TestCase {
],
"negative int" => [
'HTTP_X_OC_MTIME' => -34,
- 'expected result' => -34
+ 'expected result' => null
],
"negative float" => [
'HTTP_X_OC_MTIME' => -34.43,
- 'expected result' => -34
+ 'expected result' => null
],
];
}
@@ -421,7 +421,6 @@ class FileTest extends TestCase {
if ($resultMtime === null) {
$this->expectException(\InvalidArgumentException::class);
- $this->expectExceptionMessage("X-OC-MTime header must be an integer (unix timestamp).");
}
$this->doPut($file, null, $request);
@@ -447,7 +446,6 @@ class FileTest extends TestCase {
if ($resultMtime === null) {
$this->expectException(\Sabre\DAV\Exception::class);
- $this->expectExceptionMessage("X-OC-MTime header must be an integer (unix timestamp).");
}
$this->doPut($file.'-chunking-12345-2-0', null, $request);
diff --git a/apps/dav/tests/unit/Connector/Sabre/NodeTest.php b/apps/dav/tests/unit/Connector/Sabre/NodeTest.php
index 18ccd0fe690..00fd0ebd8aa 100644
--- a/apps/dav/tests/unit/Connector/Sabre/NodeTest.php
+++ b/apps/dav/tests/unit/Connector/Sabre/NodeTest.php
@@ -164,8 +164,54 @@ class NodeTest extends \Test\TestCase {
->disableOriginalConstructor()
->getMock();
- $node = new \OCA\DAV\Connector\Sabre\File($view, $info);
+ $node = new \OCA\DAV\Connector\Sabre\File($view, $info);
$this->invokePrivate($node, 'shareManager', [$shareManager]);
$this->assertEquals($expected, $node->getSharePermissions($user));
}
+
+ public function sanitizeMtimeProvider() {
+ return [
+ [123456789, 123456789],
+ ['987654321', 987654321],
+ ];
+ }
+
+ /**
+ * @dataProvider sanitizeMtimeProvider
+ */
+ public function testSanitizeMtime($mtime, $expected) {
+ $view = $this->getMockBuilder(View::class)
+ ->disableOriginalConstructor()
+ ->getMock();
+ $info = $this->getMockBuilder(FileInfo::class)
+ ->disableOriginalConstructor()
+ ->getMock();
+
+ $node = new \OCA\DAV\Connector\Sabre\File($view, $info);
+ $result = $this->invokePrivate($node, 'sanitizeMtime', [$mtime]);
+ $this->assertEquals($expected, $result);
+ }
+
+ public function invalidSanitizeMtimeProvider() {
+ return [
+ [-1337], [0], ['abcdef'], ['-1337'], ['0'], [12321], [24 * 60 * 60 - 1]
+ ];
+ }
+
+ /**
+ * @dataProvider invalidSanitizeMtimeProvider
+ */
+ public function testInvalidSanitizeMtime($mtime) {
+ $this->expectException(\InvalidArgumentException::class);
+
+ $view = $this->getMockBuilder(View::class)
+ ->disableOriginalConstructor()
+ ->getMock();
+ $info = $this->getMockBuilder(FileInfo::class)
+ ->disableOriginalConstructor()
+ ->getMock();
+
+ $node = new \OCA\DAV\Connector\Sabre\File($view, $info);
+ $result = $this->invokePrivate($node, 'sanitizeMtime', [$mtime]);
+ }
}