فرمها یکی از بخشهای لاینفک وب اپلیکیشنها هستند به طوری که برای ثبتنام، نظردهی و کارهای به مراتب بیشتری مورد استفاده قرار میگیرند. یکی از کاربردهای دستورات شرطی if که در آموزش گذشته با آن آشنا شدید، چک کردن دیتا فرمهای HTML است که در این آموزش پس از آشنایی با نحوهٔ ساخت یک فرم نظرسنجی، یاد خواهید گرفت که چگونه با استفاده از دستورات if else میتوان دادههای ارسالی توسط کاربران را مورد بررسی قرار داده و از پر بودن کلیهٔ فلیدها اطمینان حاصل کرد.
اگرچه آموزش ترفندهای HTML خارج از سرفصلهای آموزشی این دوره است، اما به منظور درک بهتر سازوکار فرمها، در ادامه اقدام به نوشتن یک فرم نظردهی کرده سپس بخشهای مهم آن را توضیح خواهیم داد:
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title>Simple HTML Form</title>
</head>
<body>
<form action="handle_form.php" method="POST">
Your name: <br>
<input type="text" name="name"><br>
<br> Your email: <br>
<input type="text" name="email"><br>
<br> Your gender: <br>
<input type="radio" name="gender" value="male"> Male<br>
<input type="radio" name="gender" value="female"> Female<br>
<br> Your age: <br>
<select name="age">
<option value="10-20">10-20</option>
<option value="20-30">20-30</option>
<option value="30-40">30-40</option>
<option value="40-50">40-50</option>
<option value="50-60">50-60</option>
<option value="60-70">60-70</option>
<option value="70-80">70-80</option>
</select>
<br> Your comments: <br>
<textarea name="comment" rows="15" cols="50"></textarea><br><br>
<input type="submit" value="Submit">
</form>
</body>
</html>
کدهای فوق را میتوانیم در فایلی تحت عنوان form.html ذخیره سازیم.
به خاطر داشته باشید |
آنچه در نامگذاری این فرم حائز اهمیت میباشد این است که با توجه به اینکه هیچ نوع کد PHP داخل این فایل نیست، به سادگی میتوانیم پسوند html. را برای آن انتخاب کنیم (البته اگر این فایل را با پسوند php. نیز ذخیره سازیم هیچ مشکلی پیش نخواهد آمد) |
در اسکریپت فوق دو اتریبیوت وجود دارد که بسیار حائز اهمیت هستند که عبارتند از action و method. کاری که Value (مقدار) در نظر گرفته شده برای اتریبیوت action انجام میدهد این است که حاوی آدرس صفحهای است که اسکریپت PHP مد نظرمان که قرار است دیتای فرم را هَندل کند در آن قرار دارد؛ اتریبیوت method هم -همانطور که از نامش مشخص است- متد یا روشی که قرار است این فرم هَندل شود را مشخص میسازد که یا میتواند GET باشد و یا POST.
آشنایی با تفاوتهای متدهای GET و POST
مقدار اتریبیوت method یکی از متدهای پروتکل HTTP است که هر کدام خصوصیات و بالتبع کاربردهای خاص خود را دارند که در ادامه بیشتر پیرامون این تفاوتها بحث خواهیم کرد.
متد GET دیتای ارسالی توسط فرم HTML را در قالب یکسری Name-Value (نام-مقدار) از طریق URL برای اسکریپت مد نظر ارسال میکند. به عنوان مثال داریم:
http://www.example.com/script.php?name=Behzad&gender=M
همانطور که در URL فوق مشخص است، پس از نام اسکریپت مد نظر (script.php) یک علامت ? قرار داده سپس دیتای مد نظر خود را در قالب دو جفت Name-Value که با علامت & از یکدیگر جدا شدهاند ارسال کردهایم.
در واقع، دو پارامتر به نامهای name و gender داریم که به ترتیب مقادیر Behzad و M برای آنها در نظر گرفته شده است. به طور کلی، مزایای متد GET عبارتند از:
- پس از استفاده از این متد، میتوان URL را در مرورگر بوکمارک کرد.
- بدون از دست رفتن دیتا، میتوان صفحه را رِفرش کرد.
در عین حال، این متد یکسری نقاط ضعف هم دارا است که عمدهترین آنها عبارتند از:
- دیتای حجیمی را نمیتوان با استفاده از این متد به سمت سرور ارسال کرد.
- دیتای ارسالی به سمت سرور قابل رؤیت است.
- زمانهایی که امنیت دیتای ارسالی زیاد است، این متد کارایی ندارد.
با این تفاسیر، میشود گفت که میتوان از متد GET زمانهایی استفاده کرد که صرفاً بخواهیم یکسری دادهها را از دیتابیس فراخوانی کنیم و یا یکسری پارامتر را از صفحهای به صفحهٔ دیگر پاس (انتقال) دهیم.
در مقابل، از متد POST زمانهایی میتوانیم استفاده کنیم که:
- حجم دیتای ارسالی بسیار زیاد است.
- امنیت دادهها بالا است.
در فرم فوق، از آنجا که قصد داریم دیتای نسبتاً زیادی برای اسکریپتی که در فایل handle_form.php وجود دارد ارسال کنیم، از متد POST استفاده کردهایم.
به خاطر داشته باشید |
نوشتن مقادیر اتریبیوت method هم میتواند با حروف کوچک باشد و هم با حروف بزرگ؛ به عبارت دیگر، هم میتوان به صورت get/post و هم GET/POST آنها را نوشت اما روال نوشتن با حروف بزرگ است. |
چیزی که در ارتباط با فرم فوق باید مد نظر قرار داد، نام یکسان برای فیلد gender است. در حقیقت، نام یکسان gender برای هر دو فیلد مربوطه این امکان را برای کاربر فراهم میآورد تا بتواند فقط و فقط یکی از آیتمها را انتخاب کند. در حقیقت اگر بخش مربوط به انتخاب جنسیت در فرم فوق را به صورت زیر ریفکتور کنیم:
<input type="radio" name="gender_one" value="male"> Male<br>
<input type="radio" name="gender_two" value="female"> Female<br>
همانطور که مشاهده میشود، برای اتریبیوتهای name از نامهای مختلفی همچون gender_one و gender_two استفاده شده که این امکان را در اختیار ما قرار خواهند داد که بتوانیم در آن واحد هر دو گزینه را انتخاب کنیم.
یکی از مزیتهای زبان PHP این است که تعامل بسیار خوبی با زبان HTML دارد و همین قضیه موجب گردیده کار با آن بسیار آسان گردد. در ادامه، برای اینکه بتوانیم دیتای فرم را با استفاده از زبان PHP هَندل کنیم، فایلی میسازیم تحت عنوان handle_form.php حاوی کدهای زیر:
<?php
$name = $_REQUEST['name'];
$email = $_REQUEST['email'];
$gender = $_REQUEST['gender'];
$age = $_REQUEST['age'];
$comment = $_REQUEST['comment'];
echo "Your name is: $name;<br> Your email is: $email;<br> Your gender is: $gender;<br> Your age is: $age;<br> Your comment is: $comment";
پیش از این، به طور خیلی خلاصه با مفهوم متغیرهای اصطلاحاً Super Global آشنا شدیم که یکی از آنها REQUEST_$ است که این وظیفه را دارا است تا کلیهٔ دیتای ارسال شده توسط فرمها (حاوی متدهای GET و POST) را در خود ذخیره سازد.
نکتهای که در اینجا باید توجه ویژهای به آن کرد این است که مقادیر (کلیدهای) در نظر گرفته شده در REQUEST_$ دقیقاً میبایست با مقادیری که برای اتریبیوت name در فیلدهای فرم انتخاب کردیم، یکسان باشد. به عبارت دیگر، اگر برای فیلد مرتبط با ایمیل کاربر مقداری همچون email را برای اتریبیوت name در نظر گرفته باشیم، در اسکریپت PHP خود نیز باید به شکل زیر به آن دست یابیم:
$_REQUEST['email']
و چنانچه این کلید را با حرف اول بزرگ و به شکل Email بنویسیم، با هشداری به صورت زیر مواجه خواهیم شد:
Notice: Undefined index: Email in /var/www/sokanacademy-php-course/03/03-form/handle_form.php on line 2
حال کلیهٔ فیلدهای فرم فوق را پر کرده و روی دکمهٔ Submit کلیک میکنیم:
Your name is: Behzad;
Your email is: behzad.moradi@example.com;
Your gender is: male;
Your age is: 30-40;
Your comment is: This is my first comment.
میبینیم که توانستیم کلیهٔ مقادیر فرم را بگیریم. کاری که در اینجا انجام دادهایم، صرفاً پرینت کردن مقادیر روی صفحه است و این در حالی است که میتوان این مقادیر را برای کاربر خاصی ایمیل کرد و آیا آنها را در دیتابیس ذخیره ساخت که در فصول آینده بیشتر پیرامون این مباحث توضیح خواهیم داد.
نکته |
در ارتباط با تست این فرم میبایست دو نکته را مد نظر داشته باشیم و آن هم این که محل قرارگیری فایل handle_form.php دقیقاً میبایست در همان پوشهای باشد که فایل form.html قرار گرفته است. همچنین این فرم را باید از طریق لوکالهاست تست کرد و در غیر این صورت، اسکریپت PHP اجرا نخواهد شد. |
اکنون قصد داریم یک بار دیگر فرم نظرسنجی فوق را اصطلاحاً Submit (ارسال) کنیم اما بدون این که هیچکدام از فیلدهای این فرم را پر کنیم یا انتخاب کنیم:
Notice: Undefined index: gender in /var/www/sokanacademy-php-course/03/03-form/handle_form.php on line 4
Your name is: ;
Your email is: ;
Your gender is: ;
Your age is: 10-20;
Your comment is:
میبینیم که در مورد فیلد جنسیت با ارور مواجه شده و الباقی فیلدها هم خالی نمایش داده میشوند اما این در حالی است که به سادگی و با استفاده از دستورات شرطی if میتوان از پر بودن کلیهٔ فیلدها اطمینان حاصل کرد.
همانطور که در فایل HTML فوق مشخص است، فیلد مرتبط با سن کاربر دارای یک مقدار پیشفرض (۲۰-۱۰) است و قصد داریم تا در ادامه، آپشنی همچون «لطفاً یک گزینه را انتخاب نمایید» در معرض دید کاربر قرار دهیم. برای این منظور داریم:
<select name="age">
<option value="default">Please choose an option</option>
<option value="10-20">10-20</option>
<option value="20-30">20-30</option>
<option value="30-40">30-40</option>
<option value="40-50">40-50</option>
<option value="50-60">50-60</option>
<option value="60-70">60-70</option>
<option value="70-80">70-80</option>
</select>
در حقیقت، آپشنی با مقدار default و عنوان Please choose an option پیش از سایر آپشنها در نظر گرفتهایم و این در حالی است که اگر اکنون صفحه را رِفرش کنیم، خواهیم دید که کاربر بایستی روی این فیلد کلیک کرده و یکی از گزینهها را انتخاب نماید. حال برای اطمینان حاصل کردن از پر بودن کلیهٔ فیلدها، ابتدا اسکریپت قرار گرفته داخل فایل handle_form.php را تکمیل کرده سپس به تفسیر آنها خواهیم پرداخت:
<?php
$name = $_REQUEST['name'];
$email = $_REQUEST['email'];
$gender = $_REQUEST['gender'];
$age = $_REQUEST['age'];
$comment = $_REQUEST['comment'];
if (empty($name)) {
echo "Please fill in the name field <br>";
} else {
if (empty($email)) {
echo "Please fill in the email field <br>";
} else {
if (! isset($gender)) {
echo "Please fill in the gender field <br>";
} else {
if ($age == 'default') {
echo "Please fill in the age field <br>";
} else {
if (empty($comment)) {
echo "Please fill in the comment field <br>";
} else {
echo "Your name is: $name;<br> Your email is: $email;<br> Your gender is: $gender;<br> Your age is: $age;<br> Your comment is: $comment";
}
}
}
}
}
اکنون یک بار بدون این که هیچکدام از فیلدها را پر کنیم، روی دکمهٔ Submit کلیک میکنیم:
Notice: Undefined index: gender in /var/www/sokanacademy-php-course/03/03-form/handle_form.php on line 4
Please fill in the name field
میبینیم هشداری دریافت میکنیم مبنی بر این که اندیس gender نامعلوم (Undefined) است. برای رفع این مشکل، دو راهکار پیش رو داریم؛ راهکار اول، استفاده از اپراتور @ است که پیش از این به معرفی آن پرداختیم. در واقع، کاری که این اپراتور انجام میدهد این است که جلوی نمایش هشدارها و اخطارهایی که توسط مفسر PHP ایجاد میشوند را میگیرد (گفتیم که تا حد ممکن از این اپراتور برای چنین کاربردهایی نباید استفاده کرد چرا که در پروژههای بزرگ ممکن است فرایند دیباگینگ پروژه را با دردسر مواجه سازد):
$gender = @$_REQUEST['gender'];
از این پس، دیگر تحت هیچ عنوان اخطاری در معرض دید کاربر قرار نخواهد گرفت. به عنوان راهکار دوم، میتوان از یک دستور شرطی استفاده کرد:
if (isset($_REQUEST['gender'])) {
$gender = $_REQUEST['gender'];
}
گفتیم که فانکشن ()isset چک میکند ببیند که آیا یک متغیر Set (مقداردهی) شده است یا خیر؛ اگر اصطلاحاً Set شده بود، مقدار true و در غیر این صورت، مقدار false را باز میگرداند. هر کدام از ترفندهای فوق را که استفاده کنیم، پس از کلیک کردن روی دکمهٔ Submit بدون پر کردن هیچ کدام از فیلدها، خروجی به صورت زیر خواهد بود:
Please fill in the name field
حال برسیم به تفسیر دستورات شرطی که پر بودن فیلدهای فرم را چک میکنند. در اولین دستور شرطی که در خط هشتم قرار دارد، از فانکشنی تحت عنوان ()empty استفاده کردهایم که به صورت Built-in (از پیش تعریف شده) در هستهٔ زبان PHP نوشته شده است. کاری که متد ()empty انجام میدهد این است که چک میکند ببیند آیا پارامتر ورودیاش خالی است یا خیر؛ اگر خالی بود، مقدار true را باز میگرداند و در غیر این صورت، مقدار false اصطلاحاً return خواهد شد.
از آنجا که ما هیچ چیزی در فیلد مربوط به نام وارد نکردهایم، مسلماً این فانکشن مقدار true را باز میگرداند و وارد بلوک if میشویم و عبارت Please fill in the name field چاپ میشود (لازم به ذکر است برای این که تمامی هشدارها در خطوط مجزایی در معرض دید کاربر قرار گیرند، از تگ <br> در پایان هر پیام استفاده کردهایم).
اما اگر کاربر فیلد مربوط به نام را پر کرده باشد، وارد بلوک else اولین دستور if میشویم و مجدداً با استفاده از یک بلوک if else دیگر در خط یازدهم، پر/خالی بودن فیلد مربوط به ایمیل را چک کردهایم.
در ارتباط با متغیر gender$، باتوجه به این که این متغیر از جنسی نیست که بتواند پر/خالی باشد (در واقع این فیلد از نوع Radio است)، برای چک کردن آن از فانکشن ()isset به همراه اپراتور ! استفاده کردهایم. به عبارت دیگر، دستور دادهایم که اگر این متغیر مقداردهی نشده بود، وارد بلوک مرتبط با if شویم.
در ادامه، به متغیر age$ میرسیم. نکتهای که در ارتباط با Value (مقدار) این متغیر وجود دارد این است که اگر آن را با تابع ()empty چک کنیم، این تابع همواره مقدار false را نمایش خواهد داد (به عبارت دیگر، همواره نشان خواهد داد که این فیلد مقداردهی شده است). دلیل اینچنین مسئلهای این است که ما به عنوان مقدار اولین آپشن این فیلد، استرینگ default را مورد استفاده قرار دادهایم.
در چنین مواقعی، بایستی از اپراتور مقایسهای == استفاده کنیم. همانطور که ملاحظه میشود، در خط هفدهم گفتهایم اگر مقدار متغیر age$ برابر با استرینگ default بود، ارور نمایش داده شود و در غیر این صورت، به مرحلهٔ بعد برویم.
در نهایت، پر/خالی بودن متغیر comment$ را هم در خط بیستم چک کرده و اگر کاربر متنی وارد فیلد مربوطه کرده باشد، وارد بلوک else مرتبط خواهیم شد و مقادیر کلیهٔ متغیرها چاپ خواهند شد:
Your name is: Behzad;
Your email is: behzad.moradi@example.com;
Your gender is: male;
Your age is: 30-40;
Your comment is: This is my first comment.
به نظر میرسد که با ذکر مثال فوق، با کاربرد دستورات شرطی در ارتباط با چک کردن فیلدهای فرمهای HTML آشنا شده باشید. آنچه در این مثال حائز اهمیت میباشد این است که این اسکریپت یک آسیبپذیری جدی دارد. به عبارت دیگر، اگر کاربری پیدا شود و با استفاده از ابزار Inspect Element مرورگر اقدام به تغییر کدهای HTML کرده و مثلاً مقداری همچون oops را برای یکی از فیلدهای مرتبط با جنسیت یا سن در نظر بگیرد چهطور؟
<select name="age">
<option value="oops">Please choose an option</option>
<option value="10-20">10-20</option>
<option value="20-30">20-30</option>
<option value="30-40">30-40</option>
<option value="40-50">40-50</option>
<option value="50-60">50-60</option>
<option value="60-70">60-70</option>
<option value="70-80">70-80</option>
</select>
میبینیم که فرد هکر در اولین آپشن به جای استرینگ default از استرینگی همچون oops استفاده کرده است و اگر کلیهٔ فیلدهای فرم را پر کرده و فیلد مرتبط با سن را دستنخوره باقی بگذاریم، فرم اصطلاحاً Valid (معتبر) بوده و اسکریپت ما بدون نمایش دادن هیچ هشداری تا پایان اجرا میگردد:
Your name is: Behzad;
Your email is: behzad.moradi@example.com;
Your gender is: male;
Your age is: oops;
Your comment is: This is my first comment.
در واقع، ما در دستور شرطی مرتبط با متغیر age$ صرفاً گفتهایم که اگر مقدار این متغیر برابر با استرینگ default بود به کاربر هشدار داده شود (این قضیه برای فیلد مرتبط با مشخص کردن جنسیت هم کاملاً صادق است).
در چنین مواقعی، با استفاده از دستورات شرطی مختلف و همچنین یکسری فانکشنهای شخصیسازی شده حاوی الگوریتمهایی به منظور ارسال دیتای صحیح به سمت سرور، میتوان جلوی اینگونه رفتارهای کاربران (هکرها) را گرفت.