Pisząc poradę o wysyłaniu emaili od razu chciałem zrobić klasę zabezpieczająca przed wszystkim. Jednak w pewnym momencie zauważyłem, że kod ma około 100 linii, a miała być to prosta porada. Dlatego wróciłem do możliwie najprostszego kodu, a tu mam zamiar trochę rozwinąć swoją myśl. Na tyle dużo, aby pokazać, o co mi chodzi, ale nie a tyle aby zrobić to za ciebie :).
Główne wady poprzedniego kodu:
- brak walidacji danych wejściowych
- brak informacji dlaczego mail się nie wysłał (nieprawidłowe dane, błędne działanie funkcji mail)
- tylko jeden odbiorca
Już w tamtej poradzie wspomniałem, że celowo kod jest trochę "nadmiarowy". Teraz postaram się odpowiedzieć na pytanie, po co jest nadmiarowy.
Walidacja danych wejściowych
Zróbmy sobie tablicę pozwalającą nam walidować dane wejściwoe wedle uznania (wykorzystując wyrażenia regularne).
1. class Mailer {
2.
3. private $validation = array ( 'to' => '/@/i' , 'subject' => '/.{1,100}/' );
4.
5. private $to = false;
Celowo dałem linię przed i linię po, abyś wiedział gdzie ową zmianę w kodzie umieścić.
Jak widać, chcę sprawdzać jedynie email oraz tytuł. Wyrażenia regularne są tu bardzo ubogie. Celowo - nie chodzi tu o naukę regExpów, ale o pewne podejście do projektu. Zauważ, że w tablicy yMailer::validation nie ma pola 'content'. Zakładam, że content nie musi być validowany.
Skoro już mamy tablicę walidacyjną, to jeszcze by się przydało ją jakoś wykorzystać. Ale w którym momencie? Ja proponuję, aby zrobić to przy dodawaniu danych.
01. public function recipient( $to ) {
02. if ( $this ->validInput( $to , 'to' )) {
03. $this ->to = $to ;
04. return true;
05. }
06. return false;
07. }
08.
09. public function subject( $subject ) {
10. if ( $this ->validInput( $subject , 'subject' )) {
11. $this ->subject = $subject ;
12. return true;
13. }
14. return false;
15. }
16. public function content( $content ) {
17. if ( $this ->validInput( $content , 'content' )) {
18. $this ->content = $content ;
19. return true;
20. }
21. return false;
22. }
W powyższym kodzie jeszcze jedna rzecz jest niejasna. Czym u licha jest yMailer::validInput?! Już dopisuję:
01. private $content = false;
02. private function validInput( $input , $name ) {
03. if (isset( $this ->validation[ $name ])) {
04. if (preg_match( $this ->validation[ $name ], $input )) {
05. return true;
06. } else {
07. return false;
08. }
09. } else {
10. return true;
11. }
12. }
13.
14. public function recipient( $to ) {
No, teraz już mamy walidację.Nadal jednak coś mi się nie podoba. Co jeśli chcę mieć inne wyrażenia regularne do sprawdzania podanych wartości? Może zrobić yMailer::validation publiczną? Byłaby możliwość ustawiania sobie dowolnej zmiennej... Nie. To by nam znowu związało interfejs z implementacją, a dążymy do tego, aby interfejs pozostawał taki sam. Nawet jeśli wewnątrz klasa działa zupełnie inaczej.
Proponuję zrobić tak:
1. private $content = false;
2. public function __construct( $validation = null ) {
3. if ( is_array ( $validation )) {
4. $this ->validation = $validation ;
5. }
6. }
7.
8. private function validInput( $input , $name ) {
Teraz możemy tworzyć obiektyu na dwa sposoby:
1. $mail1 = new yMailer( array ( 'content' => '/wow/' ));
2. $mail2 = new yMailer();
jak widzisz, teraz możemy sami decydować co chcemy sprawdzać i według jakich zasad. Oczywiście jest tu przydatna znajomość wyrażeń regularnych... Jednak w internecie można znaleźć wiele miejsc, gdzie można się ich poduczyć, lub znaleźć gotowce.
Kolejnym punktem w wadach powyższej klasy było 'brak informacji o tym, 'dlaczego mail nie został wysłany'.
Ustalmy 3 scenariusze:
- porażka z powodu nieprawidłowych danych (np. wywołanie yMailer::send przed ustawieniem wszystkich danych)
- porażka z powodu czegoś innego (ale dane same w sobie wypełnione)
- sukces - funkcja mail zwraca prawdę
Jak widać są trzy scenariusze. A więc true / false to za mało. Zrobimy -1 / 0 / 1.
Oto nowy kod metody yMailer::send():
01. public function send() {
02. $result = -1;
03. if (false === $this ->to or
04. false === $this ->subject or
05. false === $this ->content) {
06. if (mail( $this ->to, $this ->subject, $this ->content)) {
07. $result = 1;
08. } else {
09. $result = 0;
10. }
11. }
12.
13. return $result ;
14. }
Dzięki takiemu zabiegowi mogę teraz stwierdzić, czy nie udało się wysłać maila z powodu nieustawienia jakichś danych, czy też wina leży w czym innym.
Refaktoryzacja kodu
Zawsze po napisaniu i przetestowaniu kodu sprawdź, czy nie da się go uprościć. Ja widzę w powyższym dwa miejsca, które chciałbym uprościć. Pierwsze to powtarzający się kod metod yMailer::recipient(), yMailer::content() i yMailer::subject(). Zobaczmy czy coś się da z tym zrobić...
01. private function add( $value , $name ) {
02.
03. if ( $this ->validInput( $value , $name )) {
04.
05. $this -> $name = $value ;
06.
07. return true;
08.
09. }
10.
11.
12. return false;
13.
14. }
15.
16.
17. public function recipient( $to ) {
18.
19. return $this ->add( $to , 'to' );
20.
21. }
22.
23.
24. public function subject( $subject ) {
25.
26. return $this ->add( $subject , 'subject' );
27.
28. }
29.
30.
31. public function content( $content ) {
32.
33. return $this ->add( $content , 'content' );
34.
35. }
Od razu trochę czytelniej. Co więcej musimy sprawdzić jedynie czy działa metoda yMailer::add(), a nie testować 3 inne. Zmusza nas to dodatkwo do schamtyczności w kodzie. Same plusy. Minusem może być to, że trochę na pierwszy rzut oka zaciemniło nam obraz.
To teraz dla odmiany zróbmy coś co nam pozwoli uniknąć komentarzy w kodzie - komentować się będzie sam kod! Chodzi o metodę yMailer::send() i jej długi warunek, który będzie się z pewnością wydłużał (zaraz zechcesz dodać jeszcze więcej parametrów, zmienna ty, zmienna tam i w kodzie jest bałagan).
01. private function allDataSet() {
02. if (false != $this ->to and
03. false != $this ->subject and
04. false != $this ->content) {
05. return true;
06. }
07. return false;
08. }
09.
10. public function send() {
11. $result = self::WRONG_DATA;
12. if ( $this ->allDataSet()) {
13. if (mail( $this ->to, $this ->subject, $this ->content)) {
14. $result = self::SEND_SUCC;
15. } else {
16. $result = self::SEND_FAIL;
17. }
18. }
19.
20. return $result ;
21. }
A co tu robią jakieś self::WRONG_DATA, self::SEND_SUCC, self::SEND_FAIL? To stałe, należy je zdefiniować na początku klasy:
1. private $content = false;
2. const WRONG_DATA = -1;
3. const SEND_SUCC = 1;
4. const SEND_FAIL = 0;
5. public function _construct( $validation = null )
Po co uzywać stałych? Choćby po to, aby kod był cxzytalniejszy. Nie ma nic przyjemniejszego jak nagle zwracana wartość w stylu 214... I co to oznacza. Jeśli ją zdefiniujesz wcześniej jako
stałą czytając kod od razu wiadomo co ma się dziać. Koduj aby łatwo było czytać twój kod, bo piszesz go raz, a czytasz... na pewno więcej. Czasem setki razy.
Oczywiście, w sytuacji takiej jak tu, gdzie masz 3 zmienne do sprawdzenia to wyrzucenie tego do osobnej metody może być nadmiarowością kodu. Jednak pamiętaj, że to jest bardziej klasa przykładowa, co robić jak się za wiele rzeczy składa do kupy niż klasa, której masz używać. Choć, w sumie, jeśli by jeszcze nad nią poracować chwilę to wyszłaby bartdzo fajna klasa do obsługi maila.
Na naszej liście widnieje jeszcze 3 opcja: brak wsparcia dla wysyłania maili do więcej niż jednego odbiorcy. To już jest zadanie dla ciebie. Podpowiedź. pole $to zmień na tablicę i po prostu dopisuj nowych odbioróc do końca tablicy. A może warto jest rozszerzyć funkcjonalnośći z $to zrobić osobny obiekt? Powodzenia!
A oto cała klasa, ze wszystkimi zmianami jakich dokonaliśmy (licencja GPL):
01. <?php
02.
03. class yMailer {
04. private $validation = array ( 'to' => '/@/i' ,
05. 'subject' => '/.{1,100}/'
06. );
07. private $to = false;
08. private $subject = false;
09. private $content = false;
10. const WRONG_DATA = -1;
11. const SEND_SUCC = 1;
12. const SEND_FAIL = 0;
13.
14. public function __construct( $validation = null ) {
15. if ( is_array ( $validation )) {
16. $this ->validation = $validation ;
17. }
18. }
19.
20. private function validInput( $input , $name ) {
21.
22. if (isset( $this ->validation[ $name ])) {
23. if (preg_match( $this ->validation[ $name ], $input )) {
24. return true;
25. } else {
26. return false;
27. }
28. } else {
29. return true;
30. }
31. }
32.
33. private function add( $value , $name ) {
34. if ( $this ->validInput( $value , $name )) {
35. $this -> $name = $value ;
36. return true;
37. }
38. return false;
39. }
40.
41. public function recipient( $to ) {
42. return $this ->add( $to , 'to' );
43. }
44.
45. public function subject( $subject ) {
46. return $this ->add( $subject , 'subject' );
47. }
48.
49. public function content( $content ) {
50. return $this ->add( $content , 'content' );
51. }
52.
53. private function allDataSet() {
54. if (false != $this ->to and
55. false != $this ->subject and
56. false != $this ->content) {
57. return true;
58. }
59. return false;
60. }
61.
62. public function send() {
63. $result = self::WRONG_DATA;
64. if ( $this ->allDataSet()) {
65. if (mail( $this ->to, $this ->subject, $this ->content)) {
66. $result = self::SEND_SUCC;
67. } else {
68. $result = self::SEND_FAIL;
69. }
70. }
71.
72. return $result ;
73. }
74. }
75.
76.
77. $mail = new yMailer();
78.
79. echo ( $mail ->recipient( 'adres(a)serwer.pl' )) ? "ustawiony rec<hr>" : "blad adresu<hr>" ;
80. echo ( $mail ->subject( 'bbb' )) ? "subject jest poprawny<hr>" : "subject niepoprawny<hr>" ;
81. echo ( $mail ->content( 'aaa' )) ? "content<hr>" : "no content<hr>" ;
82.
83. switch ( $mail ->send()) {
84. case yMailer::WRONG_DATA :
85. echo 'Niewlasciwe dane' ;
86. break ;
87. case yMailer::SEND_FAIL:
88. echo "Błąd" ;
89. break ;
90. case yMailer::SEND_SUCC:
91. echo 'sukces' ;
92. break ;
93. }
|
Komentarze