article

هوشمند سازی پنل ورود و خروج سایت با PHP و MySQL

mysql-php-smart-login

در مطالب قبل از بخش آموزش کاربردی کار با PHP و MySQL به طور مفصل به مبحث ساخت فرم عضویت و نحوه ورود و خروج کاربران به پنل مدیریت سایت پرداختیم، بحث نشست ها (سشن یا Session) را با هم مرور کرده و با نحوه رمزنگاری کلمه عبور و همچنین ذخیره و بازیابی اطلاعات اعضاء سایت آشنا شدیم، این بار در تکمیل آموزش های قبل می خواهیم کمی بیشتر بخش مدیریت سایتمان را توسعه داده و آن را به اصطلاح هوشمندتر کنیم، در این یادداشت خواهیم دید که چگونه می توانیم امکاناتی مانند "مرا به خاطر بسپار" و  خروج خودکار کاربر بعد از چند دقیقه غیر فعال بودن را به سیستم مدیریت سایتمان اضافه کنیم.

نکته مهم: معمولا قابلیت "مرا به خاطر داشته باش" برای برنامه هایی که به امنیت بالایی احتیاج دارند توصیه نمی شود، چرا که اگر کاربر به هر دلیل از سیستم به شکل استاندارد خارج نشود (به فرض پنجره مرورگر خود را ببندد یا برق به طور ناگهانی قطع گردد)، اشخاص دیگر بر روی همان سیستم و همان مرورگر می توانند بدون وارد کردن نام کاربری و کلمه عبور و تنها با داشتن آدرس URL صفحه کاربری به قسمت مدیریت دسترسی پیدا کنند! متاسفانه این مشکلی است که بیشتر از روی عدم آگاهی کاربران و معمولا در کامپیوترهای اشتراکی مورد استفاده در اماکن عمومی مانند کافی نت ها، مراکز دانشگاهی و... اتفاق می افتد، لذا در نظر گرفتن این امکان باید با نمایش هشدارهای لازم و رعایت نکات امنیتی صورت گیرد.

ایجاد امکان "مرا به خاطر بسپار" در پنل مدیریت سایت


سایت های زیادی را دیده ایم که در قسمت ورود اطلاعات نام کاربری و کلمه عبور معمولا با دکمه ای به صورت چک باکس امکانی تحت عنوان "مرا به خاطر بسپار" یا با عبارات مشابه ارائه می کنند، این امکان در واقع به معنی این است که سیستم تا مدت زمان معینی در مراجعات بعدی از کاربر نام کاربری و کلمه عبور درخواست نکرده و به صورت هوشمندانه وضعیت ورود کاربر را به یاد خواهد داشت، اما امکان "مرا به خاطر بسپار" چگونه در پنل مدیریت ایجاد می شود؟
اگر به صورت اجمالی بخواهیم ساز و کار این امکان را بررسی کنیم به این صورت خواهد بود:
هنگامی که گزینه یادآوری را انتخاب و فرم را ارسال می کنیم برنامه با دستورات PHP یک کوکی (Cookie) با تاریخ انقضای مد نظرمان در مرورگر ایجاد کرده و با مقداری مشابه در سرور فایل سشن را نیز ایجاد می کند، مقدار کوکی - سشن معمولا عبارت متنی 22 تا 40 کاراکتری اتفاقی شامل حروف و اعداد انگلیسی است که با توابع رمزنگاری (در حال حاضر MD5 و SHA-1) ایجاد می شود، اطلاعات سشن به صورت پیش فرض برای مدت زمانی کوتاه (معمولا 1440 ثانیه معادل 24 دقیقه) نگهداری و سپس حذف خواهد شد، لذا به عنوان برنامه نویس برای اینکه مدت زمان مد نظر بیشتری اعمال کنیم (به فرض یک ماه) دو راه حل در اختیار داریم:
- ایجاد تغییرات در تنظیمات فایل php.ini حین اجرا و افزایش مدت زمان پیش فرض نگهداری سشن ها (این روش توصیه نمی شود).
- نگهداری مقدار سشن آی دی در دیتابیس و در صورت نیاز آپدیت مدت زمان اعتبار یا تنظیم مجدد کوکی و سشن (در صورت انقضاء).

تغییر تنظیمات فایل php.ini حین اجرا


برای حالت اول یعنی ایجاد تغییرات در تنظیمات فایل php.ini (حاوی تنظیمات پیش فرض مفسر PHP) معمولا نیازی به ویرایش دستی این فایل نیست، در صورت پشتیبانی سرور کافی است دستورات زیر را در ابتدای هر صفحه که نشست در آن فعال است قرار دهیم تا در حین اجرای کدها، تنظیمات پیش فرض لغو و با مقادیر مد نظرمان جایگزین شوند:
<?php
//یک ساعت به ثانیه * ساعات یک روز * تعداد روزهای یک ماه = یک ماه به ثانیه
ini_set('session.gc_maxlifetime', 3600 * 24 * 30);
ini_set('session.cookie_lifetime', 3600 * 24 * 30);
ini_set('session.cache_expire', 3600 * 24 * 30);
session_name('panel');
session_start();
?>
کسب اطلاعات بیشتر در خصوص فایل php.ini:
https://www.php.net/manual/en/function.ini-set.php
اگرچه این روش به اصلاح دم دستی و سریع است اما ممکن است همیشه نتیجه مطلوب ندهد و لذا خیلی قابل اتکا نیست، روش بهتر استفاده از دیتابیس جهت نگهداری اطلاعات کوکی - سشن آی دی و به روزرسانی یا تنظیم مجدد سشن است که در ادامه بررسی خواهیم کرد.

نگهداری مقدار سشن آی دی در دیتابیس و آپدیت یا تنظیم مجدد کوکی و سشن


در شیوه دوم و استفاده از دیتابیس تا زمانی که کوکی مورد نظر در مرورگر وجود دارد و به دلیل منقضی شدن تاریخ کوکی یا کلیک کاربر بر روی دکمه خروج Unset و حذف نشده است، مرورگر در هر درخواست مقدار کوکی را به سرور ارسال کرده و در نتیجه وضعیت دسترسی کاربر برای برنامه قابل شناسایی است و بدون ورود نام کاربری و کلمه عبور حتی با بستن پنجره مرورگر و مراجعه بعدی می تواند از امکانات بخش اعضاء سایت استفاده نماید، اما قاعدتا انجام این موارد توسط برنامه نیاز به نوشتن دستورات و ایجاد تغییراتی در ساختار و کدهای پنل مدیریت سایت به شرح زیر دارد.

تنظیم اطلاعات اتصال به دیتابیس در فایل config.php


طبق روال برای اتصال به دیتابیس نیاز به تنظیم چند پارامتر در تابع mysqli_connect است که شامل نام سرور، نام کاربری دیتابیس، کلمه عبور دیتابیس و در نهایت عنوان دیتابیس است، برای تجمیع این موارد و استفاده چندباره از تنظیمات پیکربندی در صفحات مختلف معمولا از فایل Configoration استفاده می شود که در این آموزش نیز بر همین اساس اطلاعات را در فایلی با نام دلخواه config.php به صورت نمونه زیر قرار داده ایم:
<?php
//تنظیمات اتصال به دیتابیس
$config = array(
    'host' => 'localhost',
    'db_user' => 'نام کاربری دیتابیس',
    'db_pass' => 'کلمه عبور دیتابیس',
    'db_name' => 'نام دیتابیس'
);
?>
این فایل جهت نمونه است و باید مطابق با اطلاعات دیتابیسی که در آموزش های قبل ساخته ایم تکمیل شود، در صورتی که این آموزش ها را مطالعه نکرده اید لطفا ابتدا از بخش آموزش های کاربردی سایت مطالب مرتبط به مبحث ثبت نام و عضویت کاربران و همچنین ورود به پنل مدیریت را مطالعه کنید (برای دسترسی سریعتر می توانید از دکمه "قبلی" در انتهای همین صفحه نیز استفاده کنید).

ساخت جدول sessions جهت نگهداری اطلاعات نشست در دیتابیس


برای ایجاد امکان هوشمندسازی پنل مدیریت سایت در کنار جدول users که در دو آموزش قبلی ساخته ایم نیاز به ایجاد جدول دیگری تحت عنوان sessions است که اطلاعات نشست هر کاربر را در ستون هایی نگهداری کند، گفتیم که نشست ها در PHP به صورت پیش فرض تنها 24 دقیقه معتبر هستند و در صورتی که طی این مدت کاربر صفحه ای را فراخوانی نکند خود به خود منقضی شده و کاربر باید مجددا وارد پنل شود، لذا برای نگهداری طولانی مدت اطلاعات نشست و ایجاد امکان "مرا بخاطر بسپار" گزینه متداول استفاده از ساز و کار دیتابیس است، بدین منظور کدهای زیر را در فایل create.php قرار داده و این فایل را در مرورگر خود (لوکال یا آنلاین) فراخوانی می کنیم:
<?php
//فایل تنظیمات اتصال به دیتابیس
include_once('config.php');

//اتصال به دیتابیس
$conn = mysqli_connect($config['host'], $config['db_user'], $config['db_pass'], $config['db_name']);

if(!$conn) {
    echo "PHP & MySQL Connection: Error! " . mysqli_connect_errno() . ' - ' . mysqli_connect_error();
    exit;
} else {
    echo "PHP & MySQL Connection: Ok!<br>";

    //نام جدول
    $tbl_name = "sessions";

    //ساخت جدول و ستون ها
    $sql = "CREATE TABLE $tbl_name(
    id INT UNSIGNED NOT NULL AUTO_INCREMENT PRIMARY KEY,
    username VARCHAR(255),
    remember TINYINT(1) NOT NULL DEFAULT 0,
    session_value VARCHAR(255),
    session_date DATETIME)
    ENGINE=MyISAM DEFAULT CHARACTER SET=utf8 COLLATE=utf8_persian_ci"
;
    $query = mysqli_query($conn, $sql);

    if(!$query) {
        echo "Creating Table $tbl_name: Error! " . mysqli_error($conn) . "<br>";
    } else {
        echo "Creating Table $tbl_name: OK!<br>";
    }

    //افزودن ستون ها به ایندکس جهت افزایش سرعت پرس و جوها
    if(!$query) {
        $sql = "ALTER TABLE $tbl_name ADD INDEX `username` (`username`)";
        $query = mysqli_query($conn, $sql);
    }

    if(!$query) {
        $sql = "ALTER TABLE $tbl_name ADD INDEX `session_value` (`session_value`)";
        $query = mysqli_query($conn, $sql);
    }

    if(!$query) {
        echo "Altering Table $tbl_name: Error! " . mysqli_error($conn);
    } else {
        echo "Altering Table $tbl_name: OK!";
    }
}

//پایان اتصال
mysqli_close($conn);
?>
توضیح:
- ستون id را به صورت UNSIGNED در نظر گرفته ایم، این پارامتر یعنی ستون id صرفا شامل اعداد مثبت خواهد بود و به این صورت حداکثر ظرفیت اسمی آن از عدد 2147483647 به 4294967295 افزایش خواهد یافت:
https://dev.mysql.com/doc/refman/5.6/en/integer-types.html
- جدول sessions شامل پنج ستون id, username, remember, session_value و session_date است که هر ستون را با هدف خاصی در نظر گرفته ایم.
- ستون id شماره ردیف یکتا، ستون username نام کاربر، ستون remember وضعیت یادآوری، ستون session_value مقادیر کوکی - سشن آی دی و در نهایت ستون session_date تاریخ انقضای نشست را در خود جای می دهند.
نکته: این جدول در یک دیتابیس و کنار جدول users که در آموزش های گذشته برای ثبت نام و عضویت کاربر استفاده شده باید ایجاد گردد.

افزودن چک باکس یادآوری به فرم HTML ورود


برای ایجاد امکان "مرا بخاطر بسپار" به فرم HTML که در آموزش قبل جهت ورود به پنل مدیریت سایت ساخته ایم یک فیلد input از نوع چک باکس (checkbox) اضافه می کنیم، در صورتی که این آموزش را مطالعه نکرده اید لطفا ابتدا از بخش آموزش های کاربردی سایت مطلب مرتبط به مبحث ورود به پنل مدیریت را مطالعه کنید (برای دسترسی سریعتر می توانید از دکمه "قبلی" در انتهای همین صفحه نیز استفاده کنید).
<?php
//پیش فرض
$access = false;

//بررسی وضعیت دسترسی کاربر
include_once('access.php');

if($access === true) {
    //انتقال به پنل مدیریت
    header("Location: panel.php");
    exit;
}

//اگر فرم ارسال نشده باشد
if(!isset($check)) {
    $text = null;
    $username = null;
    $checked = null;
}
?>
<!DOCTYPE html>
<html lang="fa">
<head>
<meta charset="utf-8">
<meta name="viewport" content="width=device-width, initial-scale=1">
<title>وبگو | فرم HTML هوشمند ورود کاربر به پنل مدیریت سایت</title>
<!-- Webgoo.ir -->
<style>
body {
font-family: Tahoma, Geneva, sans-serif;
direction: rtl;
font-size: 12px;
line-height: 22px;
}
.error {
height: 30px;
width: 100%;
color: #dd000a;
}
</style>
</head>
<body>
<?php echo $text; ?>
<form action="login.php" method="post">
<label for="user">نام کاربری:</label>
<input name="username" id="user" type="text" value="<?php echo $username; ?>" maxlength="255" dir="ltr">
<label for="pass">کلمه عبور:</label>
<input name="password" id="pass" type="password" maxlength="255" dir="ltr">
<label for="remember">مرا بخاطر بسپار:</label>
<input name="remember" id="remember" type="checkbox" value="1"<?php echo $checked; ?>>
<input name="check" type="hidden" value="1">
<input type="submit" value="ورود به پنل">
</form>
<hr>
- قبل از ورود کاربر به پنل مدیریت سایت ابتدا باید اطلاعات اتصال به دیتابیس را در فایل config.php در آرایه config تعریف کرده باشیم، نام کاربری در لوکال هاست معمولا root و بدون کلمه عبور است.<br>
- اجرای صحیح این کد مستلزم ساخت دیتابیس، جدول users، ستون ها و درج نمونه اطلاعاتی است که در آموزش نحوه ساخت فرم عضویت سایت توضیح داده ایم.<br>
((در صورتی که آموزش نحوه ساخت فرم عضویت سایت را مطالعه نکرده و جدول users را با نمونه کاربر فرضی نساخته اید، لطفا ابتدا به بخش آموزش های کاربردی MySQL مراجعه و این مطلب را مطالعه کنید.))<br>
- همچنین باید جدول و ستون های مربوط به ثبت نشست ها در دیتابیس ایجاد شوند، برای این کار آدرس فعلی مرورگر را کپی کرده و با افزودن عبارت create.php در انتهای آدرس، در پنجره جدید این فایل را یک بار فراخوانی می کنیم، به طور مثال:<br>
http://localhost/php-mysql-signin-smart/create.php<br>
- دقت کنیم برای رمزنگاری کلمه عبور در مرحله عضویت کاربر از تابع password_hash استفاده کرده ایم، این تابع از نسخه PHP 5.5 در دسترس است، برای سازگاری نسخه های قدیمی تر کتابخه password_compat به کدها اضافه شده است.<br>
</body>
</html>
توضیح:
- این فایل در مقایسه با فایل index.php آموزش قبل تغییراتی داشته است، از جمله شروع نشست و اضافه شدن امکان بررسی وضعیت دسترسی کاربر در فایل access.php که در صفحه وارد (include) شده است.
- چک باکس یادآوری در فرم بالا با نام remember به صورت تگ input و نوع checkbox است.
- برای اینکه به طور پیش فرض چک باکس فعال بوده و به حالت انتخاب شده درآمده باشد پارامتر checked را به صورت متغیر با مقدار زیر ویرایش می کنیم:
$checked = ' checked="checked"';
اگر بخواهیم به طور پیش فرض تیک چک باکس غیر فعال باشد می توانیم این قسمت را null در نظر بگیریم.
- مقادیر value که برای فرم بالا و قسمت چک باکس عدد 1 در نظر گرفته شده فرضی است و می تواند هر مقداری باشد، به طور معمول مقادیر دیگری مانند true، ok و... نیز در نظر گرفته می شود، اما همان طور که در ادامه خواهیم دید نکته مهم این است که مقدار این فیلد یا وضعیت ارسال مقادیر آن را در سمت سرور و در کدهای PHP به عنوان متغیر از فرم HTML دریافت و بررسی خواهیم کرد تا از انتخاب شدن یا نشدن گزینه "یادآوری" توسط کاربر اطمینان حاصل کنیم.

بررسی اطلاعات ارسالی از فرم ورود در فایل login.php


پس از ارسال فرم توسط کاربر اطلاعات فیلدها به سرور منتقل شده و در این مرحله باید ببینیم نام کاربری و کلمه عبور با آنچه که پیشتر در قسمت ثبت نام در دیتابیس ثبت شده مطابقت دارد یا خیر، در صورت عدم تطابق ضمن نمایش پیغام خطا فرم ورود را مجددا بارگذاری می کنیم و در صورتی که اطلاعات ارسال شده با مقدار موجود تطابق داشت، ضمن تنظیم کوکی در مرورگر و ثبت مقدار سشن در دیتابیس کاربر را به فایل panel.php (فایل اصلی پنل مدیریت) منتقل خواهیم کرد، برای انجام این فرایندها در فایل login.php که در آموزش پیشین ساختیم تغییرات زیر را اعمال می کنیم:
<?php
//شروع نشست
session_name('panel');
session_start();

//تنظیم منطقه زمانی
date_default_timezone_set('Asia/Tehran');

//فایل تنظیمات اتصال به دیتابیس
include_once('config.php');

//تعریف توابع کلمه عبور در نسخه های پائین تر از PHP 5.5
if(!function_exists('password_hash')) {
    include_once('lib/password_compat.php');
}

//پیش فرض
$_SESSION['access'] = false;
$_SESSION['username'] = null;
$error = 0;
$text = null;
$checked = null;

//دریافت مقدار از فرم و تعریف متغیرها
@$username = $_POST['username'];
@$password = $_POST['password'];
@$remember = $_POST['remember'];
@$check = $_POST['check'];

/* بررسی معتبر بودن اطلاعات کاربر در صورت ارسال فرم */
if($check == 1) {
    //نام کاربری
    if(!isset($username) || empty($username)) {
        $error = 1;
        $text = "نام کاربری خود را وارد کنید!";
    } //کلمه عبور
    elseif(!isset($password) || empty($password)) {
        $error = 1;
        $text = "کلمه عبور خود را وارد کنید!";
    }

    //بررسی خطا
    if($error === 0) {
        //اتصال به دیتابیس
        $conn = mysqli_connect($config['host'], $config['db_user'], $config['db_pass'], $config['db_name']);

        if(!$conn) {
            echo "PHP & MySQL Connection: Error! " . mysqli_connect_errno() . ' - ' . mysqli_connect_error();
            exit;
        } else {
            //ایمن سازی پارامترها
            $username = mysqli_real_escape_string($conn, $username);

            //نام جدول
            $tbl_name = "users";

            //اانتخاب اطلاعات از جدول و ستون
            $sql = "SELECT `password` FROM $tbl_name WHERE `username` = '$username' LIMIT 1";
            $query = mysqli_query($conn, $sql);

            if(!$query) {
                echo "Selecting From Table $tbl_name: Error! " . mysqli_error($conn) . '<br>';
                exit;
            } else {
                //تعداد ردیف های انتخاب شده
                $count = mysqli_num_rows($query);

                if($count === 0) {
                    $error = 1;
                    $text = "نام کاربری یا کلمه عبور اشتباه است!";
                } else {
                    while($row = mysqli_fetch_array($query)) {
                        $db_hashed_password = $row['password'];

                        //تطبیق کلمه عبور
                        if(password_verify($password, $db_hashed_password)) {
                            //تنظیم متغیرهای سشن
                            $_SESSION['login'] = true;
                            $_SESSION['access'] = true;
                            $_SESSION['username'] = $username;
                            $_SESSION['remember'] = $remember;

                            //تعیین زمان کوکی یادآوری
                            if($remember == 1) {
                                //انقضای کوکی برای یک ماه
                                $cookie_time = 3600 * 24 * 30;
                            } else {
                                //انقضای کوکی برای نیم ساعت
                                $cookie_time = 1800;
                            }

                            //نام جدول
                            $tbl_name = "sessions";

                            //اانتخاب اطلاعات از جدول و ستون
                            $sql = "SELECT username FROM $tbl_name WHERE `username` = '$username' LIMIT 1";
                            $query = mysqli_query($conn, $sql);

                            if(!$query) {
                                echo "Selecting From Table $tbl_name: Error! " . mysqli_error($conn) . '<br>';
                                exit;
                            } else {
                                //تعداد ردیف های انتخاب شده
                                $count = mysqli_num_rows($query);

                                //پارامترهای سشن برای دیتابیس
                                $session_value = session_id();
                                $session_date = date('Y-m-d H:i:s', time() + $cookie_time);

                                //آپدیت یا ثبت اطلاعات نشست در دیتابیس
                                if($count == 1) {
                                    $sql = "UPDATE $tbl_name SET `remember` = '$remember', `session_value` = '$session_value', `session_date` = '$session_date' WHERE `username` = '$username' LIMIT 1";
                                    $query = mysqli_query($conn, $sql);
                                } else {
                                    $sql = "INSERT INTO $tbl_name(`username`, `remember`, `session_value`, `session_date`) VALUES('$username', '$remember', '$session_value', '$session_date')";
                                    $query = mysqli_query($conn, $sql);
                                }

                                if(!$query) {
                                    echo "Table $tbl_name Updating/Inserting: Error! " . mysqli_error($conn);
                                    exit;
                                }
                            }

                            //انتقال به پنل مدیریت
                            header("Location: panel.php");
                            exit;
                        } else {
                            $error = 1;
                            $text = "نام کاربری یا کلمه عبور اشتباه است!";
                        }
                    }
                }
            }
        }

        //پایان اتصال
        mysqli_close($conn);
    }
}

//بارگذاری مجدد فرم ورود در صورت بروز خطا یا فراخوانی مستقیم فایل لاگین
if($check != 1 || $error == 1) {
    if($error == 1) {
        $text = '<div class="error">' . $text . '</div>';
    }

    if($remember == 1) {
        $checked = ' checked="checked"';
    }

    include_once('index.php');
}
?>
توضیح:
- با ارسال اطلاعات فرم ورود به فایل login.php باید بررسی کنیم و ببینیم که نام کاربری و کلمه عبور ارسال شده با مقادیر موجود در دیتابیس مطابقت دارد یا خیر، در صورتی که اطلاعات صحیح باشد ضمن تنظیم متغیرهای سشن (login, access, username, remember) وضعیت نشست را در جدول sessions درج یا در صورت وجود به روزرسانی می کنیم.
- به دلیل استفاده از توابع header و session_start هیچ نوع خروجی نباید پیش از اجرای این توابع به مرورگر ارسال شود که شامل فضای خالی یا کاراکترهای نامرئی BOM (مخفف Byte Order Mark) نیز می شود، در غیر این صورت با پیغام خطای 
Warning: Cannot modify header information - headers already sent...
مواجه خواهیم شد، به همین دلیل در فایل login.php پیش از اجرای توابع header و session_start هیچ خروجی در سمت مرورگر چاپ نمی شود و با ایجاد انتقال از طریق تابع header و در نظر گرفتن پارامتر location کاربر را به طور خودکار به صفحه کاربری منتقل یا در صورت خطا فرم ورود (index.php) را در انتهای صفحه include می کنیم.
نکته: برای اطمینان بهتر است فایل ها را در برنامه ++Notepade اجرا و از قسمت Encoding گزینه Encoding in UTF-8 without BOM را انتخاب و مجددا فایل را ذخیره کنیم.
- مدت زمان اعتبار نشست در دو حالت متغیر خواهد بود، زمانی که کاربر تیک چک باکس یادآوری را انتخاب کرده باشد، این زمان را 30 روز معادل 2592000 ثانیه در نظر می گیریم (3600 ثانیه در 24 ساعت در 30 روز = 2592000 ثانیه)، 3600 میزان 1 ساعت به ثانیه است (60 ثانیه در 60 دقیقه)، اما اگر تیک یادآوری انتخاب نشده باشد زمان را 30 دقیقه معادل 1800 ثانیه در نظر می گیرم، این اعداد برای ثبت یا به روزرسانی اطلاعات نشست در دیتابیس و کوکی در مرورگر به کار می روند.
- برای محاسبه تاریخ و زمان از توابع date و time استفاده کرده ایم، تابع date تاریخ را با فرمت های متداول به فرض
2020-2-24 02:44:55
خروجی داده و تابع time تاریخ را به صورت برچسب زمان (Timestamp) به فرض 1581155023 با فرمت Unix خروجی می دهد.
- در صورت تطبیق صحیح اطلاعات ارسالی از فرم HTML با مقادیر موجود در دیتابیس، کاربر به فایل panel.php که در واقع فایل اصلی پنل کاربری سایت است منتقل می شود.

پنل کاربری سایت در فایل panel.php


فایل panel.php در واقع هسته اصلی بخش مدیریت سایتمان است که دیگر امکانات پنل در زیرمجموعه آن تعریف می شوند، در این فایل دو حالت متصور است:
در حالت اول اگر کاربر مستقیم از فرم ورود به این فایل ارجاع داده شده باشد یعنی هنوز کوکی یادآوری تنظیم نشده و باید ببینیم چک باکس "مرا بخاطر بسپار" انتخاب شده یا خیر، اگر چک باکس انتخاب شده باشد (متغیر سشن remember برابر 1 باشد) مدت زمان کوکی مشابه آنچه در مرحله لاگین مشاهده کردیم برابر با 30 روز (2592000 ثانیه) خواهد بود، در غیر اینصورت کوکی صرفا برای نیم ساعت (1800 ثانیه) تنظیم می شود.
در حالت دوم اگر کاربر قبلا به قسمت مدیریت ارجاع شده و در حال حاضر در پنل سایت حضور دارد، وضعیت دسترسی با متغیر سشن access بررسی می شود، این متغیر هم در فایل login.php و هم در فایل access.php تنظیم شده است، access.php فایلی است که وضعیت دسترسی کاربر را در هر درخواست زیر نظر دارد! (در ادامه این فایل را بررسی خواهیم کرد).
محتویات فایل panel.php با توجه به نیاز و ساختار برنامه می تواند متفاوت باشد، به عنوان مثال:
<?php
//پیش فرض
$access = false;
$cookie_time = 0;

//بررسی وضعیت دسترسی کاربر
include_once('access.php');

//فقط برای اولین ورود
if((isset($_SESSION['login']) && $_SESSION['login'] === true)) {
    //تنظیم کوکی یادآوری
    if(isset($_SESSION['remember']) && $_SESSION['remember'] === 1) {
        //تنظیم کوکی برای یک ماه
        $cookie_time = 3600 * 24 * 30;
        setcookie("panel", session_id(), time() + $cookie_time, "/");
    } else {
        //تنظیم کوکی برای نیم ساعت
        $cookie_time = 1800;
        setcookie("panel", session_id(), time() + $cookie_time, "/");
    }
}

//بررسی متغیرهای نشست
if((isset($_SESSION['access']) && $_SESSION['access'] === true)) {
    //پیام خوش آمدگویی و سایر کدهای پنل مدیریت
    echo 'Welcome To Control Panel!<br>User Name: ' . $_SESSION['username'];
    echo '&nbsp;[<a href="logout.php" title="Logout">Logout</a>]<br>';

    echo '<span id="show-time"></span>' . "\n";

    echo '<script>var countdown_timer; var server_time = ' . time() . '; var cookie_time = ' . (time() + $cookie_time) . ';
    window.onload = function(){countdown_timer = setInterval("reminedTime(\'show-time\');", 1000);}</script>'
. "\n";

    echo '<script src="script.js"></script>' . "\n";
} else {
    //دسترسی غیرمجاز
    header("HTTP/1.1 403 Forbidden");
    echo 'HTTP 403 Forbidden<br>Access Denied!';
    exit;
}
?>
توضیح:
- از تابع setcookie برای تنظیم کوکی جهت تعیین وضعیت دسترسی به بخش اعضاء سایت برای 30 روز یا نیم ساعت (بسته به انتخاب یا عدم انتخاب چک باکس یادآوری) استفاده می کنیم.
- تابع setcookie باید یک آرگیومنت به صورت زمان به ثانیه داشته باشد که نمایانگر مدت زمان اعتبار کوکی است، به فرض برای یک ساعت مقادیر 3600 ثانیه یا برای یک ماه معادل 2592000 ثانیه، بدین منظور ما از تابع time با افزودن یک مقدار عددی به آن استفاده کرده ایم.
- نام کوکی با تابع session_name و مقادیر کوکی - سشن با تابع session_id تنظیم می شود، همچنین برای تعیین مسیر (path) دسترسی کوکی پارامتر آخر را اضافه می کنیم، با تنظیم این قسمت به صورت "/" کوکی برای تمام دایرکتوری های زیرمجموعه سایت قابل دسترسی خواهد بود، در صورت عدم استفاده از پارامتر path به طور پیش فرض، کوکی فقط در دایرکتوری که آن را ایجاد کرده ایم قابل دسترسی و استفاده است، به فرض با تنظیم آن به صورت "/user/" کوکی فقط در این فولدر و دایرکتوری های زیرمجموعه آن قابل دسترسی و استفاده است از این رو بهتر است همواره از روش ذکر شده در نمونه کد بالا استفاده کنیم.
- در مقابل امکان "مرا به خاطر داشته باش" قابلیت دیگری را نیز می توان برای پنل مدیریت سایت در نظر گرفت که معمولا به دلیل وجود حساسیت های خاص و ارتقاء امنیت صورت می گیرد، این امکان ایجاد یک سیستم خروج خودکار است که اگر به فرض مدت زمانی کاربر بدون فعالیت در صفحه باقی بماند نشست او خود به خود منقضی می شود، در این حالت حتی در موقعی که پنجره مرورگر هنوز باز است اعتبار نشست کاربر در سرور به پایان می رسد و او باید مجددا در پنل سایت وارد شود، این امکان معمولا با در نظر گرفتن مدت زمان کم برای اعتبار نشست در سیستم هایی که جنبه امنیت برایشان از اهمیت بالایی برخوردار است مانند سیستم تراکنش های بانکی و مالی، کنترل پنل های هاست، سیستم های آزمون آنلاین و...) به کار گرفته می شود.
- برای ایجاد شمارشگر معکوس (Countdown) و نمایش میزان زمان باقی مانده از اعتبار نشست کاربر دستورات PHP و JavaScript را به صورت ترکیبی استفاده کرده ایم، در سمت سرور تاریخ و زمان سرور و تاریخ اعتبار کوکی در تابع جاوا اسکریپتی reminedTime چاپ شده تا به عنوان آرگیومنت به این تابع داده شوند.
- با تابع setInterval به صورت پیوسته در هر ثانیه مجددا تابع reminedTime اجرا و آخرین زمان باقیمانده از اعتبار نشست محاسبه و در بلاک span با آی دی show-time خروجی داده می شود.

نمایش مدت زمان باقیمانده از اعتبار نشست با فایل script.js


تابع reminedTime که وظیفه نمایش مدت زمان باقیمانده از اعتبار نشست را به عهده دارد به صورت زیر در فایل script.js کدنویسی می کنیم:
//استخراج مدت زمان باقیمانده بر اساس اختلاف بین تاریخ سرور با تاریخ انقضای کوکی به ثانیه
var seconds = cookie_time - server_time;

//تابع برای محاسبه و نمایش زمان باقیمانده تا انقضای کوکی
function reminedTime(id){
    var days = Math.floor(seconds / 24 / 60 / 60);
    var hours_left = Math.floor((seconds) - (days * 86400));
    var hours = Math.floor(hours_left / 3600);
    var minutes_left = Math.floor((hours_left) - (hours * 3600));
    var minutes = Math.floor(minutes_left / 60);
    var remaining_seconds = seconds % 60;

    //خروجی
    var output = pad(days) + ":" + pad(hours) + ":" + pad(minutes) + ":" + pad(remaining_seconds) + ' Remained to Logout!';

    if(seconds == 0){
        clearInterval(countdown_timer);
        output = "You Need Login Again, Please Wait 5 Seconds!";
        setTimeout("goToURL('index.php');", 5000);
    } else{
        seconds--;
    }

    //چاپ خروجی در بلاک
    document.getElementById(id).innerHTML = output;
}

function pad(n){
    return (n < 10 ? "0" + n : n);
}

function goToURL(url){
    window.location.href = url;
}
توضیح:
- مدت زمان باقیمانده به ثانیه از حاصل تفریق زمان فعلی سرور و زمان انقضای کوکی به دست می آید، این زمان برای استفاده های بعدی در متغیر seconds مقداردهی شده است.
- در توضیح فایل panel.php گفتیم که تابع setInterval به صورت مداوم در هر ثانیه (1000 میلی ثانیه) تابع reminedTime را فراخوانی می کند، این کار تا زمانی که مقدار متغیر seconds از 0 بزرگتر باشد ادامه پیدا خواهد کرد.
- با 0 شدن مدت زمان باقیمانده، تابع clearInterval فراخوانی و اجرای پیوسته متغیر countdown_timer متوقف می شود.

بررسی وضعیت دسترسی کاربر با فایل access.php


برای بررسی وضعیت فعلی دسترسی کاربر از فایل access.php استفاده کرده ایم، به طور خلاصه این فایل وظیفه دریافت کوکی از مرورگر (در صورت وجود) و تطبیق آن با دیتابیس را بر عهده دارد، همچنین در این فایل نشست با تابع session_start شروع یا ادامه می یابد و اطلاعات دیتابیس با دستور UPDATE به روزرسانی می شود:
<?php
//شروع نشست اگر قبلا (در فایل لاگین) شروع نشده باشد
if(session_status() == PHP_SESSION_NONE || session_id() == '') {
    session_name('panel');
    session_start();
}

//تنظیم منطقه زمانی
date_default_timezone_set('Asia/Tehran');

//فایل تنظیمات اتصال به دیتابیس
include_once('config.php');

//پیش فرض
$access = false;
$panel_cookie = null;
$db_username = null;
$db_remember = 0;

//بررسی و دریافت مقدار کوکی از مرورگر
if(isset($_COOKIE['panel']) && !empty($_COOKIE['panel'])) {
    //مقدار کوکی
    $panel_cookie = $_COOKIE['panel'];

    //بررسی اعتبار کوکی و وضعیت یادآوری در دیتابیس
    $conn = mysqli_connect($config['host'], $config['db_user'], $config['db_pass'], $config['db_name']);

    if(!$conn) {
        echo "PHP & MySQL Connection: Error! " . mysqli_connect_errno() . ' - ' . mysqli_connect_error();
        //exit;
    } else {
        //نام جدول
        $tbl_name = "sessions";

        //تاریخ و زمان حاضر
        $now_date = date('Y-m-d H:i:s', time());

        //اانتخاب اطلاعات از جدول و ستون
        $sql = "SELECT username, remember FROM $tbl_name WHERE `session_value` = '$panel_cookie' AND `session_date` > '$now_date' LIMIT 1";
        $query = mysqli_query($conn, $sql);

        if(!$query) {
            echo "Selecting From Table $tbl_name: Error! " . mysqli_error($conn) . '<br>';
            exit;
        } else {
            //تعداد ردیف های انتخاب شده
            $count = mysqli_num_rows($query);

            //کوکی در دیتابیس وجود دارد
            if($count === 1) {
                //اگر کاربر از فرم ورود به پنل منتقل نشده باشد
                if(!isset($_SESSION['login'])) {
                    //بررسی مقدار یادآوری
                    while($row = mysqli_fetch_array($query)) {
                        $db_username = $row['username'];
                        $db_remember = $row['remember'];
                    }

                    //تنظیم متغیرهای سشن
                    $_SESSION['access'] = true;
                    $_SESSION['username'] = $db_username;
                    $_SESSION['remember'] = $db_remember;

                    //آپدیت کوکی یادآوری
                    if($db_remember === 1) {
                        //آپدیت کوکی برای یک ماه
                        $cookie_time = 3600 * 24 * 30;
                        setcookie("panel", session_id(), time() + $cookie_time, "/");
                    } else {
                        //آپدیت کوکی برای نیم ساعت
                        $cookie_time = 1800;
                        setcookie("panel", session_id(), time() + $cookie_time, "/");
                    }

                    //تاریخ و زمان جدید
                    $session_date = date('Y-m-d H:i:s', time() + $cookie_time);

                    //به روزرسانی زمان اعتبار نشست در دیتابیس
                    $sql = "UPDATE $tbl_name SET `session_date` = '$session_date' WHERE `username` = '$db_username' LIMIT 1";
                    $query = mysqli_query($conn, $sql);

                    if(!$query) {
                        echo "Updating Table $tbl_name: Error! " . mysqli_error($conn);
                        exit;
                    }
                }

                //دسترسی مجاز
                $access = true;
            } else {
                //دسترسی غیرمجاز
                $_SESSION['access'] = false;
                $access = false;
            }
        }
    }

    //پایان اتصال
    mysqli_close($conn);
} else {
    //دسترسی غیرمجاز
    $_SESSION['access'] = false;
    $access = false;
}
?>
توضیح:
- فایل access.php در صفحه index.php نیز استفاده شده و خود این فایل (در صورت بروز خطا در هنگام ورود) در فایل login.php وارد شده است، در این حالت برای جلوگیری از تداخل توابع نشست در فایل access.php ابتدا برقراری نشست را با شرط if و تابع session_status بررسی کرده ایم، در صورتی که تابع session_start قبلا اجرا شده باشد با اجرای مجدد این تابع و بروز تداخل خطای زیر را دریافت خواهیم کرد:
Notice: A session had already been started - ignoring session_start() in ... on line ...
- فایل access.php در ابتدای صفحات مورد نیاز وارد می شود و مواردی کلی پیکربندی مانند تنظیم منطقه زمانی و همچنین فایل config.php حاوی تنظیمات اتصال به دیتابیس در این فایل گنجانده شده است.

خروج از پنل کاربری با فایل logout.php


آخرین قسمت مورد نیاز برای پنل کاربری سایت در نظر گرفتن امکانی برای خروج استاندارد کاربر و منقضی شدن کامل نشست است، این قابلیت به طور معمولا با کلیک کاربر بر روی لینک یا دکمه خروج و فراخوانی فایل حاوی دستورات PHP اتفاق می افتد که در این آموزش به صورت دلخواه logout.php نامگذاری و با کدهای زیر تکمیل شده است:
<?php
//شروع نشست
session_name('panel');
session_start();

//فایل تنظیمات اتصال به دیتابیس
include_once('config.php');

//اتصال به دیتابیس
$conn = mysqli_connect($config['host'], $config['db_user'], $config['db_pass'], $config['db_name']);

if(!$conn) {
    echo "PHP & MySQL Connection: Error! " . mysqli_connect_errno() . ' - ' . mysqli_connect_error();
    exit;
} else {
    //نام جدول
    $tbl_name = "sessions";

    //پارامترهای سشن برای دیتابیس
    $session_username = $_SESSION['username'];

    //حذف اطلاعات سشن از دیتابیس
    $sql = "UPDATE $tbl_name SET `session_value` = null, `session_date` = null WHERE `username` = '$session_username' LIMIT 1";
    $query = mysqli_query($conn, $sql);

    if(!$query) {
        echo "Updating Table $tbl_name: Error! " . mysqli_error($conn);
        exit;
    }
}

//حذف کوکی از مرورگر
$cookie_time = 3600;
setcookie("panel", session_id(), time() - $cookie_time, "/");

//منقضی کردن و حذف اطلاعات نشست از سرور
session_unset();
session_destroy();

//انتقال به صفحه ورود
header("Location: index.php");
exit;
?>
توضیح:
- برای انجام خروج کامل باید اطلاعات نشست را در سه بخش دیتابیس، کوکی و فایل سشن  منقضی و حذف کنیم.
- در قسمت دیتابیس با اجرای دستور UPDATE مقادیر ستون های session_value و session_date را برای نام کاربری جاری به صورت null در نظر می گیریم.
- برای منقضی کردن کوکی کافی است با تابع setcookie و نامی مشابه یک کوکی با مدت زمان گذشته (مقدار منفی) تنظیم کنیم، در اینجا کوکی panel با مدت زمان منفی 3600 ثانیه یعنی یک ساعت قبل در نظر گرفته شده و به این ترتیب کوکی جدید جایگزین کوکی فعلی در مرورگر می شود و با توجه به تنظیم تاریخ گذشته مرورگر اطلاعات آن را در درخواست های بعدی ارسال نخواهد کرد.
- در آخرین مرحله باید اطلاعات نشست را در سرور منقضی و حذف کنیم، این اطلاعات شامل متغیرهای نشست و فایل فیزیکی حاوی سشن آی دی است که بر روی هارددیسک سرور ذخیره شده است، برای انجام این دو هدف توابع session_unset و session_destroy را فراخوانی می کنیم.

دانلود نمونه فایل های آموزش


برای جمع بندی کار و تست آنلاین یا آفلاین می توانیم فایل هایی که در این آموزش بررسی کردیم را از لینک زیر دریافت کنیم:
دانلود نمونه فایل های آموزش هوشمند سازی پنل ورود و خروج سایت با PHP و MySQL
sectionدسته بندی: آموزش کاربردی » MySQL
related مطالب بیشتر:
» تعویض کد امنیتی Captcha با Ajax و MySQL
» صفحه بندی مطالب و محتوا با PHP و MySQL
» نحوه رسم چارت و نمودار آماری با PHP و MySQL
» آموزش حذف گروهی اطلاعات از MySQL با استفاده از چک باکس
» ایجاد لینک دانلود مدت دار با PHP و MySQL
commentنظرات (۷۰ یادداشت برای این مطلب ارسال شده است)
more یادداشت های جدید بر اساس تاریخ ارسال در انتهای یادداشت های موجود نمایش داده می شوند.
نویسنده: سعید
۱۸:۵۵ ۱۳۹۱/۰۵/۰۲
سلام آموزشهای سایتتون فوق العاده اند. لطفا با قدرت به کارتون ادامه بدین.
یه سوالی داشتم در مورد php :
من میخوام موقع حذف یک رکورد پایگاه داده ، اول یک کادری ظاهر شود که بپرسد که آیا از حذف مظمئن هستید یا نه ؟ که اگر کابر روی بله کلیک کرد عمل حذف انجام شود؟ چه طوری میشه این کار رو انجام داد ؟
یه سوال دیگه هم دارم من موقعی که کدهای پی اچ پی رو در یک قالب آماده قرار میدم قسمت ساید بار به پایین سقوط میکنه مشکل از چی میتونه باشه ؟ قالب با css و html نوشته شده. ممنون
پاسخ: 
سلام
ممنون از لطف شما، در مورد سوال اول: این کار به طور معمول با ارسال مقادیر به صورت متد get یا post در دو مرحله به همان فایل یا یک فایل دیگر (به فرض با نام delete.php) صورت می گیرد، به فرض لینک اولیه حذف فایل اگر به صورت زیر باشد:
yoursite.com/index.php?action=del&id=4324
در فایل مقصد پس از دریافت مقادیر del از پارامتر action یک لینک دیگر به عنوان تایید چاپ می کنیم به صورت زیر:
yoursite.com/index.php?action=del&id=4324&confirmed=true
با کلیک مجدد کاربر بر روی لینک دوم و بررسی مقادیر پارامتر فرضی confirmed، می توانیم از مطمئن بودن کاربر از حذف اطلاعات اطمینان حاصل کنیم؛ همین کار را از طریق فرم های html و متد post می توان انجام داد، البته انجام صحیح این کار تا حدود زیادی بستگی به تسلط شما بر روی برنامه ای دارد که قصد ویرایش آن را دارید.
در مورد سوال دوم: منظور از قرار دادن کدهای php در قالب چیست؟ php زبان سمت سرور است و تنها خروجی آن در صفحات html نمایش داده می شود، شاید حروف خروجی کد شما در قسمتی بدون فاصله و شکستگی است یا مقادیری به خروجی ارسال می شود که از عرض سایدبار بیشتر است و این باعث تغییر اندازه سایدبار شده و نهایتا به دلیل کمبود فضای در نظر گرفته شده، به پائین سقوط می کند.
رفع دقیق مشکل نیاز به بررسی کد و صفحه مربوطه دارد.
نویسنده: سعید
۱۱:۱۵ ۱۳۹۱/۰۵/۰۷
سلام مطلب کاربردی بود.ممنون
ببخشید باز من اومدم با سوال :
دارم یه سایت طراحی میکنم که کاربر بعد از ورود به قسمت کاربریش با صفحه ای مواجه میشود که مشخصات به همراه عکس خود را مشاهده میکند (تا اینجا مشکلی نیست همش درست انجام شده)، حال با کلیک بر روی لینکی در صفحه کاربری به صفحه دیگری میرود که دارای فرمی است که فیلدهای آن را تکمیل کرده و با کلیک بر روی دکمه ثبت به صفحه ثبت در پایگاه منتقل میشود. حال در این صفحه من میخوام غیر از ثبت مقادیر فیلدها که با $_post گرفتم، 2 تا از اطلاعات کاربر هم در بانک اضافه شود. مثلا کد عضویت و نام کاربر. این مقادیر را چگونه باید از سشن دریافت کرد و همراه مقادیر فرم در بانک و در فیلد مربوطه ذخیره کرد؟ ممنون (الان مقادیر فرم درج میشود ولی مقادیر کد عضویت و نام کاربر درج نمیشود) دستورات را به شکل زیر نوشتم :
$code1=$_SESSION['code'];
mysql_connect('localhost','root','');
mysql_select_db("project");
$f2="select * from collegians where collegian_code='$code'";
$g2=mysql_query($f2);
$row2=mysql_fetch_array($g2);
$_SESSION['lname']=$row2['Lname'];
$col_name=$_SESSION['lname'];
$query="insert INTO unitsselect (collegian_name, collegian_code, lesson_name, units, master_name, unterm_id ) values ('{$lname}', '{$col_code}', '{$lesson}', {$unit}, '{$masname}', '{$punit}')";
پاسخ: 
سلام
استفاده از سشن در این مورد ضرورتی ندارد، تنها در صورتی از سشن استفاده می شود که بخواهیم متغیرهای موقتی و حساسی (به طور مثال نام کاربری و کلمه عبور) که مربوط به شناسایی هویت کاربر است در هنگامی که او به سیستم وارد شده بین صفحات کاربری مختلف در دسترس باشند، تا امکان یا عدم امکان دسترسی در هر صفحه بررسی شود یا اعتبار نشست او تمدید گردد.
در خصوص مشکل شما با توجه به اطلاعاتی که قرار داده اید، می توان در قسمت mysql_fetch_array متغیرهایی با مقادیری از ردیف های موجود در جدول collegians ایجاد کرد و از آنها در ادامه استفاده نمود.
به نظر می رسد شما در کار با mysql دچار اشتباهاتی شده اید، این آموزش را ملاحظه کنید:
انتخاب ردیف ها با دستور SELECT FROM در MySQL
نویسنده: علی
۱۴:۲۶ ۱۳۹۱/۰۵/۱۴
سلام
لطفا آموزش کد تشخیص انسان از نرم افزار را هم در سایت قرار دهید. مثل همین کد حاصل از عبارت زیر یادداشت. ممنون
پاسخ: 
سلام
ضمن تشکر از پیشنهاد شما، در ادامه آموزش ها و به تدریج به این مطلب نیز خواهیم پرداخت.
نویسنده: فاطمه
۱۹:۲۸ ۱۳۹۱/۰۵/۱۵
سلام سایت شما عالیه میخواستم فقط تشکر کنم لطفا ادامه بدین.
پاسخ: 
سلام
نظر لطف شماست.
نویسنده: saeed
۲۳:۵۶ ۱۳۹۱/۰۵/۲۸
با عرض سلام و خسته نباشيد و همچنين بابت اين سايت خوب و پر محتوا خيلي ممنون بابت اين فايل هاي كمكي چون خيلي اين فايل ها بدرد من ميخورن اگه كد ساختن آمار بازديد در سايت خودتون بزاريد با ديتابيس خيلي ممنون ميشيم، يه عضويت هم بيزاريد كه بتونيم مطالب روز دنيا رو بگريم از سايت شما با تشكر از سايت خوبتون
پاسخ: 
سلام
ممنون، نظر لطف شماست.
در مورد مطالب و آموزش های درخواستی طبق روال و آهسته و پیوسته پیش می رویم، به موضوع آمار بازدیدها نیز در آینده خواهیم پرداخت، در مورد عضویت و خبرنامه، بهترین امکان برای اطلاع از به روز رسانی سایت، استفاده از فید آر اس اس آن است.
نویسنده: نجمه
۱۲:۵۹ ۱۳۹۱/۰۶/۰۵
سلام سایتتون واقعا عالیه بهتون تبریک میگم.من تازه کار هستم دارم اموزش میبینم اما خیلی مشکل دارم .بعضی از کدهارو درک نمیکنم .لطفا اگه میشه کلاس login و ثبت فرم را بذارید
پاسخ: 
سلام
واقعیت مطلب این است که login یا ثبت نام چیزی جداگانه ای نیست و مبتنی بر آشنایی و مهارت کلی شما با PHP است، لذا بهترین روش برای اینکه در کدنویسی دچار مشکل نشوید این است که از ابتدا مباحث را با مثال یاد بگیرید، برنامه نویسی در بیشتر مواقع بدون دانستن مقدمات کار، ممکن است پیچیده به نظر برسد.
نویسنده: rojyar
۱۳:۱۰ ۱۳۹۱/۰۷/۱۵
سلام واقعآ عالی بود...
میشه درخواست کنم که اگه امکان داره مراحل ساخت یک سایت کوچک با php و mysql رو آموزش بدید که داینامیک باشه؟
درحد مثلآ ایجاد صفحه ی جدید و مثلآ مدیریت اعضا و یه بخش مثلآ خبرهای سایت
اگه این رو آموزش بدید در قالب یک سایت کوچک، مطمئنآ ما یاد خواهیم گرفت چه طور یه سایت با php طراحی کنیم
در واقع انجام برنامه نویسی درقالب پروژه خیلی مفید هستش..
لطف کنید این زحمت رو برای متحمل بشید، انشاالله اجرتو با خدا و بزرگان اهل بیت....
پاسخ: 
سلام
سپاس از نظر لطف و پیشنهاد شما، البته مواردی که عنوان کردید در قالب یک مطلب نمی گنجند، سعی می کنیم آموزش های مورد نیاز را برای طراحی یک سایت با php در قالب مطالب جداگانه ارائه دهیم تا از کنار هم نهادن آنها بتوانید ایده هایتان را در این خصوص به واقعیت تبدیل کنید.
نویسنده: علی
۱۳:۲۷ ۱۳۹۱/۰۹/۰۸
با عرض سلام و خسته نباشید
سایت شما بسیار عالی می باشد و مطالب بسیار آموزنده ای در آن می توان یافت
امیدوارم که خداوند پاداش زحمتتان را بدهد.
پاسخ: 
سلام
ممنون از نظر لطف شما.
نویسنده: ansherli
۲۱:۰۲ ۱۳۹۱/۱۰/۲۶
سلام
ببخشید من می خواستم بدونم فرق include با تابع header در چی هستش؟
من وقتی برای header بعد یا قبلش echo میزارم اجرا نمیشه ولی با include کاملا درست اجرا میشه!
//بررسی معتبر بودن اطلاعات ارسالی کاربر
//نام کاربری
if (!isset($username) || $username == ''){
include("login.php");
echo "<font color='#FFFFFF'><table align='center' ><tr><td>!فیلد نام کاربری نباید خالی باشد</td></tr></table></font>";
$check_error = 1;
}
//کلمه عبور
elseif (!isset($password) || $password == ''){
include("login.php");
echo "<font color='#FFFFFF'><table align='center'><tr><td>!فیلد کلمه عبور نباید خالی باشد</td></tr></table></font>";
$check_error = 1;
}
کدهام رو براتون ارسال می کنم . می خوام بببینم مشکلی ایجاد نمی کنه اگه از include استفاده کنم ؟ چطور میشه از همین header استفاده کرد و echo ها هم اجرا بشه؟
من بعضی از کدها رو توی session.php قرار دادم که با
<?php include("session.php");?>
فراخوانی می کنم توی صفحات
کدهام:
http://uplod.ir/bjn7zcj2h3yj/webgoo.zip.htm
ممنون میشم راهنماییم کنید.
پاسخ: 
سلام
تابع header و include کاملا متفاوت هستند، همان طور که از نام header پیداست برای ارسال سربرگ های HTTP به مرورگر از آن استفاده می شود (به طور مثال انتقال به یک صفحه دیگر، ارسال کدهای وضعیت HTTP مانند کد 404، 301 و...) اما از include برای وارد کردن کدهای php درون یک فایل php دیگر و دسترسی آن فایل به کدها (ترکیب فایل ها) استفاده می شود، به همین دلیل وقتی از دستور زیر استفاده می کنید و این دستور اجرا می شود:
header("location:login.php");
به محض اجرا به فایل login.php منتقل شده و سایر دستورات php نادیده گرفته می شوند.
نویسنده: عادل
۲۲:۵۹ ۱۳۹۱/۱۱/۰۷
سلام ...
دیگه نامردی بود اگه پیام نمیذاشتم ...
مطالب بسیار عالیست ..
خیلی استفاده کردیم ...
تشکر از زحمات شما.
پاسخ: 
سلام
نظر لطف شما است.
نویسنده: reza
۲۰:۴۷ ۱۳۹۱/۱۱/۲۰
با سلام
آقا سایتتون عالیه فقط 1 مورد این که وقتی بخوایم که کاربر بعد از ورود به سایت به صفحه شخصی خودش هدایت بشه کدش چی هست؟
یعنی برای هر کاربر 1 صفحه جداگانه با محتویات جداگانه.
1 مورد دیگه این که کدی وجود داره که کاربر به صورت مستقیم نتونه از مرورگر به یه صفحه دسترسی پیدا کنه؟ یعنی اخطار بده بعد بگه اول باید لوگین کنی. ممنونم
راستی خیلی واجبه.
پاسخ: 
سلام
برای مواردی که می خواهید انجام دهید، کدی آماده وجود ندارد، باید با برنامه نویسی php (مخصوصا بخش توابع هدر، سشن و کوکی) آشنا باشید و کدها را مطابق سلیقه خودتان بنویسید، در آموزش های کاربردی php و mysql در این خصوص نمونه کدهایی آموزش داده شده است، اما پیش از استفاده باید با مقدمات کار آشنا باشید.
نویسنده: ehsan
۱۸:۰۲ ۱۳۹۱/۱۲/۲۲
سلام
setcookie(session_name(),session_id(),time()+$login_time,("/"));
میشه در مورد
session_name() , session_id()
توضیح بدید.
پاسخ: 
سلام
هنگامی که از سشن (تابع session_start) در کدهای خود استفاده می کنیم (اطلاعات سشن ها به صورت فایل، در پوشه ای در سرور ذخیره می شوند)، هر سشن در php چند پارامتر در خود دارد، یکی از این پارامترها نام نشست است که با تابع session_name در دسترس است (در حالت پیش فرض معمولا نام سشن PHPSESSID است)، پارامتر دیگر مقادیر سشن است که با تابع session_id قابل چاپ یا استفاده است (این مقادیر معمولا یک کد درهم یا hash است)، حال وقتی می خواهیم یک کوکی در مرورگر کاربر ایجاد کنیم (با تابع setcookie)، از این مقادیر برای تنظیم پارامترهای کوکی نیز می توانیم استفاده کنیم.
دقت کنید که سشن در سرور و کوکی در مرورگر ایجاد می شود.
نویسنده: ehsan
۰۹:۲۰ ۱۳۹۱/۱۲/۲۶
با سلام. همه مراحل را قدم به قدم اجرا کردم.
موقع لاگین شدن این پیغام ظاهر میشود.
Notice: Undefined variable: check_error in
پاسخ: 
سلام
پیغام ظاهر شده در واقع یک راهنمایی است و دلیل آن به تنظیمات نمایش خطا در php ارتباط دارد، برای رفع آن می توانید یک مقدار پیش فرض برای متغیر check_error تعریف کنید، این کار برای آموزش فعلی انجام شد، می توانید کدهایتان را مطابق آخرین تغییرات ویرایش کنید.
نویسنده: سعید زارع
۱۲:۳۲ ۱۳۹۲/۰۲/۰۹
سلام دوست عزیز
ای کاش اموزش هاتون به صورت فیلم می بود
بخش های متنوع و خوبی دارید فقط خیلی سنگینه واسه کسی که تا حالا زیاد کار نکرده
اگه میشد این اموزش ها رو به صورت تصویری با دریم ویور و xwamp اموزش میدادین خیلی راحت تر میتونستیم یاد بگیریم
پاسخ: 
سلام
آموزش های کاربردی با فرض آشنایی کاربران با مقدمات کار تهیه می شوند، اگر به نظر سخت یا پیچیده است، باید به مقدمات مبحث رجوع کنید، در مورد آموزش به صورت فیلم، متاسفانه به دلیل محدودیت های فنی امکان پذیر نیست ضمن اینکه این نوع آموزش برای زبان های برنامه نویسی روش متداولی نیست، چرا که اصولا یادگیری برنامه نویسی نیاز به نوشتن و برررسی کد دارد و صرف دیدن و شنیدن کفایت نمی کند.
نویسنده: mahdi
۱۵:۲۷ ۱۳۹۲/۰۴/۲۲
با سلام ...
ابتدا تشکر از آموزش های خوب و با ارزشتون :
من با استفاده از php یک فرم ثبت نام و ورود کاریر درست کردم که کاربر پس از ورود ، به پنل کاربری خودش منتقل میشه .
حالا اگه ما بخوایم موقعی که کاربر وارد پنل کاربریش شد اطلاعات مربوط به خودش رو ببینه ، باید چطوری بنویسیم کدش رو ؟
یه راه حلی به ذهنم اومد استفاده از سسشن های موقع ورود اما فقط برای user و pass که موقع ورود خواسته میشن کار ساز هست و بقیه column های دیتابیس مثل کشور یا ... رو نمیده ! باید چیکار کرد ؟ آیا باید به دیتابیس وصل شد و با استفاده از توابع مخصوص خودش column ها رو گرفت ؟ لطفا در صورت راهنمایی کد آن را هم قرار دهید .
با تشکر
پاسخ: 
سلام
راه حل شما صحیح است، یعنی زمانی که نام کاربری و کلمه عبور را با دیتابیس چک می کنید، در صورت صحیح بودن، سایر اطلاعات را نیز استخراج کرده و در سشن آی دی هایی ذخیره کنید (روش سریع تر)، یا اینکه برای هر صفحه، این کار را مجددا با استفاده از به فرض آی دی کاربر که در یک سشن آی دی وجود دارد انجام دهید، به طور مثال:
<?php
//پرس و جوی فرضی
$result = mysql_query ("SELECT name, family, age FROM register WHERE username = '$username' AND password = '$password'")
or die(mysql_error());
while($row = mysql_fetch_array($result)){
$_SESSION['name'] = $row['name'];
$_SESSION['family'] = $row['family'];
$_SESSION['age'] = $row['age'];
}
//گرفتن خروجی فرضی
echo $_SESSION['name'];
?>
more لطفا پیش از ارسال یادداشت نکات زیر را مد نظر داشته باشید:
- مواردی که به کلی خارج از موضوع این مطلب هستند را در فرم منوی "تماس با ما" مطرح و پاسخ را از طریق ایمیل دریافت کنید.
- به سوالات کلی، مبهم، غیرضروری و مشکلاتی که تلاشی برای رفع آن نکرده باشید پاسخ کوتاه و مختصر داده خواهد شد!
- کدها و اسکریپت های طولانی را ترجیحا در یک صفحه وب آنلاین قرار دهید تا امکان تست و بررسی وجود داشته باشد.
- تمام یادداشت ها بررسی و برای هر کاربر زمان مشخصی جهت پاسخگویی در نظر گرفته می شود، لطفا از طرح سوالات متعدد در بازه زمانی کوتاه خودداری کنید.





6 × 9
 refresh
آخرین دیدگاه ها
more برای دسترسی سریع به یادداشت مربوطه می توانید از لینک مطلب در کادر زیر استفاده کنید.
form نبی
در:
‏asc و desc رو جابجا نوشتی
۱۳۹۹/۰۷/۰۴

form ساناز محمدی
در:
سلام مرسی از کدی که گذاشتید ♥
۱۳۹۹/۰۷/۰۳

form mahtab
در:
سلام خسته نباشین ببخشید میخواستم بپرسم که چجوری میتونیم یه کلیپ رو از کامپیوتر از انیستا دانلود کنیم ؟؟ اها اینم بگم...
۱۳۹۹/۰۷/۰۲

form سعید
در:
سلام دستتون درد نکنه از پروژه شما استفاده کردم فقط یه مشکل اگه در یک صفحه دو تا لیست کشویی داشته باشیم چطوری...
۱۳۹۹/۰۷/۰۱

form Iman Mafakheri
در:
سلام من یه قالب خارجی اوردم راست چینش کردم حتی فونتشم تغییر دادم اما متاسفانه وقتی متن فارسی مینویسم حروف رو جدا جدا مینویسه نمیدونم...
۱۳۹۹/۰۶/۳۰

form میثم صدیق
در:
سلام برای اضافه کردن یک المان با (append) من یک کلید گذاشتم اضافه میشه ولی با هر بار کلیک کردن اضافه میشود اگر بخواییم یک...
۱۳۹۹/۰۶/۲۹

form amin
در:
سلام و خسته نباشید می خواستم Slash ( / ) رو به صورت اتوماتیک از تمامی URL ها حذف کنم البته با کمک htaccess...
۱۳۹۹/۰۶/۲۵

form محمد
در:
سلام وقت بخیر من یه همچین کدی دارم میشه لطفا راهنمایی کنید مشکلش چیه؟ سپاسگزارم
۱۳۹۹/۰۶/۲۴

form سروش
در:
سلام . من اطلاعات را از دیتابیس دریافت میکنم و در جدول میبینم . میخوام مثلا 6 مورد آخر را در یک ردیف ببینم و...
۱۳۹۹/۰۶/۲۳

form Behdad kanani
در:
سلام اصلا نمی شه اینکارو انجام داد
۱۳۹۹/۰۶/۲۲

form جعفری
در:
سلام قبل از تبدیل تاریخ نوشتید مثلا خب این مقدار تاریخ برای من در دیتابیس در جدولی بنام startedtm بصورت یونیکس...
۱۳۹۹/۰۶/۲۰

form احمد
در:
با عرض سلام و خسته نباشید ببخشید برای فایل دانلودی باید از چه دستوری استفاده کنم فایل با پسوند pdf رو میخوام بزارم اگر امکانش...
۱۳۹۹/۰۶/۲۰

form شیما
در:
سلام آیا وقتی در قسمت وبلاگ دوستان وبلاگی رو ثبت میکنیم صاحب وبلاگ با استفاده از برنامه های خاصی میتونه متوجه بشه ؟
۱۳۹۹/۰۶/۱۹

form mahtab
در:
سلام خسته نباشین من نمیدونم چرا مدیریت وبم برام باز میشه اما وقتی میزنم مشاهده وب رو میزنم نمیاره واسم فقط امیدوارم هک...
۱۳۹۹/۰۶/۱۸

form ساناز محمدی
در:
سلام دوباره این ساب دامین بنده هست اگر یکی از مطالب را مشاهده کنید هر محصولی که ارسال شده داخل از طریق مدیریت یک...
۱۳۹۹/۰۶/۱۷
form محسن
در:
سلام و عرض ادب مجدد کد زیر هم کار نمیکنه مشکل چی میتونه باشه ؟!
۱۳۹۹/۰۶/۱۶
form ساناز محمدي
در:
سلام بنده یه ساب دامین دارم با دامنه شخصی حالا داخل این ساب دامین طرف آدرس سایت خودش رو داخل ساب دامین بنده میذاره مثلآ...
۱۳۹۹/۰۶/۱۶
form محسن
در:
سلام و عرض ادب لطفا راهنمایی کنید مشکل کد زیر چیه
۱۳۹۹/۰۶/۱۶
form احمد
در:
با عرض سلام ببخشید من با دستور append میخوام یه سطری رو به سبدم اضافه کنم ولی چطوری باید داخل append تگ ها رو بنویسم...
۱۳۹۹/۰۶/۱۵
form احمد
در:
با عرض سلام مجدد و خسته نباشید ببخشید طبق فرمایش شما من قبل از دستور else دستور if را نوشته بودم تا جایی که اطلاع...
۱۳۹۹/۰۶/۱۴
form احمد
در:
با سلام ببخشید طریقه استفاده از دستور else در ایجکس به چه صورت هست ایا باید دوباره فانکشن دان رو در هنگام شرط گذاشتن بزارم....
۱۳۹۹/۰۶/۱۳
form مجتهد
در:
سلام ابتدائا از سایت مفیدتون تشکر می کنم. من خیلی از آموزه هام رو از سایت شما یاد گرفتم. یه سوال دارم...
۱۳۹۹/۰۶/۱۳
form محمد حسین
در:
سلام لینک های من وقتی کلیک شوند اررور 404 تولید میکنند. علت چیست؟
۱۳۹۹/۰۶/۱۱
form حسین
در:
سلام بر شما و با تشکر از سایت خوبتون. من ابتدا با تابع mail پیش رفتم و یک if ایجاد کردم که اگر ایمیل...
۱۳۹۹/۰۶/۱۰
form mehdi
در:
خطای -22 مربوط به چیه؟
۱۳۹۹/۰۶/۱۰
  در انتظار بررسی: ۰
 پاسخگویی به سوالات ممکن است تا 24 ساعت زمان ببرد.