use a persistent connection instead of running a new smbclient for each command

This commit is contained in:
Robin Appelman 2012-12-16 02:33:20 +01:00
commit e76f9974d1
10 changed files with 104 additions and 52 deletions

View file

@ -9,18 +9,16 @@
namespace SMB\Command;
abstract class Command {
const CLIENT = 'smbclient';
/**
* @var \SMB\Share $connection
*/
protected $share;
/**
* @var \SMB\Connection $connection
* @param \SMB\Share $connection
*/
protected $connection;
/**
* @param \SMB\Connection $connection
*/
public function __construct($connection) {
$this->connection = $connection;
public function __construct($share) {
$this->share = $share;
}
/**
@ -28,9 +26,8 @@ abstract class Command {
* @return array
*/
protected function execute($command) {
$auth = $this->escape($this->connection->getAuthString());
$command = self::CLIENT . ' -N -U ' . $auth . ' ' . $command . ' 2> /dev/null';
exec($command, $output);
$this->share->write($command . PHP_EOL);
$output = $this->share->read();
return $output;
}
@ -58,4 +55,12 @@ abstract class Command {
$path = str_replace('/', '\\', $path);
return '"' . trim(escapeshellarg($path), "'") . '"';
}
/**
* @param string $path
* @return string
*/
public function escapeLocalPath($path) {
return '"' . trim(escapeshellarg($path), "'") . '"';
}
}

View file

@ -8,15 +8,17 @@
namespace SMB\Command;
class Dir extends Simple {
class Dir extends Command {
public function __construct($connection) {
parent::__construct($connection);
$this->command = 'cd';
}
public function run($arguments) {
$arguments['postfix']=' ; dir';
return parent::run($arguments);
$path = $this->escapePath($arguments['path']);
$this->execute('cd ' . $path);
$output = $this->execute('dir');
$this->execute('cd /');
return $this->parseOutput($output);
}
/**

View file

@ -20,10 +20,7 @@ abstract class Double extends Command {
public function run($arguments) {
$path1 = $this->escapePath($arguments['path1']);
$path2 = $this->escapePath($arguments['path2']);
$share = $arguments['share'];
$postFix = (isset($arguments['postfix'])) ? $arguments['postfix'] : '';
$cmd = $this->escape('//' . $this->connection->getHost() . '/' . $share);
$cmd .= " -c '" . $this->command . ' ' . $path1 . ' ' . $path2 . $postFix . "'";
$cmd = $this->command . ' ' . $path1 . ' ' . $path2;
$output = $this->execute($cmd);
return $this->parseOutput($output);
}

View file

@ -9,19 +9,10 @@
namespace SMB\Command;
class Get extends Command {
public function __construct($connection) {
parent::__construct($connection);
$this->command = 'get';
}
public function run($arguments) {
$path1 = $this->escapePath($arguments['path1']);
$path2 = $this->escape($arguments['path2']); //seccond path is local, needs different escaping
$share = $arguments['share'];
$postFix = (isset($arguments['postfix'])) ? $arguments['postfix'] : '';
$cmd = $this->escape('//' . $this->connection->getHost() . '/' . $share);
$cmd .= " -c '" . $this->command . ' ' . $path1 . ' ' . $path2 . $postFix . "'";
$output = $this->execute($cmd);
$path2 = $this->escapeLocalPath($arguments['path2']); //second path is local, needs different escaping
$output = $this->execute('get ' . $path1 . ' ' . $path2);
return $this->parseOutput($output);
}

View file

@ -10,7 +10,9 @@ namespace SMB\Command;
class ListShares extends Command {
public function run($arguments) {
$output = $this->execute('-gL ' . $this->escape($this->connection->getHost()));
$auth = $this->escape($this->connection->getAuthString());
$command = self::CLIENT . ' -N -U ' . $auth . ' ' . '-gL ' . $this->escape($this->connection->getHost()) . ' 2> /dev/null';
exec($command, $output);
return $this->parseOutput($output);
}

View file

@ -9,19 +9,10 @@
namespace SMB\Command;
class Put extends Command {
public function __construct($connection) {
parent::__construct($connection);
$this->command = 'put';
}
public function run($arguments) {
$path1 = $this->escape($arguments['path1']); //first path is local, needs different escaping
$path1 = $this->escapeLocalPath($arguments['path1']); //first path is local, needs different escaping
$path2 = $this->escapePath($arguments['path2']);
$share = $arguments['share'];
$postFix = (isset($arguments['postfix'])) ? $arguments['postfix'] : '';
$cmd = $this->escape('//' . $this->connection->getHost() . '/' . $share);
$cmd .= " -c '" . $this->command . ' ' . $path1 . ' ' . $path2 . $postFix . "'";
$output = $this->execute($cmd);
$output = $this->execute('put ' . $path1 . ' ' . $path2);
return $this->parseOutput($output);
}

View file

@ -19,10 +19,7 @@ abstract class Simple extends Command {
public function run($arguments) {
$path = $this->escapePath($arguments['path']);
$share = $arguments['share'];
$postFix = (isset($arguments['postfix'])) ? $arguments['postfix'] : '';
$cmd = $this->escape('//' . $this->connection->getHost() . '/' . $share);
$cmd .= " -c '" . $this->command . ' ' . $path . $postFix . "'";
$cmd = $this->command . ' ' . $path;
$output = $this->execute($cmd);
return $this->parseOutput($output);
}

View file

@ -9,6 +9,8 @@
namespace SMB;
class Connection {
const CLIENT = 'smbclient';
/**
* @var string $host
*/

View file

@ -16,3 +16,6 @@ class AlreadyExistsException extends \Exception {
class NotEmptyException extends \Exception {
}
class ConnectionError extends \Exception {
}

View file

@ -14,6 +14,19 @@ class Share {
*/
private $connection;
/**
* @var resource $process
*/
private $process;
/**
* @var resource[] $pipes
*
* $pipes[0] holds STDIN for smbclient
* $pipes[1] holds STDOUT for smbclient
*/
private $pipes;
/**
* @var string $name
*/
@ -26,6 +39,27 @@ class Share {
public function __construct($connection, $name) {
$this->connection = $connection;
$this->name = $name;
$descriptorSpec = array(
0 => array("pipe", "r"),
1 => array("pipe", "w"),
// 1 => array("file", "/tmp/smbout", 'a'),
2 => array("file", "/tmp/smberror", "a")
);
$command = Connection::CLIENT . ' -N -U ' . $this->connection->getAuthString() .
' //' . $this->connection->getHost() . '/' . $this->name;
$this->process = proc_open($command, $descriptorSpec, $this->pipes, null, array(
'CLI_FORCE_INTERACTIVE' => 'y' // Needed or the prompt isn't displayed!!
));
if (!is_resource($this->process)) {
throw new ConnectionError();
}
// stream_set_blocking($this->pipes[1], 0);
}
public function __destruct() {
proc_close($this->process);
}
/**
@ -35,7 +69,7 @@ class Share {
* @return array
*/
public function dir($path) {
return (new Command\Dir($this->connection))->run(array('path' => $path, 'share' => $this->name));
return (new Command\Dir($this))->run(array('path' => $path));
}
/**
@ -45,7 +79,7 @@ class Share {
* @return bool
*/
public function mkdir($path) {
return (new Command\Mkdir($this->connection))->run(array('path' => $path, 'share' => $this->name));
return (new Command\Mkdir($this))->run(array('path' => $path));
}
/**
@ -55,7 +89,7 @@ class Share {
* @return bool
*/
public function rmdir($path) {
return (new Command\Rmdir($this->connection))->run(array('path' => $path, 'share' => $this->name));
return (new Command\Rmdir($this))->run(array('path' => $path));
}
/**
@ -65,7 +99,7 @@ class Share {
* @return bool
*/
public function del($path) {
return (new Command\Del($this->connection))->run(array('path' => $path, 'share' => $this->name));
return (new Command\Del($this))->run(array('path' => $path));
}
/**
@ -76,7 +110,7 @@ class Share {
* @return bool
*/
public function rename($from, $to) {
return (new Command\Rename($this->connection))->run(array('path1' => $from, 'path2' => $to, 'share' => $this->name));
return (new Command\Rename($this))->run(array('path1' => $from, 'path2' => $to));
}
/**
@ -87,7 +121,7 @@ class Share {
* @return bool
*/
public function put($source, $target) {
return (new Command\Put($this->connection))->run(array('path1' => $source, 'path2' => $target, 'share' => $this->name));
return (new Command\Put($this))->run(array('path1' => $source, 'path2' => $target));
}
/**
@ -98,6 +132,34 @@ class Share {
* @return bool
*/
public function get($source, $target) {
return (new Command\Get($this->connection))->run(array('path1' => $source, 'path2' => $target, 'share' => $this->name));
return (new Command\Get($this))->run(array('path1' => $source, 'path2' => $target));
}
/**
* send input to smbclient
*
* @param string $input
*/
public function write($input) {
fwrite($this->pipes[0], $input);
fwrite($this->pipes[0], PHP_EOL); //make sure we have a recognizable delimiter
fflush($this->pipes[0]);
}
/**
* get all unprocessed output from smbclient
*
* @return array
*/
public function read() {
fgets($this->pipes[1]);//first line is promt
$output = array();
$line = fgets($this->pipes[1]);
while (substr($line, 0, 4) !== 'smb:') { //next prompt functions as delimiter
$output[] .= $line;
$line = fgets($this->pipes[1]);
}
return $output;
// return explode(PHP_EOL, stream_get_contents($this->pipes[1]));
}
}