SCP.php 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378
  1. <?php
  2. /**
  3. * Pure-PHP implementation of SCP.
  4. *
  5. * PHP versions 4 and 5
  6. *
  7. * The API for this library is modeled after the API from PHP's {@link http://php.net/book.ftp FTP extension}.
  8. *
  9. * Here's a short example of how to use this library:
  10. * <code>
  11. * <?php
  12. * include 'Net/SCP.php';
  13. * include 'Net/SSH2.php';
  14. *
  15. * $ssh = new Net_SSH2('www.domain.tld');
  16. * if (!$ssh->login('username', 'password')) {
  17. * exit('bad login');
  18. * }
  19. *
  20. * $scp = new Net_SCP($ssh);
  21. * $scp->put('abcd', str_repeat('x', 1024*1024));
  22. * ?>
  23. * </code>
  24. *
  25. * LICENSE: Permission is hereby granted, free of charge, to any person obtaining a copy
  26. * of this software and associated documentation files (the "Software"), to deal
  27. * in the Software without restriction, including without limitation the rights
  28. * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
  29. * copies of the Software, and to permit persons to whom the Software is
  30. * furnished to do so, subject to the following conditions:
  31. *
  32. * The above copyright notice and this permission notice shall be included in
  33. * all copies or substantial portions of the Software.
  34. *
  35. * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  36. * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  37. * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  38. * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  39. * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
  40. * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
  41. * THE SOFTWARE.
  42. *
  43. * @category Net
  44. * @package Net_SCP
  45. * @author Jim Wigginton <terrafrost@php.net>
  46. * @copyright 2010 Jim Wigginton
  47. * @license http://www.opensource.org/licenses/mit-license.html MIT License
  48. * @link http://phpseclib.sourceforge.net
  49. */
  50. /**#@+
  51. * @access public
  52. * @see self::put()
  53. */
  54. /**
  55. * Reads data from a local file.
  56. */
  57. define('NET_SCP_LOCAL_FILE', 1);
  58. /**
  59. * Reads data from a string.
  60. */
  61. define('NET_SCP_STRING', 2);
  62. /**#@-*/
  63. /**#@+
  64. * @access private
  65. * @see self::_send()
  66. * @see self::_receive()
  67. */
  68. /**
  69. * SSH1 is being used.
  70. */
  71. define('NET_SCP_SSH1', 1);
  72. /**
  73. * SSH2 is being used.
  74. */
  75. define('NET_SCP_SSH2', 2);
  76. /**#@-*/
  77. /**
  78. * Pure-PHP implementations of SCP.
  79. *
  80. * @package Net_SCP
  81. * @author Jim Wigginton <terrafrost@php.net>
  82. * @access public
  83. */
  84. class Net_SCP
  85. {
  86. /**
  87. * SSH Object
  88. *
  89. * @var object
  90. * @access private
  91. */
  92. var $ssh;
  93. /**
  94. * Packet Size
  95. *
  96. * @var int
  97. * @access private
  98. */
  99. var $packet_size;
  100. /**
  101. * Mode
  102. *
  103. * @var int
  104. * @access private
  105. */
  106. var $mode;
  107. /**
  108. * Default Constructor.
  109. *
  110. * Connects to an SSH server
  111. *
  112. * @param Net_SSH1|Net_SSH2 $ssh
  113. * @return Net_SCP
  114. * @access public
  115. */
  116. function __construct($ssh)
  117. {
  118. if (!is_object($ssh)) {
  119. return;
  120. }
  121. switch (strtolower(get_class($ssh))) {
  122. case 'net_ssh2':
  123. $this->mode = NET_SCP_SSH2;
  124. break;
  125. case 'net_ssh1':
  126. $this->packet_size = 50000;
  127. $this->mode = NET_SCP_SSH1;
  128. break;
  129. default:
  130. return;
  131. }
  132. $this->ssh = $ssh;
  133. }
  134. /**
  135. * PHP4 compatible Default Constructor.
  136. *
  137. * @see self::__construct()
  138. * @param Net_SSH1|Net_SSH2 $ssh
  139. * @access public
  140. */
  141. function Net_SCP($ssh)
  142. {
  143. $this->__construct($ssh);
  144. }
  145. /**
  146. * Uploads a file to the SCP server.
  147. *
  148. * By default, Net_SCP::put() does not read from the local filesystem. $data is dumped directly into $remote_file.
  149. * So, for example, if you set $data to 'filename.ext' and then do Net_SCP::get(), you will get a file, twelve bytes
  150. * long, containing 'filename.ext' as its contents.
  151. *
  152. * Setting $mode to NET_SCP_LOCAL_FILE will change the above behavior. With NET_SCP_LOCAL_FILE, $remote_file will
  153. * contain as many bytes as filename.ext does on your local filesystem. If your filename.ext is 1MB then that is how
  154. * large $remote_file will be, as well.
  155. *
  156. * Currently, only binary mode is supported. As such, if the line endings need to be adjusted, you will need to take
  157. * care of that, yourself.
  158. *
  159. * @param string $remote_file
  160. * @param string $data
  161. * @param int $mode
  162. * @param callable $callback
  163. * @return bool
  164. * @access public
  165. */
  166. function put($remote_file, $data, $mode = NET_SCP_STRING, $callback = null)
  167. {
  168. if (!isset($this->ssh)) {
  169. return false;
  170. }
  171. if (empty($remote_file)) {
  172. user_error('remote_file cannot be blank', E_USER_NOTICE);
  173. return false;
  174. }
  175. if (!$this->ssh->exec('scp -t ' . escapeshellarg($remote_file), false)) { // -t = to
  176. return false;
  177. }
  178. $temp = $this->_receive();
  179. if ($temp !== chr(0)) {
  180. return false;
  181. }
  182. if ($this->mode == NET_SCP_SSH2) {
  183. $this->packet_size = $this->ssh->packet_size_client_to_server[NET_SSH2_CHANNEL_EXEC] - 4;
  184. }
  185. $remote_file = basename($remote_file);
  186. if ($mode == NET_SCP_STRING) {
  187. $size = strlen($data);
  188. } else {
  189. if (!is_file($data)) {
  190. user_error("$data is not a valid file", E_USER_NOTICE);
  191. return false;
  192. }
  193. $fp = @fopen($data, 'rb');
  194. if (!$fp) {
  195. return false;
  196. }
  197. $size = filesize($data);
  198. }
  199. $this->_send('C0644 ' . $size . ' ' . $remote_file . "\n");
  200. $temp = $this->_receive();
  201. if ($temp !== chr(0)) {
  202. return false;
  203. }
  204. $sent = 0;
  205. while ($sent < $size) {
  206. $temp = $mode & NET_SCP_STRING ? substr($data, $sent, $this->packet_size) : fread($fp, $this->packet_size);
  207. $this->_send($temp);
  208. $sent+= strlen($temp);
  209. if (is_callable($callback)) {
  210. call_user_func($callback, $sent);
  211. }
  212. }
  213. $this->_close();
  214. if ($mode != NET_SCP_STRING) {
  215. fclose($fp);
  216. }
  217. return true;
  218. }
  219. /**
  220. * Downloads a file from the SCP server.
  221. *
  222. * Returns a string containing the contents of $remote_file if $local_file is left undefined or a boolean false if
  223. * the operation was unsuccessful. If $local_file is defined, returns true or false depending on the success of the
  224. * operation
  225. *
  226. * @param string $remote_file
  227. * @param string $local_file
  228. * @return mixed
  229. * @access public
  230. */
  231. function get($remote_file, $local_file = false)
  232. {
  233. if (!isset($this->ssh)) {
  234. return false;
  235. }
  236. if (!$this->ssh->exec('scp -f ' . escapeshellarg($remote_file), false)) { // -f = from
  237. return false;
  238. }
  239. $this->_send("\0");
  240. if (!preg_match('#(?<perms>[^ ]+) (?<size>\d+) (?<name>.+)#', rtrim($this->_receive()), $info)) {
  241. return false;
  242. }
  243. $this->_send("\0");
  244. $size = 0;
  245. if ($local_file !== false) {
  246. $fp = @fopen($local_file, 'wb');
  247. if (!$fp) {
  248. return false;
  249. }
  250. }
  251. $content = '';
  252. while ($size < $info['size']) {
  253. $data = $this->_receive();
  254. // SCP usually seems to split stuff out into 16k chunks
  255. $size+= strlen($data);
  256. if ($local_file === false) {
  257. $content.= $data;
  258. } else {
  259. fputs($fp, $data);
  260. }
  261. }
  262. $this->_close();
  263. if ($local_file !== false) {
  264. fclose($fp);
  265. return true;
  266. }
  267. return $content;
  268. }
  269. /**
  270. * Sends a packet to an SSH server
  271. *
  272. * @param string $data
  273. * @access private
  274. */
  275. function _send($data)
  276. {
  277. switch ($this->mode) {
  278. case NET_SCP_SSH2:
  279. $this->ssh->_send_channel_packet(NET_SSH2_CHANNEL_EXEC, $data);
  280. break;
  281. case NET_SCP_SSH1:
  282. $data = pack('CNa*', NET_SSH1_CMSG_STDIN_DATA, strlen($data), $data);
  283. $this->ssh->_send_binary_packet($data);
  284. }
  285. }
  286. /**
  287. * Receives a packet from an SSH server
  288. *
  289. * @return string
  290. * @access private
  291. */
  292. function _receive()
  293. {
  294. switch ($this->mode) {
  295. case NET_SCP_SSH2:
  296. return $this->ssh->_get_channel_packet(NET_SSH2_CHANNEL_EXEC, true);
  297. case NET_SCP_SSH1:
  298. if (!$this->ssh->bitmap) {
  299. return false;
  300. }
  301. while (true) {
  302. $response = $this->ssh->_get_binary_packet();
  303. switch ($response[NET_SSH1_RESPONSE_TYPE]) {
  304. case NET_SSH1_SMSG_STDOUT_DATA:
  305. if (strlen($response[NET_SSH1_RESPONSE_DATA]) < 4) {
  306. return false;
  307. }
  308. extract(unpack('Nlength', $response[NET_SSH1_RESPONSE_DATA]));
  309. return $this->ssh->_string_shift($response[NET_SSH1_RESPONSE_DATA], $length);
  310. case NET_SSH1_SMSG_STDERR_DATA:
  311. break;
  312. case NET_SSH1_SMSG_EXITSTATUS:
  313. $this->ssh->_send_binary_packet(chr(NET_SSH1_CMSG_EXIT_CONFIRMATION));
  314. fclose($this->ssh->fsock);
  315. $this->ssh->bitmap = 0;
  316. return false;
  317. default:
  318. user_error('Unknown packet received', E_USER_NOTICE);
  319. return false;
  320. }
  321. }
  322. }
  323. }
  324. /**
  325. * Closes the connection to an SSH server
  326. *
  327. * @access private
  328. */
  329. function _close()
  330. {
  331. switch ($this->mode) {
  332. case NET_SCP_SSH2:
  333. $this->ssh->_close_channel(NET_SSH2_CHANNEL_EXEC, true);
  334. break;
  335. case NET_SCP_SSH1:
  336. $this->ssh->disconnect();
  337. }
  338. }
  339. }