2014-04-02 14:23:22 +00:00
|
|
|
<?php
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @author Andreas Fischer <bantu@phpbb.com>
|
2014-12-09 23:02:44 +00:00
|
|
|
* @copyright 2014 Andreas Fischer
|
2014-04-02 14:23:22 +00:00
|
|
|
* @license http://www.opensource.org/licenses/mit-license.html MIT License
|
|
|
|
*/
|
|
|
|
|
2022-02-23 02:48:51 +00:00
|
|
|
namespace phpseclib3\Tests\Functional\Net;
|
|
|
|
|
2019-11-07 05:41:40 +00:00
|
|
|
use phpseclib3\Net\SFTP;
|
2022-02-23 02:48:51 +00:00
|
|
|
use phpseclib3\Tests\PhpseclibFunctionalTestCase;
|
2014-12-10 01:31:41 +00:00
|
|
|
|
2022-02-23 02:48:51 +00:00
|
|
|
class SFTPUserStoryTest extends PhpseclibFunctionalTestCase
|
2014-04-02 14:23:22 +00:00
|
|
|
{
|
2022-02-17 02:25:59 +00:00
|
|
|
protected static $scratchDir;
|
|
|
|
protected static $exampleData;
|
|
|
|
protected static $exampleDataLength;
|
|
|
|
protected static $buffer;
|
2014-04-02 14:23:22 +00:00
|
|
|
|
2015-07-15 01:52:31 +00:00
|
|
|
public static function setUpBeforeClass()
|
2014-04-02 14:23:22 +00:00
|
|
|
{
|
|
|
|
parent::setUpBeforeClass();
|
|
|
|
|
|
|
|
self::$scratchDir = uniqid('phpseclib-sftp-scratch-');
|
|
|
|
|
|
|
|
self::$exampleData = str_repeat('abcde12345', 1000);
|
|
|
|
self::$exampleDataLength = 10000;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function testConstructor()
|
|
|
|
{
|
2014-12-10 01:31:41 +00:00
|
|
|
$sftp = new SFTP($this->getEnv('SSH_HOSTNAME'));
|
2014-04-02 14:23:22 +00:00
|
|
|
|
2020-12-13 01:22:36 +00:00
|
|
|
$this->assertIsObject(
|
2017-12-07 20:08:19 +00:00
|
|
|
$sftp,
|
2014-04-02 14:23:22 +00:00
|
|
|
'Could not construct NET_SFTP object.'
|
|
|
|
);
|
|
|
|
|
|
|
|
return $sftp;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2015-03-29 16:07:17 +00:00
|
|
|
* @depends testConstructor
|
|
|
|
*/
|
2014-04-02 14:23:22 +00:00
|
|
|
public function testPasswordLogin($sftp)
|
|
|
|
{
|
|
|
|
$username = $this->getEnv('SSH_USERNAME');
|
|
|
|
$password = $this->getEnv('SSH_PASSWORD');
|
|
|
|
$this->assertTrue(
|
|
|
|
$sftp->login($username, $password),
|
|
|
|
'SSH2/SFTP login using password failed.'
|
|
|
|
);
|
|
|
|
|
|
|
|
return $sftp;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2015-03-29 16:07:17 +00:00
|
|
|
* @depends testPasswordLogin
|
|
|
|
*/
|
2014-04-02 14:23:22 +00:00
|
|
|
public function testPwdHome($sftp)
|
|
|
|
{
|
|
|
|
$this->assertEquals(
|
|
|
|
$this->getEnv('SSH_HOME'),
|
|
|
|
$sftp->pwd(),
|
|
|
|
'Failed asserting that pwd() returns home directory after login.'
|
|
|
|
);
|
|
|
|
|
|
|
|
return $sftp;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2015-03-29 16:07:17 +00:00
|
|
|
* @depends testPwdHome
|
|
|
|
*/
|
2014-04-02 14:23:22 +00:00
|
|
|
public function testMkDirScratch($sftp)
|
|
|
|
{
|
|
|
|
$dirname = self::$scratchDir;
|
|
|
|
|
|
|
|
$this->assertTrue(
|
|
|
|
$sftp->mkdir($dirname),
|
|
|
|
"Failed asserting that a new scratch directory $dirname could " .
|
|
|
|
'be created.'
|
|
|
|
);
|
|
|
|
|
|
|
|
$this->assertFalse(
|
|
|
|
$sftp->mkdir($dirname),
|
|
|
|
"Failed asserting that a new scratch directory $dirname could " .
|
|
|
|
'not be created (because it already exists).'
|
|
|
|
);
|
|
|
|
|
|
|
|
return $sftp;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2015-03-29 16:07:17 +00:00
|
|
|
* @depends testMkDirScratch
|
|
|
|
*/
|
2014-04-02 14:23:22 +00:00
|
|
|
public function testChDirScratch($sftp)
|
|
|
|
{
|
|
|
|
$this->assertTrue(
|
|
|
|
$sftp->chdir(self::$scratchDir),
|
|
|
|
sprintf(
|
|
|
|
'Failed asserting that working directory could be changed ' .
|
|
|
|
'to scratch directory %s.',
|
|
|
|
self::$scratchDir
|
|
|
|
)
|
|
|
|
);
|
|
|
|
|
|
|
|
$pwd = $sftp->pwd();
|
|
|
|
|
|
|
|
$this->assertStringStartsWith(
|
|
|
|
$this->getEnv('SSH_HOME'),
|
|
|
|
$pwd,
|
|
|
|
'Failed asserting that the home directory is a prefix of the ' .
|
|
|
|
'current working directory.'
|
|
|
|
);
|
|
|
|
|
|
|
|
$this->assertStringEndsWith(
|
|
|
|
self::$scratchDir,
|
|
|
|
$pwd,
|
|
|
|
'Failed asserting that the scratch directory name is a suffix ' .
|
|
|
|
'of the current working directory.'
|
|
|
|
);
|
|
|
|
|
|
|
|
return $sftp;
|
|
|
|
}
|
|
|
|
|
2014-05-21 15:29:08 +00:00
|
|
|
/**
|
2015-03-29 16:07:17 +00:00
|
|
|
* @depends testChDirScratch
|
|
|
|
*/
|
2014-05-21 15:29:08 +00:00
|
|
|
public function testStatOnDir($sftp)
|
|
|
|
{
|
2014-05-21 22:38:04 +00:00
|
|
|
$this->assertNotSame(
|
2017-11-27 08:30:14 +00:00
|
|
|
[],
|
2014-05-21 22:38:04 +00:00
|
|
|
$sftp->stat('.'),
|
|
|
|
'Failed asserting that the cwd has a non-empty stat.'
|
2014-05-21 15:29:08 +00:00
|
|
|
);
|
|
|
|
|
|
|
|
return $sftp;
|
|
|
|
}
|
|
|
|
|
2022-02-17 02:25:59 +00:00
|
|
|
public static function demoCallback($length)
|
2015-04-15 00:29:01 +00:00
|
|
|
{
|
|
|
|
$r = substr(self::$buffer, 0, $length);
|
|
|
|
self::$buffer = substr(self::$buffer, $length);
|
2015-07-15 01:52:31 +00:00
|
|
|
if (strlen($r)) {
|
|
|
|
return $r;
|
|
|
|
}
|
2015-04-15 00:29:01 +00:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2014-04-02 14:23:22 +00:00
|
|
|
/**
|
2015-03-29 16:07:17 +00:00
|
|
|
* @depends testStatOnDir
|
|
|
|
*/
|
2014-04-02 14:23:22 +00:00
|
|
|
public function testPutSizeGetFile($sftp)
|
|
|
|
{
|
|
|
|
$this->assertTrue(
|
|
|
|
$sftp->put('file1.txt', self::$exampleData),
|
|
|
|
'Failed asserting that example data could be successfully put().'
|
|
|
|
);
|
|
|
|
|
|
|
|
$this->assertSame(
|
|
|
|
self::$exampleDataLength,
|
2020-02-12 05:01:18 +00:00
|
|
|
$sftp->filesize('file1.txt'),
|
2014-04-02 14:23:22 +00:00
|
|
|
'Failed asserting that put example data has the expected length'
|
|
|
|
);
|
|
|
|
|
|
|
|
$this->assertSame(
|
|
|
|
self::$exampleData,
|
|
|
|
$sftp->get('file1.txt'),
|
|
|
|
'Failed asserting that get() returns expected example data.'
|
|
|
|
);
|
|
|
|
|
2020-12-19 03:34:21 +00:00
|
|
|
$this->assertTrue(
|
|
|
|
$sftp->put('file1.txt', 'xxx', SFTP::RESUME),
|
|
|
|
'Failed asserting that an upload could be successfully resumed'
|
|
|
|
);
|
|
|
|
|
|
|
|
$this->assertSame(
|
|
|
|
self::$exampleDataLength + 3,
|
|
|
|
$sftp->filesize('file1.txt'),
|
|
|
|
'Failed asserting that put example data has the expected length'
|
|
|
|
);
|
|
|
|
|
|
|
|
$this->assertSame(
|
2020-12-19 07:14:06 +00:00
|
|
|
self::$exampleData . 'xxx',
|
2020-12-19 03:34:21 +00:00
|
|
|
$sftp->get('file1.txt'),
|
|
|
|
'Failed asserting that get() returns expected example data.'
|
|
|
|
);
|
|
|
|
|
2014-04-02 14:23:22 +00:00
|
|
|
return $sftp;
|
|
|
|
}
|
|
|
|
|
2015-04-08 23:59:00 +00:00
|
|
|
/**
|
2016-04-10 16:30:59 +00:00
|
|
|
* @depends testStatOnDir
|
|
|
|
*/
|
2015-04-08 23:59:00 +00:00
|
|
|
public function testPutSizeGetFileCallback($sftp)
|
|
|
|
{
|
2015-06-21 00:59:48 +00:00
|
|
|
self::$buffer = self::$exampleData;
|
2015-04-08 23:59:00 +00:00
|
|
|
$this->assertTrue(
|
2020-12-13 01:22:36 +00:00
|
|
|
$sftp->put('file1.txt', [__CLASS__, 'demoCallback'], $sftp::SOURCE_CALLBACK),
|
2015-04-08 23:59:00 +00:00
|
|
|
'Failed asserting that example data could be successfully put().'
|
|
|
|
);
|
|
|
|
|
|
|
|
$this->assertSame(
|
|
|
|
self::$exampleDataLength,
|
2020-02-12 05:01:18 +00:00
|
|
|
$sftp->filesize('file1.txt'),
|
2015-04-08 23:59:00 +00:00
|
|
|
'Failed asserting that put example data has the expected length'
|
|
|
|
);
|
|
|
|
|
|
|
|
$this->assertSame(
|
|
|
|
self::$exampleData,
|
|
|
|
$sftp->get('file1.txt'),
|
|
|
|
'Failed asserting that get() returns expected example data.'
|
|
|
|
);
|
|
|
|
|
|
|
|
return $sftp;
|
|
|
|
}
|
|
|
|
|
2014-05-21 15:29:08 +00:00
|
|
|
/**
|
2015-03-29 16:07:17 +00:00
|
|
|
* @depends testPutSizeGetFile
|
|
|
|
*/
|
2014-06-09 14:22:30 +00:00
|
|
|
public function testTouch($sftp)
|
|
|
|
{
|
|
|
|
$this->assertTrue(
|
|
|
|
$sftp->touch('file2.txt'),
|
|
|
|
'Failed asserting that touch() successfully ran.'
|
|
|
|
);
|
|
|
|
|
|
|
|
$this->assertTrue(
|
|
|
|
$sftp->file_exists('file2.txt'),
|
|
|
|
'Failed asserting that touch()\'d file exists'
|
|
|
|
);
|
|
|
|
|
|
|
|
return $sftp;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2015-03-29 16:07:17 +00:00
|
|
|
* @depends testTouch
|
|
|
|
*/
|
2014-06-09 14:22:30 +00:00
|
|
|
public function testTruncate($sftp)
|
|
|
|
{
|
|
|
|
$this->assertTrue(
|
|
|
|
$sftp->touch('file3.txt'),
|
|
|
|
'Failed asserting that touch() successfully ran.'
|
|
|
|
);
|
|
|
|
|
|
|
|
$this->assertTrue(
|
|
|
|
$sftp->truncate('file3.txt', 1024 * 1024),
|
|
|
|
'Failed asserting that touch() successfully ran.'
|
|
|
|
);
|
|
|
|
|
|
|
|
$this->assertSame(
|
|
|
|
1024 * 1024,
|
2020-02-12 05:01:18 +00:00
|
|
|
$sftp->filesize('file3.txt'),
|
2014-06-09 14:22:30 +00:00
|
|
|
'Failed asserting that truncate()\'d file has the expected length'
|
|
|
|
);
|
|
|
|
|
|
|
|
return $sftp;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2015-03-29 16:07:17 +00:00
|
|
|
* @depends testTruncate
|
2016-04-10 16:30:59 +00:00
|
|
|
* @group github850
|
|
|
|
*/
|
|
|
|
public function testChModOnFile($sftp)
|
|
|
|
{
|
|
|
|
$this->assertNotFalse(
|
|
|
|
$sftp->chmod(0755, 'file1.txt'),
|
|
|
|
'Failed asserting that chmod() was successful.'
|
|
|
|
);
|
|
|
|
|
|
|
|
return $sftp;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @depends testChModOnFile
|
2015-03-29 16:07:17 +00:00
|
|
|
*/
|
2014-05-21 15:29:08 +00:00
|
|
|
public function testChDirOnFile($sftp)
|
|
|
|
{
|
|
|
|
$this->assertFalse(
|
|
|
|
$sftp->chdir('file1.txt'),
|
|
|
|
'Failed to assert that the cwd cannot be changed to a file'
|
|
|
|
);
|
|
|
|
|
|
|
|
return $sftp;
|
|
|
|
}
|
|
|
|
|
2014-04-02 14:23:22 +00:00
|
|
|
/**
|
2015-03-29 16:07:17 +00:00
|
|
|
* @depends testChDirOnFile
|
|
|
|
*/
|
2014-05-13 23:11:19 +00:00
|
|
|
public function testFileExistsIsFileIsDirFile($sftp)
|
|
|
|
{
|
|
|
|
$this->assertTrue(
|
|
|
|
$sftp->file_exists('file1.txt'),
|
|
|
|
'Failed asserting that file_exists() on example file returns true.'
|
|
|
|
);
|
|
|
|
|
|
|
|
$this->assertTrue(
|
|
|
|
$sftp->is_file('file1.txt'),
|
|
|
|
'Failed asserting that is_file() on example file returns true.'
|
|
|
|
);
|
|
|
|
|
|
|
|
$this->assertFalse(
|
|
|
|
$sftp->is_dir('file1.txt'),
|
|
|
|
'Failed asserting that is_dir() on example file returns false.'
|
|
|
|
);
|
|
|
|
|
|
|
|
return $sftp;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2015-03-29 16:07:17 +00:00
|
|
|
* @depends testFileExistsIsFileIsDirFile
|
|
|
|
*/
|
2014-05-13 23:11:19 +00:00
|
|
|
public function testFileExistsIsFileIsDirFileNonexistent($sftp)
|
|
|
|
{
|
|
|
|
$this->assertFalse(
|
2014-06-09 14:22:30 +00:00
|
|
|
$sftp->file_exists('file4.txt'),
|
2014-05-13 23:11:19 +00:00
|
|
|
'Failed asserting that a nonexistent file does not exist.'
|
|
|
|
);
|
|
|
|
|
|
|
|
$this->assertFalse(
|
2014-06-09 14:22:30 +00:00
|
|
|
$sftp->is_file('file4.txt'),
|
2014-05-13 23:11:19 +00:00
|
|
|
'Failed asserting that is_file() on nonexistent file returns false.'
|
|
|
|
);
|
|
|
|
|
|
|
|
$this->assertFalse(
|
2014-06-09 14:22:30 +00:00
|
|
|
$sftp->is_dir('file4.txt'),
|
2014-05-13 23:11:19 +00:00
|
|
|
'Failed asserting that is_dir() on nonexistent file returns false.'
|
|
|
|
);
|
|
|
|
|
|
|
|
return $sftp;
|
|
|
|
}
|
|
|
|
|
2014-06-09 14:22:30 +00:00
|
|
|
/**
|
2015-03-29 16:07:17 +00:00
|
|
|
* @depends testFileExistsIsFileIsDirFileNonexistent
|
|
|
|
*/
|
2014-06-09 14:22:30 +00:00
|
|
|
public function testSortOrder($sftp)
|
|
|
|
{
|
|
|
|
$this->assertTrue(
|
|
|
|
$sftp->mkdir('temp'),
|
|
|
|
"Failed asserting that a new scratch directory temp could " .
|
|
|
|
'be created.'
|
|
|
|
);
|
|
|
|
|
|
|
|
$sftp->setListOrder('filename', SORT_DESC);
|
|
|
|
|
|
|
|
$list = $sftp->nlist();
|
2017-11-27 08:30:14 +00:00
|
|
|
$expected = ['.', '..', 'temp', 'file3.txt', 'file2.txt', 'file1.txt'];
|
2014-06-09 14:22:30 +00:00
|
|
|
|
2014-06-09 14:42:07 +00:00
|
|
|
$this->assertSame(
|
2014-06-09 14:22:30 +00:00
|
|
|
$list,
|
|
|
|
$expected,
|
|
|
|
'Failed asserting that list sorted correctly.'
|
|
|
|
);
|
|
|
|
|
|
|
|
$sftp->setListOrder('filename', SORT_ASC);
|
|
|
|
|
|
|
|
$list = $sftp->nlist();
|
2017-11-27 08:30:14 +00:00
|
|
|
$expected = ['.', '..', 'temp', 'file1.txt', 'file2.txt', 'file3.txt'];
|
2014-06-09 14:22:30 +00:00
|
|
|
|
2014-06-09 14:42:07 +00:00
|
|
|
$this->assertSame(
|
2014-06-09 14:22:30 +00:00
|
|
|
$list,
|
|
|
|
$expected,
|
|
|
|
'Failed asserting that list sorted correctly.'
|
|
|
|
);
|
|
|
|
|
|
|
|
$sftp->setListOrder('size', SORT_DESC);
|
|
|
|
|
|
|
|
$files = $sftp->nlist();
|
|
|
|
|
|
|
|
$last_size = 0x7FFFFFFF;
|
|
|
|
foreach ($files as $file) {
|
|
|
|
if ($sftp->is_file($file)) {
|
2020-02-12 05:01:18 +00:00
|
|
|
$cur_size = $sftp->filesize($file);
|
2014-06-09 14:22:30 +00:00
|
|
|
$this->assertLessThanOrEqual(
|
2015-07-15 01:52:31 +00:00
|
|
|
$last_size,
|
|
|
|
$cur_size,
|
2014-06-09 14:22:30 +00:00
|
|
|
'Failed asserting that nlist() is in descending order'
|
|
|
|
);
|
|
|
|
$last_size = $cur_size;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $sftp;
|
|
|
|
}
|
|
|
|
|
2014-05-13 23:11:19 +00:00
|
|
|
/**
|
2015-03-29 16:07:17 +00:00
|
|
|
* @depends testSortOrder
|
|
|
|
*/
|
2014-08-16 17:49:55 +00:00
|
|
|
public function testResourceXfer($sftp)
|
|
|
|
{
|
|
|
|
$fp = fopen('res.txt', 'w+');
|
|
|
|
$sftp->get('file1.txt', $fp);
|
|
|
|
rewind($fp);
|
|
|
|
$sftp->put('file4.txt', $fp);
|
|
|
|
fclose($fp);
|
|
|
|
|
|
|
|
$this->assertSame(
|
|
|
|
self::$exampleData,
|
|
|
|
$sftp->get('file4.txt'),
|
|
|
|
'Failed asserting that a file downloaded into a resource and reuploaded from a resource has the correct data'
|
|
|
|
);
|
|
|
|
|
|
|
|
return $sftp;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2015-03-29 16:07:17 +00:00
|
|
|
* @depends testResourceXfer
|
|
|
|
*/
|
2014-06-19 14:00:23 +00:00
|
|
|
public function testSymlink($sftp)
|
|
|
|
{
|
|
|
|
$this->assertTrue(
|
2014-06-19 18:57:42 +00:00
|
|
|
$sftp->symlink('file3.txt', 'symlink'),
|
2014-06-19 14:00:23 +00:00
|
|
|
'Failed asserting that a symlink could be created'
|
|
|
|
);
|
2014-06-20 04:57:32 +00:00
|
|
|
|
|
|
|
return $sftp;
|
2014-06-19 14:00:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2015-03-29 16:07:17 +00:00
|
|
|
* @depends testSymlink
|
|
|
|
*/
|
2015-06-21 00:33:19 +00:00
|
|
|
public function testStatLstatCache($sftp)
|
|
|
|
{
|
|
|
|
$stat = $sftp->stat('symlink');
|
|
|
|
$lstat = $sftp->lstat('symlink');
|
|
|
|
$this->assertNotEquals(
|
2015-07-15 01:52:31 +00:00
|
|
|
$stat,
|
|
|
|
$lstat,
|
2015-06-21 00:33:19 +00:00
|
|
|
'Failed asserting that stat and lstat returned different output for a symlink'
|
|
|
|
);
|
|
|
|
|
|
|
|
return $sftp;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @depends testStatLstatCache
|
|
|
|
*/
|
|
|
|
public function testLinkFile($sftp)
|
|
|
|
{
|
|
|
|
$this->assertTrue(
|
|
|
|
$sftp->is_link('symlink'),
|
|
|
|
'Failed asserting that symlink is a link'
|
|
|
|
);
|
|
|
|
$this->assertTrue(
|
|
|
|
$sftp->is_file('symlink'),
|
|
|
|
'Failed asserting that symlink is a file'
|
|
|
|
);
|
|
|
|
$this->assertFalse(
|
|
|
|
$sftp->is_dir('symlink'),
|
|
|
|
'Failed asserting that symlink is not a directory'
|
|
|
|
);
|
|
|
|
|
|
|
|
return $sftp;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @depends testLinkFile
|
|
|
|
*/
|
2014-06-19 14:00:23 +00:00
|
|
|
public function testReadlink($sftp)
|
|
|
|
{
|
2020-12-12 21:11:04 +00:00
|
|
|
$this->assertIsString(
|
2015-07-15 01:52:31 +00:00
|
|
|
$sftp->readlink('symlink'),
|
2014-06-19 15:42:20 +00:00
|
|
|
'Failed asserting that a symlink\'s target could be read'
|
2014-06-19 14:00:23 +00:00
|
|
|
);
|
2014-06-20 04:57:32 +00:00
|
|
|
|
|
|
|
return $sftp;
|
2014-06-19 14:00:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2015-03-29 16:07:17 +00:00
|
|
|
* @depends testReadlink
|
2015-06-27 03:25:22 +00:00
|
|
|
* @group github716
|
|
|
|
*/
|
|
|
|
public function testStatOnCWD($sftp)
|
|
|
|
{
|
|
|
|
$stat = $sftp->stat('.');
|
2020-12-12 21:11:04 +00:00
|
|
|
$this->assertIsArray(
|
2015-07-15 01:52:31 +00:00
|
|
|
$stat,
|
2015-06-27 12:43:35 +00:00
|
|
|
'Failed asserting that stat on . returns an array'
|
|
|
|
);
|
2015-06-27 03:25:22 +00:00
|
|
|
$lstat = $sftp->lstat('.');
|
2020-12-12 21:11:04 +00:00
|
|
|
$this->assertIsArray(
|
2015-07-15 01:52:31 +00:00
|
|
|
$lstat,
|
2015-06-27 12:43:35 +00:00
|
|
|
'Failed asserting that lstat on . returns an array'
|
2015-06-27 03:25:22 +00:00
|
|
|
);
|
|
|
|
|
|
|
|
return $sftp;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* on older versions this would result in a fatal error
|
2022-02-17 02:25:59 +00:00
|
|
|
*
|
2015-06-27 03:25:22 +00:00
|
|
|
* @depends testStatOnCWD
|
2015-03-29 16:07:17 +00:00
|
|
|
* @group github402
|
|
|
|
*/
|
2014-07-17 16:22:59 +00:00
|
|
|
public function testStatcacheFix($sftp)
|
|
|
|
{
|
2014-07-18 21:37:36 +00:00
|
|
|
// Name used for both directory and file.
|
|
|
|
$name = 'stattestdir';
|
|
|
|
$this->assertTrue($sftp->mkdir($name));
|
|
|
|
$this->assertTrue($sftp->is_dir($name));
|
|
|
|
$this->assertTrue($sftp->chdir($name));
|
|
|
|
$this->assertStringEndsWith(self::$scratchDir . '/' . $name, $sftp->pwd());
|
|
|
|
$this->assertFalse($sftp->file_exists($name));
|
|
|
|
$this->assertTrue($sftp->touch($name));
|
|
|
|
$this->assertTrue($sftp->is_file($name));
|
|
|
|
$this->assertTrue($sftp->chdir('..'));
|
|
|
|
$this->assertStringEndsWith(self::$scratchDir, $sftp->pwd());
|
|
|
|
$this->assertTrue($sftp->is_dir($name));
|
|
|
|
$this->assertTrue($sftp->is_file("$name/$name"));
|
|
|
|
$this->assertTrue($sftp->delete($name, true));
|
2014-07-17 16:35:58 +00:00
|
|
|
|
|
|
|
return $sftp;
|
2014-07-17 16:22:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2015-03-29 16:07:17 +00:00
|
|
|
* @depends testStatcacheFix
|
|
|
|
*/
|
2014-04-02 14:23:22 +00:00
|
|
|
public function testChDirUpHome($sftp)
|
|
|
|
{
|
|
|
|
$this->assertTrue(
|
|
|
|
$sftp->chdir('../'),
|
|
|
|
'Failed asserting that directory could be changed one level up.'
|
|
|
|
);
|
|
|
|
|
|
|
|
$this->assertEquals(
|
|
|
|
$this->getEnv('SSH_HOME'),
|
|
|
|
$sftp->pwd(),
|
|
|
|
'Failed asserting that pwd() returns home directory.'
|
|
|
|
);
|
|
|
|
|
|
|
|
return $sftp;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2015-03-29 16:07:17 +00:00
|
|
|
* @depends testChDirUpHome
|
|
|
|
*/
|
2014-05-13 23:11:19 +00:00
|
|
|
public function testFileExistsIsFileIsDirDir($sftp)
|
|
|
|
{
|
|
|
|
$this->assertTrue(
|
|
|
|
$sftp->file_exists(self::$scratchDir),
|
|
|
|
'Failed asserting that file_exists() on scratch dir returns true.'
|
|
|
|
);
|
|
|
|
|
|
|
|
$this->assertFalse(
|
|
|
|
$sftp->is_file(self::$scratchDir),
|
|
|
|
'Failed asserting that is_file() on example file returns false.'
|
|
|
|
);
|
|
|
|
|
|
|
|
$this->assertTrue(
|
|
|
|
$sftp->is_dir(self::$scratchDir),
|
|
|
|
'Failed asserting that is_dir() on example file returns true.'
|
|
|
|
);
|
|
|
|
|
|
|
|
return $sftp;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2015-03-29 16:07:17 +00:00
|
|
|
* @depends testFileExistsIsFileIsDirDir
|
|
|
|
*/
|
2014-10-14 16:17:14 +00:00
|
|
|
public function testTruncateLargeFile($sftp)
|
|
|
|
{
|
|
|
|
$filesize = (4 * 1024 + 16) * 1024 * 1024;
|
|
|
|
$filename = 'file-large-from-truncate-4112MiB.txt';
|
|
|
|
$this->assertTrue($sftp->touch($filename));
|
|
|
|
$this->assertTrue($sftp->truncate($filename, $filesize));
|
2020-02-12 05:01:18 +00:00
|
|
|
$this->assertSame($filesize, $sftp->filesize($filename));
|
2014-10-14 16:17:14 +00:00
|
|
|
|
|
|
|
return $sftp;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2015-03-29 16:07:17 +00:00
|
|
|
* @depends testTruncateLargeFile
|
|
|
|
*/
|
2014-04-02 14:23:22 +00:00
|
|
|
public function testRmDirScratch($sftp)
|
|
|
|
{
|
|
|
|
$this->assertFalse(
|
|
|
|
$sftp->rmdir(self::$scratchDir),
|
|
|
|
'Failed asserting that non-empty scratch directory could ' .
|
|
|
|
'not be deleted using rmdir().'
|
|
|
|
);
|
|
|
|
|
|
|
|
return $sftp;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2015-03-29 16:07:17 +00:00
|
|
|
* @depends testRmDirScratch
|
|
|
|
*/
|
2014-04-02 14:23:22 +00:00
|
|
|
public function testDeleteRecursiveScratch($sftp)
|
|
|
|
{
|
|
|
|
$this->assertTrue(
|
|
|
|
$sftp->delete(self::$scratchDir),
|
|
|
|
'Failed asserting that non-empty scratch directory could ' .
|
|
|
|
'be deleted using recursive delete().'
|
|
|
|
);
|
|
|
|
|
|
|
|
return $sftp;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2015-03-29 16:07:17 +00:00
|
|
|
* @depends testDeleteRecursiveScratch
|
|
|
|
*/
|
2014-04-02 14:23:22 +00:00
|
|
|
public function testRmDirScratchNonexistent($sftp)
|
|
|
|
{
|
|
|
|
$this->assertFalse(
|
|
|
|
$sftp->rmdir(self::$scratchDir),
|
|
|
|
'Failed asserting that nonexistent scratch directory could ' .
|
|
|
|
'not be deleted using rmdir().'
|
|
|
|
);
|
2015-07-04 17:57:50 +00:00
|
|
|
|
|
|
|
return $sftp;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @depends testRmDirScratchNonexistent
|
|
|
|
* @group github706
|
|
|
|
*/
|
|
|
|
public function testDeleteEmptyDir($sftp)
|
|
|
|
{
|
|
|
|
$this->assertTrue(
|
|
|
|
$sftp->mkdir(self::$scratchDir),
|
|
|
|
'Failed asserting that scratch directory could ' .
|
|
|
|
'be created.'
|
|
|
|
);
|
2020-12-12 21:11:04 +00:00
|
|
|
$this->assertIsArray(
|
2015-07-04 17:57:50 +00:00
|
|
|
$sftp->stat(self::$scratchDir),
|
2018-03-03 12:07:14 +00:00
|
|
|
'Failed asserting that stat on an existent empty directory returns an array'
|
2015-07-04 17:57:50 +00:00
|
|
|
);
|
|
|
|
$this->assertTrue(
|
|
|
|
$sftp->delete(self::$scratchDir),
|
|
|
|
'Failed asserting that empty scratch directory could ' .
|
|
|
|
'be deleted using recursive delete().'
|
|
|
|
);
|
|
|
|
$this->assertFalse(
|
|
|
|
$sftp->stat(self::$scratchDir),
|
|
|
|
'Failed asserting that stat on a deleted directory returns false'
|
|
|
|
);
|
2015-07-14 20:23:04 +00:00
|
|
|
|
|
|
|
return $sftp;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @depends testDeleteEmptyDir
|
|
|
|
* @group github735
|
|
|
|
*/
|
|
|
|
public function testStatVsLstat($sftp)
|
|
|
|
{
|
|
|
|
$this->assertTrue($sftp->mkdir(self::$scratchDir));
|
|
|
|
$this->assertTrue($sftp->chdir(self::$scratchDir));
|
|
|
|
$this->assertTrue($sftp->put('text.txt', 'zzzzz'));
|
|
|
|
$this->assertTrue($sftp->symlink('text.txt', 'link.txt'));
|
|
|
|
$this->assertTrue($sftp->mkdir('subdir'));
|
|
|
|
$this->assertTrue($sftp->symlink('subdir', 'linkdir'));
|
|
|
|
|
|
|
|
$sftp->clearStatCache();
|
|
|
|
|
|
|
|
// pre-populate the stat cache
|
|
|
|
$sftp->nlist();
|
|
|
|
|
|
|
|
$stat = $sftp->stat('link.txt');
|
|
|
|
$this->assertSame($stat['type'], NET_SFTP_TYPE_REGULAR);
|
|
|
|
$stat = $sftp->lstat('link.txt');
|
|
|
|
$this->assertSame($stat['type'], NET_SFTP_TYPE_SYMLINK);
|
|
|
|
|
|
|
|
$stat = $sftp->stat('linkdir');
|
|
|
|
$this->assertSame($stat['type'], NET_SFTP_TYPE_DIRECTORY);
|
|
|
|
$stat = $sftp->lstat('link.txt');
|
|
|
|
$this->assertSame($stat['type'], NET_SFTP_TYPE_SYMLINK);
|
|
|
|
|
|
|
|
$sftp->disableStatCache();
|
|
|
|
|
|
|
|
$sftp->nlist();
|
|
|
|
|
|
|
|
$stat = $sftp->stat('link.txt');
|
|
|
|
$this->assertSame($stat['type'], NET_SFTP_TYPE_REGULAR);
|
|
|
|
$stat = $sftp->lstat('link.txt');
|
|
|
|
$this->assertSame($stat['type'], NET_SFTP_TYPE_SYMLINK);
|
|
|
|
|
|
|
|
$stat = $sftp->stat('linkdir');
|
|
|
|
$this->assertSame($stat['type'], NET_SFTP_TYPE_DIRECTORY);
|
|
|
|
$stat = $sftp->lstat('link.txt');
|
|
|
|
$this->assertSame($stat['type'], NET_SFTP_TYPE_SYMLINK);
|
|
|
|
|
|
|
|
$sftp->enableStatCache();
|
2016-04-10 16:30:59 +00:00
|
|
|
|
|
|
|
return $sftp;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @depends testStatVsLstat
|
|
|
|
* @group github830
|
|
|
|
*/
|
|
|
|
public function testUploadOffsets($sftp)
|
|
|
|
{
|
|
|
|
$sftp->put('offset.txt', 'res.txt', SFTP::SOURCE_LOCAL_FILE, 0, 10);
|
|
|
|
$this->assertSame(
|
|
|
|
substr(self::$exampleData, 10),
|
|
|
|
$sftp->get('offset.txt'),
|
|
|
|
'Failed asserting that portions of a file could be uploaded.'
|
|
|
|
);
|
|
|
|
|
|
|
|
$sftp->put('offset.txt', 'res.txt', SFTP::SOURCE_LOCAL_FILE, self::$exampleDataLength - 100);
|
|
|
|
$this->assertSame(
|
|
|
|
substr(self::$exampleData, 10, -90) . self::$exampleData,
|
|
|
|
$sftp->get('offset.txt'),
|
|
|
|
'Failed asserting that you could upload into the middle of a file.'
|
|
|
|
);
|
2016-05-05 15:54:29 +00:00
|
|
|
|
|
|
|
return $sftp;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @depends testUploadOffsets
|
|
|
|
*/
|
|
|
|
public function testReadableWritable($sftp)
|
|
|
|
{
|
|
|
|
$sftp->chmod(0000, 'offset.txt');
|
|
|
|
$this->assertFalse($sftp->is_writable('offset.txt'));
|
|
|
|
$this->assertFalse($sftp->is_writeable('offset.txt'));
|
|
|
|
$this->assertFalse($sftp->is_readable('offset.txt'));
|
|
|
|
|
|
|
|
$sftp->chmod(0777, 'offset.txt');
|
|
|
|
$this->assertTrue($sftp->is_writable('offset.txt'));
|
|
|
|
$this->assertTrue($sftp->is_writeable('offset.txt'));
|
|
|
|
$this->assertTrue($sftp->is_readable('offset.txt'));
|
|
|
|
|
|
|
|
$this->assertFalse($sftp->is_writable('nonexistantfile.ext'));
|
|
|
|
$this->assertFalse($sftp->is_writeable('nonexistantfile.ext'));
|
|
|
|
$this->assertFalse($sftp->is_readable('nonexistantfile.ext'));
|
2016-06-29 05:18:35 +00:00
|
|
|
|
|
|
|
return $sftp;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @depends testReadableWritable
|
|
|
|
* @group github999
|
|
|
|
*/
|
|
|
|
public function testExecNlist($sftp)
|
|
|
|
{
|
|
|
|
$sftp->enablePTY();
|
|
|
|
$sftp->exec('ping google.com -c 5');
|
|
|
|
sleep(5);
|
|
|
|
$sftp->nlist();
|
2018-08-13 15:28:33 +00:00
|
|
|
|
2021-10-13 01:35:38 +00:00
|
|
|
$this->assertTrue(true);
|
|
|
|
|
2018-08-13 15:28:33 +00:00
|
|
|
return $sftp;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @depends testExecNlist
|
|
|
|
*/
|
|
|
|
public function testRawlistDisabledStatCache($sftp)
|
|
|
|
{
|
|
|
|
$this->assertTrue($sftp->mkdir(self::$scratchDir));
|
|
|
|
$this->assertTrue($sftp->chdir(self::$scratchDir));
|
|
|
|
$this->assertTrue($sftp->put('text.txt', 'zzzzz'));
|
|
|
|
$this->assertTrue($sftp->mkdir('subdir'));
|
|
|
|
$this->assertTrue($sftp->chdir('subdir'));
|
|
|
|
$this->assertTrue($sftp->put('leaf.txt', 'yyyyy'));
|
|
|
|
$this->assertTrue($sftp->chdir('../../'));
|
|
|
|
|
|
|
|
$list_cache_enabled = $sftp->rawlist('.', true);
|
|
|
|
|
|
|
|
$sftp->clearStatCache();
|
|
|
|
|
|
|
|
$sftp->disableStatCache();
|
|
|
|
|
|
|
|
$list_cache_disabled = $sftp->rawlist('.', true);
|
|
|
|
|
2022-02-17 02:25:59 +00:00
|
|
|
$this->assertEquals(
|
|
|
|
$list_cache_enabled,
|
|
|
|
$list_cache_disabled,
|
2022-03-09 00:59:30 +00:00
|
|
|
'The files should be the same regardless of stat cache'
|
2022-02-17 02:25:59 +00:00
|
|
|
);
|
2022-01-08 15:57:10 +00:00
|
|
|
|
|
|
|
return $sftp;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @depends testRawlistDisabledStatCache
|
|
|
|
*/
|
|
|
|
public function testChownChgrp($sftp)
|
|
|
|
{
|
|
|
|
$stat = $sftp->stat(self::$scratchDir);
|
|
|
|
$this->assertTrue($sftp->chown(self::$scratchDir, $stat['uid']));
|
|
|
|
$this->assertTrue($sftp->chgrp(self::$scratchDir, $stat['gid']));
|
|
|
|
|
|
|
|
$sftp->clearStatCache();
|
|
|
|
$stat2 = $sftp->stat(self::$scratchDir);
|
|
|
|
$this->assertSame($stat['uid'], $stat2['uid']);
|
|
|
|
$this->assertSame($stat['gid'], $stat2['gid']);
|
2014-04-02 14:23:22 +00:00
|
|
|
}
|
2014-06-09 14:44:27 +00:00
|
|
|
}
|