*/ class Mage_Connect_Ftp { /** * Connection object * * @var resource */ protected $_conn = false; /** * Check connected, throw exception if not * * @throws Exception * @return null */ protected function checkConnected() { if(!$this->_conn) { throw new Exception(__CLASS__." - no connection established with server"); } } /** * ftp_mkdir wrapper * * @param string $name * @return string */ public function mdkir($name) { $this->checkConnected(); return @ftp_mkdir($this->_conn, $name); } /** * Make dir recursive * * @param string $path * @param int $mode */ public function mkdirRecursive($path, $mode = 0777) { $this->checkConnected(); $dir = explode('/', $path); $path= ""; $ret = true; for ($i=0; $i < count($dir); $i++) { $path .= "/" .$dir[$i]; if(!@ftp_chdir($this->_conn, $path)) { @ftp_chdir($this->_conn,"/"); if(!@ftp_mkdir($this->_conn,$path)) { $ret=false; break; } else { @ftp_chmod($this->_conn, $mode, $path); } } } return $ret; } /** * Try to login to server * * @param string $login * @param string $password * @throws Exception on invalid login credentials * @return boolean */ public function login($login = "anonymous", $password = "test@gmail.com") { $this->checkConnected(); $res = @ftp_login($this->_conn, $login, $password); if(!$res) { throw new Exception("Invalid login credentials"); } return $res; } /** * Validate connection string * * @param string $string * @throws Exception * @return string */ public function validateConnectionString($string) { if (empty($string)) { throw new Exception("Connection string is empty"); } $data = @parse_url($string); if(false === $data) { throw new Exception("Connection string invalid: '{$string}'"); } if($data['scheme'] != 'ftp') { throw new Exception("Support for scheme '{$data['scheme']}' unsupported"); } return $data; } /** * Connect to server using connect string * Connection string: ftp://user:pass@server:port/path * user,pass,port,path are optional parts * * @param string $string * @param int $timeout * @return null */ public function connect($string, $timeout = 90) { $params = $this->validateConnectionString($string); $port = isset($params['port']) ? intval($params['port']) : 21; $this->_conn = @ftp_connect($params['host'], $port, $timeout); if(!$this->_conn) { throw new Exception("Cannot connect to host: {$params['host']}"); } ftp_pasv($this->_conn, true); if(isset($params['user']) && isset($params['pass'])) { $this->login($params['user'], $params['pass']); } else { $this->login(); } if(isset($params['path'])) { if(!$this->chdir($params['path'])) { throw new Exception ("Cannot chdir after login to: {$params['path']}"); } } } /** * ftp_fput wrapper * * @param string $remoteFile * @param resource $handle * @param int $mode FTP_BINARY | FTP_ASCII * @param int $startPos * @return boolean */ public function fput($remoteFile, $handle, $mode = FTP_BINARY, $startPos = 0) { $this->checkConnected(); return @ftp_fput($this->_conn, $remoteFile, $handle, $mode, $startPos); } /** * ftp_put wrapper * * @param string $remoteFile * @param string $localFile * @param int $mode FTP_BINARY | FTP_ASCII * @param int $startPos * @return boolean */ public function put($remoteFile, $localFile, $mode = FTP_BINARY, $startPos = 0) { $this->checkConnected(); return @ftp_put($this->_conn, $remoteFile, $localFile, $mode, $startPos); } /** * Get current working directory * * @return string */ public function getcwd() { $d = $this->raw("pwd"); $data = explode(" ", $d[0], 3); if(empty($data[1])) { return false; } if(intval($data[0]) != 257) { return false; } $out = trim($data[1], '"'); if($out !== "/") { $out = rtrim($out, "/"); } return $out; } /** * ftp_raw wrapper * * @param string $cmd * @return array */ public function raw($cmd) { $this->checkConnected(); return @ftp_raw($this->_conn, $cmd); } /** * Upload local file to remote server. * Can be used for relative and absoulte remote paths * Relative: use chdir before calling this * * @param string $remote * @param string $local * @param int $dirMode * @param int $fileMode * @return boolean */ public function upload($remote, $local, $dirMode = 0777, $fileMode=0) { $this->checkConnected(); if(!file_exists($local)) { throw new Exception("Local file doesn't exist: {$local}"); } if(!is_readable($local)) { throw new Exception("Local file is not readable: {$local}"); } if(is_dir($local)) { throw new Exception("Directory given instead of file: {$local}"); } $globalPathMode = substr($remote, 0, 1) == "/"; $dirname = dirname($remote); $cwd = $this->getcwd(); if(false === $cwd) { throw new Exception("Server returns something awful on PWD command"); } if(!$globalPathMode) { $dirname = $cwd."/".$dirname; $remote = $cwd."/".$remote; } $res = $this->mkdirRecursive($dirname, $dirMode); $this->chdir($cwd); if(!$res) { return false; } $res = $this->put($remote, $local); if(!$res) { return false; } if($fileMode){ $res=$this->chmod($fileMode, $remote); } return (boolean)$res; } /** * Download remote file to local machine * * @param string $remote * @param string $local * @param int $ftpMode FTP_BINARY|FTP_ASCII * @return boolean */ public function download($remote, $local, $ftpMode = FTP_BINARY) { $this->checkConnected(); return $this->get($local, $remote, $ftpMode); } /** * ftp_pasv wrapper * * @param boolean $pasv * @return boolean */ public function pasv($pasv) { $this->checkConnected(); return @ftp_pasv($this->_conn, (boolean) $pasv); } /** * Close FTP connection * * @return null */ public function close() { if($this->_conn) { @ftp_close($this->_conn); } } /** * ftp_chmod wrapper * * @param $mode * @param $remoteFile * @return boolean */ public function chmod($mode, $remoteFile) { $this->checkConnected(); return @ftp_chmod($this->_conn, $mode, $remoteFile); } /** * ftp_chdir wrapper * * @param string $dir * @return boolean */ public function chdir($dir) { $this->checkConnected(); return @ftp_chdir($this->_conn, $dir); } /** * ftp_cdup wrapper * * @return boolean */ public function cdup() { $this->checkConnected(); return @ftp_cdup($this->_conn); } /** * ftp_get wrapper * * @param string $localFile * @param string $remoteFile * @param int $fileMode FTP_BINARY | FTP_ASCII * @param int $resumeOffset * @return boolean */ public function get($localFile, $remoteFile, $fileMode = FTP_BINARY, $resumeOffset = 0) { $remoteFile = $this->correctFilePath($remoteFile); $this->checkConnected(); return @ftp_get($this->_conn, $localFile, $remoteFile, $fileMode, $resumeOffset); } /** * ftp_nlist wrapper * * @param string $dir * @return boolean */ public function nlist($dir = "/") { $this->checkConnected(); $dir = $this->correctFilePath($dir); return @ftp_nlist($this->_conn, $dir); } /** * ftp_rawlist wrapper * * @param string $dir * @param boolean $recursive * @return array */ public function rawlist( $dir = "/", $recursive = false ) { $this->checkConnected(); $dir = $this->correctFilePath($dir); return @ftp_rawlist($this->_conn, $dir, $recursive); } /** * Convert byte count to float KB/MB format * * @param int $bytes * @return string */ public static function byteconvert($bytes) { $symbol = array('B', 'KB', 'MB', 'GB', 'TB', 'PB', 'EB', 'ZB', 'YB'); $exp = floor( log($bytes) / log(1024) ); return sprintf( '%.2f ' . $symbol[ $exp ], ($bytes / pow(1024, floor($exp))) ); } /** * Chmod string "-rwxrwxrwx" to "777" converter * * @param string $chmod * @return string */ public static function chmodnum($chmod) { $trans = array('-' => '0', 'r' => '4', 'w' => '2', 'x' => '1'); $chmod = substr(strtr($chmod, $trans), 1); $array = str_split($chmod, 3); return array_sum(str_split($array[0])) . array_sum(str_split($array[1])) . array_sum(str_split($array[2])); } /** * Checks file exists * * @param string $path * @param boolean $excludeIfIsDir * @return boolean */ public function fileExists($path, $excludeIfIsDir = true) { $path = $this->correctFilePath($path); $globalPathMode = substr($path, 0, 1) == "/"; $file = basename($path); $dir = $globalPathMode ? dirname($path) : $this->getcwd()."/".($path==basename($path)?'':$path); $data = $this->ls($dir); foreach($data as $row) { if($file == basename($row['name'])) { if($excludeIfIsDir && $row['dir']) { continue; } return true; } } return false; } /** * Get directory contents in PHP array * * @param string $dir * @param boolean $recursive * @return array */ public function ls($dir = "/", $recursive = false) { $dir= $this->correctFilePath($dir); $rawfiles = (array) $this->rawlist($dir, $recursive); $structure = array(); $arraypointer = &$structure; foreach ($rawfiles as $rawfile) { if ($rawfile[0] == '/') { $paths = array_slice(explode('/', str_replace(':', '', $rawfile)), 1); $arraypointer = &$structure; foreach ($paths as $path) { foreach ($arraypointer as $i => $file) { if ($file['name'] == $path) { $arraypointer = &$arraypointer[ $i ]['children']; break; } } } } elseif(!empty($rawfile)) { $info = preg_split("/[\s]+/", $rawfile, 9); $arraypointer[] = array( 'name' => $info[8], 'dir' => $info[0]{0} == 'd', 'size' => (int) $info[4], 'chmod' => self::chmodnum($info[0]), 'rawdata' => $info, 'raw' => $rawfile ); } } return $structure; } /** * Correct file path * * @param $str * @return string */ public function correctFilePath($str) { $str = str_replace("\\", "/", $str); $str = preg_replace("/^.\//", "", $str); return $str; } /** * Delete file * * @param string $file * @return boolean */ public function delete($file) { $this->checkConnected(); $file = $this->correctFilePath($file); return @ftp_delete($this->_conn, $file); } /** * Remove directory * * @param string $dir * @return boolean */ public function rmdir($dir) { $this->checkConnected(); $dir = $this->correctFilePath($dir); return @ftp_rmdir($this->_conn, $dir); } }