class.smtp.php 28 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957
  1. <?php
  2. //邮件发送
  3. /**
  4. * PHPMailer - PHP SMTP email transport class
  5. * NOTE: Designed for use with PHP version 5 and up
  6. * @package PHPMailer
  7. * @author Andy Prevost
  8. * @author Marcus Bointon
  9. * @copyright 2004 - 2008 Andy Prevost
  10. * @author Jim Jagielski
  11. * @copyright 2010 - 2012 Jim Jagielski
  12. * @license http://www.gnu.org/copyleft/lesser.html Distributed under the Lesser General Public License (LGPL)
  13. */
  14. /**
  15. * SMTP is rfc 821 compliant and implements all the rfc 821 SMTP
  16. * commands except TURN which will always return a not implemented
  17. * error. SMTP also provides some utility methods for sending mail
  18. * to an SMTP server.
  19. * original author: Chris Ryan
  20. */
  21. class SMTP {
  22. /**
  23. * SMTP server port
  24. * @var int
  25. */
  26. public $SMTP_PORT = 25;
  27. /**
  28. * SMTP reply line ending (don't change)
  29. * @var string
  30. */
  31. public $CRLF = "\r\n";
  32. /**
  33. * Sets whether debugging is turned on
  34. * @var bool
  35. */
  36. public $do_debug; // the level of debug to perform
  37. /**
  38. * Sets the function/method to use for debugging output.
  39. * Right now we only honor "echo" or "error_log"
  40. * @var string
  41. */
  42. public $Debugoutput = "echo";
  43. /**
  44. * Sets VERP use on/off (default is off)
  45. * @var bool
  46. */
  47. public $do_verp = false;
  48. /**
  49. * Sets the SMTP timeout value for reads, in seconds
  50. * @var int
  51. */
  52. public $Timeout = 15;
  53. /**
  54. * Sets the SMTP timelimit value for reads, in seconds
  55. * @var int
  56. */
  57. public $Timelimit = 30;
  58. /**
  59. * Sets the SMTP PHPMailer Version number
  60. * @var string
  61. */
  62. public $Version = '5.2.2-rc1';
  63. /////////////////////////////////////////////////
  64. // PROPERTIES, PRIVATE AND PROTECTED
  65. /////////////////////////////////////////////////
  66. private $smtp_conn; // the socket to the server
  67. private $error; // error if any on the last call
  68. private $helo_rply; // the reply the server sent to us for HELO
  69. /**
  70. * Outputs debugging info via user-defined method
  71. * @param string $str
  72. */
  73. private function edebug($str) {
  74. if ($this->Debugoutput == "error_log") {
  75. error_log($str);
  76. } else {
  77. echo $str;
  78. }
  79. }
  80. /**
  81. * Initialize the class so that the data is in a known state.
  82. * @access public
  83. * @return void
  84. */
  85. public function __construct() {
  86. $this->smtp_conn = 0;
  87. $this->error = null;
  88. $this->helo_rply = null;
  89. $this->do_debug = 0;
  90. }
  91. /////////////////////////////////////////////////
  92. // CONNECTION FUNCTIONS
  93. /////////////////////////////////////////////////
  94. /**
  95. * Connect to the server specified on the port specified.
  96. * If the port is not specified use the default SMTP_PORT.
  97. * If tval is specified then a connection will try and be
  98. * established with the server for that number of seconds.
  99. * If tval is not specified the default is 30 seconds to
  100. * try on the connection.
  101. *
  102. * SMTP CODE SUCCESS: 220
  103. * SMTP CODE FAILURE: 421
  104. * @access public
  105. * @return bool
  106. */
  107. public function Connect($host, $port = 0, $tval = 30) {
  108. // set the error val to null so there is no confusion
  109. $this->error = null;
  110. // make sure we are __not__ connected
  111. if($this->connected()) {
  112. // already connected, generate error
  113. $this->error = array("error" => "Already connected to a server");
  114. return false;
  115. }
  116. if(empty($port)) {
  117. $port = $this->SMTP_PORT;
  118. }
  119. // connect to the smtp server
  120. $this->smtp_conn = @fsockopen($host, // the host of the server
  121. $port, // the port to use
  122. $errno, // error number if any
  123. $errstr, // error message if any
  124. $tval); // give up after ? secs
  125. // verify we connected properly
  126. if(empty($this->smtp_conn)) {
  127. $this->error = array("error" => "Failed to connect to server",
  128. "errno" => $errno,
  129. "errstr" => $errstr);
  130. if($this->do_debug >= 1) {
  131. $this->edebug("SMTP -> ERROR: " . $this->error["error"] . ": $errstr ($errno)" . $this->CRLF . '<br />');
  132. }
  133. return false;
  134. }
  135. // SMTP server can take longer to respond, give longer timeout for first read
  136. // Windows does not have support for this timeout function
  137. if(substr(PHP_OS, 0, 3) != "WIN") {
  138. $max = ini_get('max_execution_time');
  139. if ($max != 0 && $tval > $max) { // don't bother if unlimited
  140. @set_time_limit($tval);
  141. }
  142. stream_set_timeout($this->smtp_conn, $tval, 0);
  143. }
  144. // get any announcement
  145. $announce = $this->get_lines();
  146. if($this->do_debug >= 2) {
  147. $this->edebug("SMTP -> FROM SERVER:" . $announce . $this->CRLF . '<br />');
  148. }
  149. return true;
  150. }
  151. /**
  152. * Initiate a TLS communication with the server.
  153. *
  154. * SMTP CODE 220 Ready to start TLS
  155. * SMTP CODE 501 Syntax error (no parameters allowed)
  156. * SMTP CODE 454 TLS not available due to temporary reason
  157. * @access public
  158. * @return bool success
  159. */
  160. public function StartTLS() {
  161. $this->error = null; # to avoid confusion
  162. if(!$this->connected()) {
  163. $this->error = array("error" => "Called StartTLS() without being connected");
  164. return false;
  165. }
  166. fputs($this->smtp_conn,"STARTTLS" . $this->CRLF);
  167. $rply = $this->get_lines();
  168. $code = substr($rply,0,3);
  169. if($this->do_debug >= 2) {
  170. $this->edebug("SMTP -> FROM SERVER:" . $rply . $this->CRLF . '<br />');
  171. }
  172. if($code != 220) {
  173. $this->error =
  174. array("error" => "STARTTLS not accepted from server",
  175. "smtp_code" => $code,
  176. "smtp_msg" => substr($rply,4));
  177. if($this->do_debug >= 1) {
  178. $this->edebug("SMTP -> ERROR: " . $this->error["error"] . ": " . $rply . $this->CRLF . '<br />');
  179. }
  180. return false;
  181. }
  182. // Begin encrypted connection
  183. if(!stream_socket_enable_crypto($this->smtp_conn, true, STREAM_CRYPTO_METHOD_TLS_CLIENT)) {
  184. return false;
  185. }
  186. return true;
  187. }
  188. /**
  189. * Performs SMTP authentication. Must be run after running the
  190. * Hello() method. Returns true if successfully authenticated.
  191. * @access public
  192. * @return bool
  193. */
  194. public function Authenticate($username, $password, $authtype='LOGIN', $realm='',
  195. $workstation='') {
  196. if (empty($authtype)) {
  197. $authtype = 'LOGIN';
  198. }
  199. switch ($authtype) {
  200. case 'PLAIN':
  201. // Start authentication
  202. fputs($this->smtp_conn,"AUTH PLAIN" . $this->CRLF);
  203. $rply = $this->get_lines();
  204. $code = substr($rply,0,3);
  205. if($code != 334) {
  206. $this->error =
  207. array("error" => "AUTH not accepted from server",
  208. "smtp_code" => $code,
  209. "smtp_msg" => substr($rply,4));
  210. if($this->do_debug >= 1) {
  211. $this->edebug("SMTP -> ERROR: " . $this->error["error"] . ": " . $rply . $this->CRLF . '<br />');
  212. }
  213. return false;
  214. }
  215. // Send encoded username and password
  216. fputs($this->smtp_conn, base64_encode("\0".$username."\0".$password) . $this->CRLF);
  217. $rply = $this->get_lines();
  218. $code = substr($rply,0,3);
  219. if($code != 235) {
  220. $this->error =
  221. array("error" => "Authentication not accepted from server",
  222. "smtp_code" => $code,
  223. "smtp_msg" => substr($rply,4));
  224. if($this->do_debug >= 1) {
  225. $this->edebug("SMTP -> ERROR: " . $this->error["error"] . ": " . $rply . $this->CRLF . '<br />');
  226. }
  227. return false;
  228. }
  229. break;
  230. case 'LOGIN':
  231. // Start authentication
  232. fputs($this->smtp_conn,"AUTH LOGIN" . $this->CRLF);
  233. $rply = $this->get_lines();
  234. $code = substr($rply,0,3);
  235. if($code != 334) {
  236. $this->error =
  237. array("error" => "AUTH not accepted from server",
  238. "smtp_code" => $code,
  239. "smtp_msg" => substr($rply,4));
  240. if($this->do_debug >= 1) {
  241. $this->edebug("SMTP -> ERROR: " . $this->error["error"] . ": " . $rply . $this->CRLF . '<br />');
  242. }
  243. return false;
  244. }
  245. // Send encoded username
  246. fputs($this->smtp_conn, base64_encode($username) . $this->CRLF);
  247. $rply = $this->get_lines();
  248. $code = substr($rply,0,3);
  249. if($code != 334) {
  250. $this->error =
  251. array("error" => "Username not accepted from server",
  252. "smtp_code" => $code,
  253. "smtp_msg" => substr($rply,4));
  254. if($this->do_debug >= 1) {
  255. $this->edebug("SMTP -> ERROR: " . $this->error["error"] . ": " . $rply . $this->CRLF . '<br />');
  256. }
  257. return false;
  258. }
  259. // Send encoded password
  260. fputs($this->smtp_conn, base64_encode($password) . $this->CRLF);
  261. $rply = $this->get_lines();
  262. $code = substr($rply,0,3);
  263. if($code != 235) {
  264. $this->error =
  265. array("error" => "Password not accepted from server",
  266. "smtp_code" => $code,
  267. "smtp_msg" => substr($rply,4));
  268. if($this->do_debug >= 1) {
  269. $this->edebug("SMTP -> ERROR: " . $this->error["error"] . ": " . $rply . $this->CRLF . '<br />');
  270. }
  271. return false;
  272. }
  273. break;
  274. case 'NTLM':
  275. /*
  276. * ntlm_sasl_client.php
  277. ** Bundled with Permission
  278. **
  279. ** How to telnet in windows: http://technet.microsoft.com/en-us/library/aa995718%28EXCHG.65%29.aspx
  280. ** PROTOCOL Documentation http://curl.haxx.se/rfc/ntlm.html#ntlmSmtpAuthentication
  281. */
  282. require_once('ntlm_sasl_client.php');
  283. $temp = new stdClass();
  284. $ntlm_client = new ntlm_sasl_client_class;
  285. if(! $ntlm_client->Initialize($temp)){//let's test if every function its available
  286. $this->error = array("error" => $temp->error);
  287. if($this->do_debug >= 1) {
  288. $this->edebug("You need to enable some modules in your php.ini file: " . $this->error["error"] . $this->CRLF);
  289. }
  290. return false;
  291. }
  292. $msg1 = $ntlm_client->TypeMsg1($realm, $workstation);//msg1
  293. fputs($this->smtp_conn,"AUTH NTLM " . base64_encode($msg1) . $this->CRLF);
  294. $rply = $this->get_lines();
  295. $code = substr($rply,0,3);
  296. if($code != 334) {
  297. $this->error =
  298. array("error" => "AUTH not accepted from server",
  299. "smtp_code" => $code,
  300. "smtp_msg" => substr($rply,4));
  301. if($this->do_debug >= 1) {
  302. $this->edebug("SMTP -> ERROR: " . $this->error["error"] . ": " . $rply . $this->CRLF);
  303. }
  304. return false;
  305. }
  306. $challange = substr($rply,3);//though 0 based, there is a white space after the 3 digit number....//msg2
  307. $challange = base64_decode($challange);
  308. $ntlm_res = $ntlm_client->NTLMResponse(substr($challange,24,8),$password);
  309. $msg3 = $ntlm_client->TypeMsg3($ntlm_res,$username,$realm,$workstation);//msg3
  310. // Send encoded username
  311. fputs($this->smtp_conn, base64_encode($msg3) . $this->CRLF);
  312. $rply = $this->get_lines();
  313. $code = substr($rply,0,3);
  314. if($code != 235) {
  315. $this->error =
  316. array("error" => "Could not authenticate",
  317. "smtp_code" => $code,
  318. "smtp_msg" => substr($rply,4));
  319. if($this->do_debug >= 1) {
  320. $this->edebug("SMTP -> ERROR: " . $this->error["error"] . ": " . $rply . $this->CRLF);
  321. }
  322. return false;
  323. }
  324. break;
  325. }
  326. return true;
  327. }
  328. /**
  329. * Returns true if connected to a server otherwise false
  330. * @access public
  331. * @return bool
  332. */
  333. public function Connected() {
  334. if(!empty($this->smtp_conn)) {
  335. $sock_status = socket_get_status($this->smtp_conn);
  336. if($sock_status["eof"]) {
  337. // the socket is valid but we are not connected
  338. if($this->do_debug >= 1) {
  339. $this->edebug("SMTP -> NOTICE:" . $this->CRLF . "EOF caught while checking if connected");
  340. }
  341. $this->Close();
  342. return false;
  343. }
  344. return true; // everything looks good
  345. }
  346. return false;
  347. }
  348. /**
  349. * Closes the socket and cleans up the state of the class.
  350. * It is not considered good to use this function without
  351. * first trying to use QUIT.
  352. * @access public
  353. * @return void
  354. */
  355. public function Close() {
  356. $this->error = null; // so there is no confusion
  357. $this->helo_rply = null;
  358. if(!empty($this->smtp_conn)) {
  359. // close the connection and cleanup
  360. fclose($this->smtp_conn);
  361. $this->smtp_conn = 0;
  362. }
  363. }
  364. /////////////////////////////////////////////////
  365. // SMTP COMMANDS
  366. /////////////////////////////////////////////////
  367. /**
  368. * Issues a data command and sends the msg_data to the server
  369. * finializing the mail transaction. $msg_data is the message
  370. * that is to be send with the headers. Each header needs to be
  371. * on a single line followed by a <CRLF> with the message headers
  372. * and the message body being seperated by and additional <CRLF>.
  373. *
  374. * Implements rfc 821: DATA <CRLF>
  375. *
  376. * SMTP CODE INTERMEDIATE: 354
  377. * [data]
  378. * <CRLF>.<CRLF>
  379. * SMTP CODE SUCCESS: 250
  380. * SMTP CODE FAILURE: 552,554,451,452
  381. * SMTP CODE FAILURE: 451,554
  382. * SMTP CODE ERROR : 500,501,503,421
  383. * @access public
  384. * @return bool
  385. */
  386. public function Data($msg_data) {
  387. $this->error = null; // so no confusion is caused
  388. if(!$this->connected()) {
  389. $this->error = array(
  390. "error" => "Called Data() without being connected");
  391. return false;
  392. }
  393. fputs($this->smtp_conn,"DATA" . $this->CRLF);
  394. $rply = $this->get_lines();
  395. $code = substr($rply,0,3);
  396. if($this->do_debug >= 2) {
  397. $this->edebug("SMTP -> FROM SERVER:" . $rply . $this->CRLF . '<br />');
  398. }
  399. if($code != 354) {
  400. $this->error =
  401. array("error" => "DATA command not accepted from server",
  402. "smtp_code" => $code,
  403. "smtp_msg" => substr($rply,4));
  404. if($this->do_debug >= 1) {
  405. $this->edebug("SMTP -> ERROR: " . $this->error["error"] . ": " . $rply . $this->CRLF . '<br />');
  406. }
  407. return false;
  408. }
  409. /* the server is ready to accept data!
  410. * according to rfc 821 we should not send more than 1000
  411. * including the CRLF
  412. * characters on a single line so we will break the data up
  413. * into lines by \r and/or \n then if needed we will break
  414. * each of those into smaller lines to fit within the limit.
  415. * in addition we will be looking for lines that start with
  416. * a period '.' and append and additional period '.' to that
  417. * line. NOTE: this does not count towards limit.
  418. */
  419. // normalize the line breaks so we know the explode works
  420. $msg_data = str_replace("\r\n","\n",$msg_data);
  421. $msg_data = str_replace("\r","\n",$msg_data);
  422. $lines = explode("\n",$msg_data);
  423. /* we need to find a good way to determine is headers are
  424. * in the msg_data or if it is a straight msg body
  425. * currently I am assuming rfc 822 definitions of msg headers
  426. * and if the first field of the first line (':' sperated)
  427. * does not contain a space then it _should_ be a header
  428. * and we can process all lines before a blank "" line as
  429. * headers.
  430. */
  431. $field = substr($lines[0],0,strpos($lines[0],":"));
  432. $in_headers = false;
  433. if(!empty($field) && !strstr($field," ")) {
  434. $in_headers = true;
  435. }
  436. $max_line_length = 998; // used below; set here for ease in change
  437. while(list(,$line) = @each($lines)) {
  438. $lines_out = null;
  439. if($line == "" && $in_headers) {
  440. $in_headers = false;
  441. }
  442. // ok we need to break this line up into several smaller lines
  443. while(strlen($line) > $max_line_length) {
  444. $pos = strrpos(substr($line,0,$max_line_length)," ");
  445. // Patch to fix DOS attack
  446. if(!$pos) {
  447. $pos = $max_line_length - 1;
  448. $lines_out[] = substr($line,0,$pos);
  449. $line = substr($line,$pos);
  450. } else {
  451. $lines_out[] = substr($line,0,$pos);
  452. $line = substr($line,$pos + 1);
  453. }
  454. /* if processing headers add a LWSP-char to the front of new line
  455. * rfc 822 on long msg headers
  456. */
  457. if($in_headers) {
  458. $line = "\t" . $line;
  459. }
  460. }
  461. $lines_out[] = $line;
  462. // send the lines to the server
  463. while(list(,$line_out) = @each($lines_out)) {
  464. if(strlen($line_out) > 0)
  465. {
  466. if(substr($line_out, 0, 1) == ".") {
  467. $line_out = "." . $line_out;
  468. }
  469. }
  470. fputs($this->smtp_conn,$line_out . $this->CRLF);
  471. }
  472. }
  473. // message data has been sent
  474. fputs($this->smtp_conn, $this->CRLF . "." . $this->CRLF);
  475. $rply = $this->get_lines();
  476. $code = substr($rply,0,3);
  477. if($this->do_debug >= 2) {
  478. $this->edebug("SMTP -> FROM SERVER:" . $rply . $this->CRLF . '<br />');
  479. }
  480. if($code != 250) {
  481. $this->error =
  482. array("error" => "DATA not accepted from server",
  483. "smtp_code" => $code,
  484. "smtp_msg" => substr($rply,4));
  485. if($this->do_debug >= 1) {
  486. $this->edebug("SMTP -> ERROR: " . $this->error["error"] . ": " . $rply . $this->CRLF . '<br />');
  487. }
  488. return false;
  489. }
  490. return true;
  491. }
  492. /**
  493. * Sends the HELO command to the smtp server.
  494. * This makes sure that we and the server are in
  495. * the same known state.
  496. *
  497. * Implements from rfc 821: HELO <SP> <domain> <CRLF>
  498. *
  499. * SMTP CODE SUCCESS: 250
  500. * SMTP CODE ERROR : 500, 501, 504, 421
  501. * @access public
  502. * @return bool
  503. */
  504. public function Hello($host = '') {
  505. $this->error = null; // so no confusion is caused
  506. if(!$this->connected()) {
  507. $this->error = array(
  508. "error" => "Called Hello() without being connected");
  509. return false;
  510. }
  511. // if hostname for HELO was not specified send default
  512. if(empty($host)) {
  513. // determine appropriate default to send to server
  514. $host = "localhost";
  515. }
  516. // Send extended hello first (RFC 2821)
  517. if(!$this->SendHello("EHLO", $host)) {
  518. if(!$this->SendHello("HELO", $host)) {
  519. return false;
  520. }
  521. }
  522. return true;
  523. }
  524. /**
  525. * Sends a HELO/EHLO command.
  526. * @access private
  527. * @return bool
  528. */
  529. private function SendHello($hello, $host) {
  530. fputs($this->smtp_conn, $hello . " " . $host . $this->CRLF);
  531. $rply = $this->get_lines();
  532. $code = substr($rply,0,3);
  533. if($this->do_debug >= 2) {
  534. $this->edebug("SMTP -> FROM SERVER: " . $rply . $this->CRLF . '<br />');
  535. }
  536. if($code != 250) {
  537. $this->error =
  538. array("error" => $hello . " not accepted from server",
  539. "smtp_code" => $code,
  540. "smtp_msg" => substr($rply,4));
  541. if($this->do_debug >= 1) {
  542. $this->edebug("SMTP -> ERROR: " . $this->error["error"] . ": " . $rply . $this->CRLF . '<br />');
  543. }
  544. return false;
  545. }
  546. $this->helo_rply = $rply;
  547. return true;
  548. }
  549. /**
  550. * Starts a mail transaction from the email address specified in
  551. * $from. Returns true if successful or false otherwise. If True
  552. * the mail transaction is started and then one or more Recipient
  553. * commands may be called followed by a Data command.
  554. *
  555. * Implements rfc 821: MAIL <SP> FROM:<reverse-path> <CRLF>
  556. *
  557. * SMTP CODE SUCCESS: 250
  558. * SMTP CODE SUCCESS: 552,451,452
  559. * SMTP CODE SUCCESS: 500,501,421
  560. * @access public
  561. * @return bool
  562. */
  563. public function Mail($from) {
  564. $this->error = null; // so no confusion is caused
  565. if(!$this->connected()) {
  566. $this->error = array(
  567. "error" => "Called Mail() without being connected");
  568. return false;
  569. }
  570. $useVerp = ($this->do_verp ? " XVERP" : "");
  571. fputs($this->smtp_conn,"MAIL FROM:<" . $from . ">" . $useVerp . $this->CRLF);
  572. $rply = $this->get_lines();
  573. $code = substr($rply,0,3);
  574. if($this->do_debug >= 2) {
  575. $this->edebug("SMTP -> FROM SERVER:" . $rply . $this->CRLF . '<br />');
  576. }
  577. if($code != 250) {
  578. $this->error =
  579. array("error" => "MAIL not accepted from server",
  580. "smtp_code" => $code,
  581. "smtp_msg" => substr($rply,4));
  582. if($this->do_debug >= 1) {
  583. $this->edebug("SMTP -> ERROR: " . $this->error["error"] . ": " . $rply . $this->CRLF . '<br />');
  584. }
  585. return false;
  586. }
  587. return true;
  588. }
  589. /**
  590. * Sends the quit command to the server and then closes the socket
  591. * if there is no error or the $close_on_error argument is true.
  592. *
  593. * Implements from rfc 821: QUIT <CRLF>
  594. *
  595. * SMTP CODE SUCCESS: 221
  596. * SMTP CODE ERROR : 500
  597. * @access public
  598. * @return bool
  599. */
  600. public function Quit($close_on_error = true) {
  601. $this->error = null; // so there is no confusion
  602. if(!$this->connected()) {
  603. $this->error = array(
  604. "error" => "Called Quit() without being connected");
  605. return false;
  606. }
  607. // send the quit command to the server
  608. fputs($this->smtp_conn,"quit" . $this->CRLF);
  609. // get any good-bye messages
  610. $byemsg = $this->get_lines();
  611. if($this->do_debug >= 2) {
  612. $this->edebug("SMTP -> FROM SERVER:" . $byemsg . $this->CRLF . '<br />');
  613. }
  614. $rval = true;
  615. $e = null;
  616. $code = substr($byemsg,0,3);
  617. if($code != 221) {
  618. // use e as a tmp var cause Close will overwrite $this->error
  619. $e = array("error" => "SMTP server rejected quit command",
  620. "smtp_code" => $code,
  621. "smtp_rply" => substr($byemsg,4));
  622. $rval = false;
  623. if($this->do_debug >= 1) {
  624. $this->edebug("SMTP -> ERROR: " . $e["error"] . ": " . $byemsg . $this->CRLF . '<br />');
  625. }
  626. }
  627. if(empty($e) || $close_on_error) {
  628. $this->Close();
  629. }
  630. return $rval;
  631. }
  632. /**
  633. * Sends the command RCPT to the SMTP server with the TO: argument of $to.
  634. * Returns true if the recipient was accepted false if it was rejected.
  635. *
  636. * Implements from rfc 821: RCPT <SP> TO:<forward-path> <CRLF>
  637. *
  638. * SMTP CODE SUCCESS: 250,251
  639. * SMTP CODE FAILURE: 550,551,552,553,450,451,452
  640. * SMTP CODE ERROR : 500,501,503,421
  641. * @access public
  642. * @return bool
  643. */
  644. public function Recipient($to) {
  645. $this->error = null; // so no confusion is caused
  646. if(!$this->connected()) {
  647. $this->error = array(
  648. "error" => "Called Recipient() without being connected");
  649. return false;
  650. }
  651. fputs($this->smtp_conn,"RCPT TO:<" . $to . ">" . $this->CRLF);
  652. $rply = $this->get_lines();
  653. $code = substr($rply,0,3);
  654. if($this->do_debug >= 2) {
  655. $this->edebug("SMTP -> FROM SERVER:" . $rply . $this->CRLF . '<br />');
  656. }
  657. if($code != 250 && $code != 251) {
  658. $this->error =
  659. array("error" => "RCPT not accepted from server",
  660. "smtp_code" => $code,
  661. "smtp_msg" => substr($rply,4));
  662. if($this->do_debug >= 1) {
  663. $this->edebug("SMTP -> ERROR: " . $this->error["error"] . ": " . $rply . $this->CRLF . '<br />');
  664. }
  665. return false;
  666. }
  667. return true;
  668. }
  669. /**
  670. * Sends the RSET command to abort and transaction that is
  671. * currently in progress. Returns true if successful false
  672. * otherwise.
  673. *
  674. * Implements rfc 821: RSET <CRLF>
  675. *
  676. * SMTP CODE SUCCESS: 250
  677. * SMTP CODE ERROR : 500,501,504,421
  678. * @access public
  679. * @return bool
  680. */
  681. public function Reset() {
  682. $this->error = null; // so no confusion is caused
  683. if(!$this->connected()) {
  684. $this->error = array(
  685. "error" => "Called Reset() without being connected");
  686. return false;
  687. }
  688. fputs($this->smtp_conn,"RSET" . $this->CRLF);
  689. $rply = $this->get_lines();
  690. $code = substr($rply,0,3);
  691. if($this->do_debug >= 2) {
  692. $this->edebug("SMTP -> FROM SERVER:" . $rply . $this->CRLF . '<br />');
  693. }
  694. if($code != 250) {
  695. $this->error =
  696. array("error" => "RSET failed",
  697. "smtp_code" => $code,
  698. "smtp_msg" => substr($rply,4));
  699. if($this->do_debug >= 1) {
  700. $this->edebug("SMTP -> ERROR: " . $this->error["error"] . ": " . $rply . $this->CRLF . '<br />');
  701. }
  702. return false;
  703. }
  704. return true;
  705. }
  706. /**
  707. * Starts a mail transaction from the email address specified in
  708. * $from. Returns true if successful or false otherwise. If True
  709. * the mail transaction is started and then one or more Recipient
  710. * commands may be called followed by a Data command. This command
  711. * will send the message to the users terminal if they are logged
  712. * in and send them an email.
  713. *
  714. * Implements rfc 821: SAML <SP> FROM:<reverse-path> <CRLF>
  715. *
  716. * SMTP CODE SUCCESS: 250
  717. * SMTP CODE SUCCESS: 552,451,452
  718. * SMTP CODE SUCCESS: 500,501,502,421
  719. * @access public
  720. * @return bool
  721. */
  722. public function SendAndMail($from) {
  723. $this->error = null; // so no confusion is caused
  724. if(!$this->connected()) {
  725. $this->error = array(
  726. "error" => "Called SendAndMail() without being connected");
  727. return false;
  728. }
  729. fputs($this->smtp_conn,"SAML FROM:" . $from . $this->CRLF);
  730. $rply = $this->get_lines();
  731. $code = substr($rply,0,3);
  732. if($this->do_debug >= 2) {
  733. $this->edebug("SMTP -> FROM SERVER:" . $rply . $this->CRLF . '<br />');
  734. }
  735. if($code != 250) {
  736. $this->error =
  737. array("error" => "SAML not accepted from server",
  738. "smtp_code" => $code,
  739. "smtp_msg" => substr($rply,4));
  740. if($this->do_debug >= 1) {
  741. $this->edebug("SMTP -> ERROR: " . $this->error["error"] . ": " . $rply . $this->CRLF . '<br />');
  742. }
  743. return false;
  744. }
  745. return true;
  746. }
  747. /**
  748. * This is an optional command for SMTP that this class does not
  749. * support. This method is here to make the RFC821 Definition
  750. * complete for this class and __may__ be implimented in the future
  751. *
  752. * Implements from rfc 821: TURN <CRLF>
  753. *
  754. * SMTP CODE SUCCESS: 250
  755. * SMTP CODE FAILURE: 502
  756. * SMTP CODE ERROR : 500, 503
  757. * @access public
  758. * @return bool
  759. */
  760. public function Turn() {
  761. $this->error = array("error" => "This method, TURN, of the SMTP ".
  762. "is not implemented");
  763. if($this->do_debug >= 1) {
  764. $this->edebug("SMTP -> NOTICE: " . $this->error["error"] . $this->CRLF . '<br />');
  765. }
  766. return false;
  767. }
  768. /**
  769. * Get the current error
  770. * @access public
  771. * @return array
  772. */
  773. public function getError() {
  774. return $this->error;
  775. }
  776. /////////////////////////////////////////////////
  777. // INTERNAL FUNCTIONS
  778. /////////////////////////////////////////////////
  779. /**
  780. * Read in as many lines as possible
  781. * either before eof or socket timeout occurs on the operation.
  782. * With SMTP we can tell if we have more lines to read if the
  783. * 4th character is '-' symbol. If it is a space then we don't
  784. * need to read anything else.
  785. * @access private
  786. * @return string
  787. */
  788. private function get_lines() {
  789. $data = "";
  790. $endtime = 0;
  791. /* If for some reason the fp is bad, don't inf loop */
  792. if (!is_resource($this->smtp_conn)) {
  793. return $data;
  794. }
  795. stream_set_timeout($this->smtp_conn, $this->Timeout);
  796. if ($this->Timelimit > 0) {
  797. $endtime = time() + $this->Timelimit;
  798. }
  799. while(is_resource($this->smtp_conn) && !feof($this->smtp_conn)) {
  800. $str = @fgets($this->smtp_conn,515);
  801. if($this->do_debug >= 4) {
  802. $this->edebug("SMTP -> get_lines(): \$data was \"$data\"" . $this->CRLF . '<br />');
  803. $this->edebug("SMTP -> get_lines(): \$str is \"$str\"" . $this->CRLF . '<br />');
  804. }
  805. $data .= $str;
  806. if($this->do_debug >= 4) {
  807. $this->edebug("SMTP -> get_lines(): \$data is \"$data\"" . $this->CRLF . '<br />');
  808. }
  809. // if 4th character is a space, we are done reading, break the loop
  810. if(substr($str,3,1) == " ") { break; }
  811. // Timed-out? Log and break
  812. $info = stream_get_meta_data($this->smtp_conn);
  813. if ($info['timed_out']) {
  814. if($this->do_debug >= 4) {
  815. $this->edebug("SMTP -> get_lines(): timed-out (" . $this->Timeout . " seconds) <br />");
  816. }
  817. break;
  818. }
  819. // Now check if reads took too long
  820. if ($endtime) {
  821. if (time() > $endtime) {
  822. if($this->do_debug >= 4) {
  823. $this->edebug("SMTP -> get_lines(): timelimit reached (" . $this->Timelimit . " seconds) <br />");
  824. }
  825. break;
  826. }
  827. }
  828. }
  829. return $data;
  830. }
  831. }
  832. ?>