شنبه ۲۲ شهریور ۱۴۰۴

Saturday, September 13, 2025 GMT +3:30

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

mysql-php-smart-login

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

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

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


سایت های زیادی را دیده ایم که در قسمت ورود اطلاعات نام کاربری و کلمه عبور معمولا با دکمه ای به صورت چک باکس امکانی تحت عنوان "مرا به خاطر بسپار" یا با عبارات مشابه ارائه می کنند، این امکان در واقع بدین معنی است که سیستم تا مدت زمان معینی در مراجعات بعدی از کاربر نام کاربری و کلمه عبور درخواست نکرده و به صورت هوشمندانه وضعیت ورود او را به یاد خواهد داشت، اما امکان "مرا به خاطر بسپار" چگونه در پنل مدیریت ایجاد می شود؟
اگر به صورت اجمالی بخواهیم ساز و کار این امکان را بررسی کنیم به این صورت خواهد بود:
هنگامی که گزینه یادآوری را انتخاب و فرم را ارسال می کنیم برنامه با دستورات PHP (یا JavaScript) یک کوکی (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 و حذف نشده است، مرورگر در هر درخواست مقدار کوکی را به سرور ارسال کرده و در نتیجه وضعیت دسترسی کاربر برای برنامه قابل شناسایی است و بدون ورود نام کاربری و کلمه عبور حتی با بستن پنجره مرورگر و مراجعه بعدی می تواند از امکانات بخش اعضاء سایت استفاده کند، اما قاعدتا انجام این موارد توسط برنامه نیاز به نوشتن دستورات و ایجاد تغییراتی در ساختار و کدهای پنل مدیریت سایت دارد که در ادامه خواهیم دید.
نکته: برای مشاهده اطلاعات و سربرگ هایی که بین واسط کاربری و سرور رد و بدل می شود می توانیم در مرورگرهای جدید با فشردن دکمه F12 از قسمت Console و Network استفاده کنیم.

ایجاد فولدری با نام php-mysql-signin-smart


برای تجمیع و مدیریت ساده تر فایل های سیستم هوشمند پنل ورود و خروج سایت بهتر است یک دایرکتوری مجزا در نظر بگیریم، در این آموزش ساز و کاری که در آموزش قبل در خصوص ایجاد پنل ورود و خروج سایت گفتیم را توسعه می دهیم با این حال برای جلوگیری از سردرگمی فولدری جداگانه با نام دلخواه php-mysql-signin-smart ساخته ایم که در مجموع چند فایل و فولدر زیر را شامل می شود:
- فایل config.php، حاوی اطاعات اتصال به دیتابیس.
- فایل create.php، جهت ایجاد جدول و ستون ها.
- فایل index.php، فرم HTML ورود به پنل مدیریت و ترکیب آن با دستورات PHP.
- فایل login.php، بررسی معتبر بودن اطلاعات ارسالی کاربران و انتقال به پنل مدیریت.
- فایل panel.php، پنل مدیریت کاربران سایت.
- فایل script.js، محاسبه و نمایش مدت زمان باقیمانده از نشست در سمت کاربر.
فایل access.php، بررسی وضعیت دسترسی کاربر و اعتبار نشست.
- فایل logout.php، انقضای کامل نشست و خروج از پنل مدیریت.
- فولدر lib، شامل فایلی با نام password_compat.php جهت استفاده از قابلیت های توابع جدید کلمه عبور در نسخه 5.5 و قدیمی تر PHP.
در ادامه با جزئیات این فایل ها را بررسی می کنیم.

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


طبق روال برای اتصال به دیتابیس نیاز به تنظیم چند پارامتر در تابع mysqli_connect است که شامل نام سرور، نام کاربری دیتابیس، کلمه عبور دیتابیس و در نهایت عنوان دیتابیس است، برای تجمیع این موارد و استفاده چندباره از تنظیمات پیکربندی در صفحات مختلف معمولا از فایل Configuration استفاده می شود که در این آموزش نیز بر همین اساس اطلاعات را در فایلی با نام دلخواه 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);
            $remember = mysqli_real_escape_string($conn, $remember);

            //نام جدول
            $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 با افزودن یک مقدار عددی به آن استفاده کرده ایم.
time() + $cookie_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'])) {
    //بررسی اعتبار کوکی و وضعیت یادآوری در دیتابیس
    $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 {
        //دریافت و ایمن سازی مقدار کوکی
        $panel_cookie = mysqli_real_escape_string($conn, $_COOKIE['panel']);
        
        //نام جدول
        $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 حاوی تنظیمات اتصال به دیتابیس در این فایل گنجانده شده است.
نکته: ساختار این برنامه جهت درک بهتر مطلب حالت ابتدایی و صرفا جنبه آموزشی دارد، هنگام کدنویسی برنامه های پیچیده می توانیم طوری ساختار را طراحی کنیم که چارچوب کار بهینه و به حداقل تعداد include نیاز باشد که معماری MVC (مخفف Model–View–Controller) یک نمونه متداول از آن است.

خروج از پنل کاربری با فایل 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
نکته: امکانات مرتبط با پنل کاربری و سیستم ورود و خروج سایت جزء قسمت های حساس برنامه های تحت وب هستند، لذا توصیه می شود قبل از استفاده های کاربردی تا حد امکان کدها در محیط آزمایشی تست و تمام جوانب ممکن بررسی شود.
دسته بندی: آموزش کاربردی » MySQL
related مطالب بیشتر:
آموزش حذف گروهی اطلاعات از MySQL با استفاده از چک باکس
آموزش ساخت فرم تماس با PHP و MySQL
تعویض کد امنیتی Captcha با Ajax و MySQL
نحوه رسم چارت و نمودار آماری با PHP و MySQL
جستجو در مطالب سایت با استفاده از MySQL Full-Text
دیدگاه
more ۸۵ دیدگاه برای این مطلب ارسال شده است.
more چینش دیدگاه ها به ترتیب از جدیدترین به قدیمی ترین است.
shokouhi70
۰۷:۱۹ ۱۳۹۳/۰۶/۱۳
با سلام دوباره خدمت شما اساتید محترم سایت وبگو
من میخواهم که در پنل مدیریتم یه قسمت داشته باشم برای منو و یه قسمت که اطلاعات حاوی هر دکمه منو را در آن قسمت نشان دهد و مشکل من این است که اگر از این کد استفاده کنم در لینک منو :
?pt=students
در عکس :
http://s5.picofile.com/file/8138718784/11.png
باعث میشه که این مقدار رو به همون صفحه مدیریت بفرسته ولی این کار باعث میشه که ارور سشن undefined رو به من بده به خاطر همین نمیدونستم باید چیکار کنم که این مشکل سشن را در همان صفحه مدیریت حل کنم
ممنون میشم کمکم کنید
به این صورت در هیچ سایتی نمی توانید پاسخ مشکلتان را پیدا کنید! حل آن نیاز به تست و بررسی دقیق هدف و ساختار کدهای شما دارد، متاسفانه صرف شرح ماجرا کمکی نمی کند.
shokouhi70
۱۸:۰۱ ۱۳۹۳/۰۶/۱۲
با سلام خدمت اساتید محترم سایت وبگو
من یه سوالی دارم که مربوط به پنل ادمین و سشن هاست
زمانی که در منوی ادمین دکمه ای را کلیک کنیم میخواهیم که در زیر آن منو قسمت مربوط به دکمه نشان داده شود مانند عکس زیر :
http://s5.picofile.com/file/8138718784/11.png
و عکس کد :
http://s5.picofile.com/file/8138719868/Desktop_9_3_2014_5_43_20_PM_872.jpg
ممنون میشم اگه کمک کنید
سوالتان خیلی مبهم است، تصاویرتان گویا نیست!
در حد برداشت ما، قاعدتا باید این مورد را طراحی کنید، پاسخ ساده و کوتاهی ندارد، باید مطابق نیازتان خوجی را از دیتابیس گرفته و به صورت HTML چاپ کنید، در واقع ترکیبی از PHP، HTML و CSS مورد نیاز است.
۱۰:۰۲ ۱۳۹۳/۰۶/۰۶
سلام
من مدیریت یه سایت رو برعهده دارم که برنامه نویسیش از قبل نوشته شده و من فقط گاهی کمی اون رو ویرایش میکنم
اما چون سایت با mvc نوشته شده خیلی سخته و گاهی متوجه نمیشم سوالم اینه که چطوری باید عضویت کاربرم فقط 1 روز اعتبار داشته باشه؟ یعنی بعد یک روز اون کاربر حذف شه
راه حل خودم اینه که یه کوکی بسازم که 24 ساعت بعد بیاد از جدول کاربرام ، کاربر مورد نظر و حذف کنه. اما نمیدونم کد رو کجا و چطوری بنویسم
ممنون
کوکی در مرورگر کاربر ساخته می شود و لذا راه حل مناسبی برای این گونه موارد نیست!
برای ایجاد کاربران موقت باید یک ستون برای تاریخ عضویت داشته باشید و یک کد یا تابع که در فواصل زمانی معین توسط کاربران اجرا شود، این کد تاریخ فعلی را با تاریخ کاربران چک کرده و در صورت منقضی شدن زمان، اطلاعات آنها را حذف می کند، قاعدتا برای نوشتن این قابلیت باید با برنامه نویسی وب آشنا باشید، متاسفانه در این زمینه کمک بیشتری از دستمان ساخته نیست!
Ali Zeus
۲۱:۰۷ ۱۳۹۳/۰۵/۲۸
با سلام مجدد و بسیار ممنون از پاسختون
مشکل بنده با فانکشنی که در پست دیگری ارائه دادین
function mysql_real_escape_string_in($mysql_real_escape_string){
$search = array("\\","\0","\n","\r","\x1a","'",'"');
$replace = array("\\\\","\\0","\\n","\\r","\Z","\'",'
\"');
return str_replace($search, $replace, $mysql_real_escape_string);
}
حل شد و فقط میخاستم بدونم چرا mysql_real_escape_string خروجیش 0 میشه که جواب دادین..... بسیار ممنون
امین
۱۰:۰۴ ۱۳۹۳/۰۵/۲۸
با سلام و عرض خسته نباشید.
ابتدا از پاسخی که به سوال دفعه پیشم دادید تشکر میکنم. دفعه قبل در مورد نحوه اتصال خودکار کاربر به صفحه شخصی وب سایت و استفاده از کوکی و... پرسیدم و شما گفتید از کوکی مدت دار و ذخیره مقدار اون در دیتابیس میشه استفاده کرد. حالا سوال من اینه که این دیتابیس محلیه یعنی تو سیستم کاربره یا دیتابیس سروره. اگه در دیتابیس سرور باشه چه جوری سرور سیستم هر کاربر رو شناسایی میکنه و در نتیجه اطلاعات هر فرد رو در میاره به عبارتی ایا برای هر سیستم در دیتابیس شناسه ای تعریف میکنه؟ نمی دونم تونستم مطلب رو برسونم.
به نظر درک شما از نحوه کار کوکی و سشن کامل نیست!
در وب دو مبحث به نام کوکی و سشن داریم، در سشن یا نشست به زبان ساده دو اتفاق رخ می دهد، یک کد شناسایی برای کابر در سرور ایجاد می شود و اطلاعات همان کد به صورت کوکی با مدت زمان انقضای مشخص در مرورگر کاربر ایجاد می گردد، از این به بعد هر زمان که صفحه مورد نظر یا صفحه دیگری از سایت مورد نظر (بسته به path تنظیم شده در کوکی) را فراخوانی کنید، مرورگر اطلاعات کوکی را به سرور ارسال کرده و سرور بررسی می کند تا ببیند چنین اطلاعاتی مطابق با یک سشن است یا خیر!
در مورد تنظیم کوکی و ثبت در دیتابیس نیز چنین اتفاقی رخ می دهد، یک کوکی در مرورگر کاربر ایجاد می کنیم و مقدار آن را در دیتابیس سرور نیز ذخیره می کنیم، از این به بعد هر زمان صفحه مورد نظر را فراخوانی کنید، مرورگر اطلاعات کوکی را به سرور ارسال می کند، در سرور می توان اطلاعات ارسالی را با مقادیر موجود در دیتابیس تطبیق داد.
نکته: مرورگر تنها زمانی اطلاعات کوکی را ارسال می کند که:
1- تاریخ انقضای آن نگذشته باشد.
2- path تنظیم شده برای کوکی ریشه سایت یا دایرکتوری سطح بالاتر از دایرکتوری (یا خود دایرکتوری) درخواستی باشد.
Ali Zeus
۰۱:۳۱ ۱۳۹۳/۰۵/۲۸
با سلام و بسیار ممنون از پاسختون
تست کردم ولی بازم کوکی نمیده
هم بصورت
setcookie('login_panel_error',$panel_error+1,time()+300,("/"));
و هم بصورت
setcookie('login_panel_error',$panel_error+1,time()+300, "/");
به اجبار با جاوا کوکی میدم
{$panel_error=$panel_error+1;
echo'<script type="text/javascript">document.cookie = "login_panel_error='.$panel_error.'; max-age=300"; alert("اطلاعات وارد شده صحیح نیست !"); window.location.href = "login.php";</script>'; exit;}
و با عرض معذرت سوال دیگه ای که داشتم اینه که ایا
mysql_real_escape_string
فقط در اتصال mysql_connect کاربرد داره؟
چون من با یک فانکشن
<?php function db_connect(){
$result = new mysqli('localhost', 'un', 'pass', 'db');
if(!$result){throw new Exception('dont connect');}
else{return $result;}} ?>
دیتابیس رو به فایل متصل میکنم و هر کلمه ای که از
mysql_real_escape_string
عبور میکنه خروجیش صفر میشه
در مورد مشکل اول کمک خاصی نمی توان کرد، Syntax مشکلی ندارد لذا نیاز به تست آنلاین دارد!
در مورد تابع
mysql_real_escape_string
این تابع نیاز به یک اتصال از نوع mysql_connect دارد، در صورتی که کدهای شما با اکستنشن MySQLi نوشته شده، لذا باید از توابع مربوط به MySQLi استفاده کنید، ماتند روش prepare و bind_param یا تابع mysqli_real_escape_string.
Ali Zeus
۱۴:۵۰ ۱۳۹۳/۰۵/۲۷
با سلام و ممنون از اموزش های خوبتون
سوالی داشتم که شاید ربطی به این قسمت نداشته باشه ولی ممنون میشم کمک کنید
برای فرم ورود به پنل مدیریت قابلیتی نوشتم که اگه کاربر بیش از سه بار اطلاعاتش رو اشتباه وارد کرد به مدت 5 دقیقه از ورودش جلوگیری کنه اما فقط در لوکال هاست کار میکنه
اینم سورسش
@$panel_error=$_COOKIE['login_panel_error'];
if(!$panel_error){$panel_error=0;}
if($panel_error>3){echo'به مدت 5 دقیقه اجازه ورود ندارید'; exit;}
//اگر اطلاعات صحیح نبود
if (!$check_user->num_rows){setcookie('login_panel_error',$panel_error+1,time()+300); echo'اطلاعات وارد شده صحیح نیست';}
این قسمت مربوط به ارسال کوکی و بررسی کوکی هست
طبق اطلاعات ارائه شده تنها ایرادی که می توان به آن پی برد عدم تنظیم پارامتر path برای تابع setcookie است، بهتر است در انتها از علامت / برای تعیین path کوکی استفاده کنید (در نمونه کدهای آموزش حاضر این حالت استفاده شده).
امین
۲۰:۱۳ ۱۳۹۳/۰۵/۲۶
با سلام خدمت شما.
من توی وب سایتم یک دیتابیس و یک فرم برای نام کاربری و رمز عبور دارم. اگه فردی که عضو سایته بخواد بصورت دستی نام کاربری بده و وارد سایت بشه مشکلی نیست اما چه کار کنم که مثل فیسبوک هر بار که وارد میشه مستقیم وارد صفحه خودش بشه. تمامی مراحلی که شما گفتید رو انجام دادم مثل سشن و کوکی ولی هر بار که وارد سایت میشم بازم صفحه اول سایت میاد و نام کاربری و رمز می خواد یعنی خودش وارد نمیشه. البته لازم به ذکر هستش که من از wampserver استفاده میکنم به نظر شما مشکل میتونه استفاده از wamp باشه یا جای دیگست. لطفا راهنمایی کنید. با تشکر
صرف استفاده از سشن و کوکی و... به خودی خود مشکلی را حل نمی کند! در واقع باید بدانید این مباحث چه کاربردی دارند و نحوه کار آنها را به خوبی درک کنید، در این صورت هدف مورد نظرتان علاوه بر روش آموزش داده شده، به راحتی با ایجاد یک کوکی مدت دار و ذخیره مقدار آن در دیتابیس و تطبیق اطلاعات ارسالی کوکی با دیتابیس هنگام ورود مجدد کاربر، قابل ایجاد است.
در این رابطه تفاوت خاصی بین سرور مجازی یا سایت آنلاین وجود ندارد، مگر اینکه امکاناتی مربوط به کوکی در سرور مجازی یا مرورگرتان محدود شده باشد که معمولا به صورت پیش فرض محدودیتی وجود ندارد.
hadi0098
۲۰:۵۷ ۱۳۹۳/۰۵/۲۱
سلام و عرض خسته نباشید
میخواستم بدونم آیا راهی وجود داره که بشه توی این پنل ورود رو برای بعضی از کاربران غیر فعال کرد البته من این کار رو انجام دادم اما به یه مشکلی مواجه شدم این کد
$result = mysql_query ("SELECT * FROM user WHERE username = '$username' AND userpw = '$password' AND userprivilegien != 'mod'");
توی این کد فقط کاربران mod می تونن وارد بشن ولی من میخوام که کاربران mod و admins بتونن وارد پنل بشن
اگه توضیحی بدید ممنون میشم
آنچه که از نمونه کد شما برداشت می شود، اتفاقا تنها کاربران غیر از سطح دسترسی "mod" انتخاب می شوند، در هر صورت با افزودن یک AND (یا در صورت نیاز OR) دیگر به پرس و جوی خود می توانید سطح "admins" را هم اضافه کنید (البته ظاهرا پرس و جوی فعلی شما برعکس هدف شما است!)
fati
۱۱:۲۹ ۱۳۹۳/۰۵/۱۲
سلام.
چیکار کنیم که وقتی مقدار سشن ها رو تو صفحات مختلف انتقال دادیم، به هنگام ریلود شدن صفحه ، از دستش ندیم و حفظ بشوند . مال من وقتی تازه لاگین میکنم سشنو میشناسه و چاپم میکنه اما وقتی رفرش میکنم، برای بعضیاشون پیعامNotice: Undefined index میده.
اگر به درستی از تابع سشن در صفحات مختلف استفاده کرده باشید، با رفرش صفحه اتفاق خاصی نمی افتد، پیام ممکن است ناشی از موارد دیگر باشد که در هر حال بدون بررسی دقیق کدها نمی توان نظر خاصی داد.
محمد
۱۸:۰۴ ۱۳۹۳/۰۴/۲۳
سلام
دستورات به درستی کار میکنه ولی زمان در setcookie اینطوری نیست که شما نوشتید مثلا من زمان رو 60 (یک دقیقه) تنظیم کردم ولی بیشتر از یک دقیقه است !
یه چیز دیگه هم برای من مبهمه ، اینکه شما فقط این کد را نوشتید
setcookie(session_name(),session_id(),time()+$login_time,("/"));
و البته درست هم کار می کند ولی چگونه بعد از منقضی شدن کوکی ، سشن هم منقضی میشه !!!؟؟ در این دستورات هیجا از unset برای سشن ها استفاده نشده
- زمان مشکلی ندارد! ممکن است مجددا کوکی را به روز رسانی و در نتیجه زمان را تمدید کرده باشید، برگرفته از راهنمای PHP:
 time() + 60 * 60 * 24 * 30 will set the cookie to expire in 30 days
یعنی 60 ثانیه در 60 دقیقه در 24 ساعت در 30 روز که عدد 30 روز به ثانیه به دست می آید، لذا 60 اول به میزان همان 60 ثانیه است.
http://php.net/manual/en/function.setcookie.php
- نیازی به unset کردن سشن در این مرحله نیست، در اینجا بر اساس زمان باقی مانده کاربر کد تنظیم شده، در مرحله دوم (اگر زمان کاربر منقضی شده بود) می توان او را به یک صفحه دیگر منتقل و سشن را در آنجا unset کرد، البته الزامی هم وجود ندارد، فرض کنید کاربر مرورگر را ببندد یا برق قطع شود!
manizhe
۰۴:۰۰ ۱۳۹۳/۰۱/۰۶
سلام من در پنل مدیریت ام وقتی روی دکمه ویرایش یک عکس کلیک می کنم عکس جدید در بانک جایگزین قبلی می شود ولی در پوشه ای که همه ی عکس ها ذخیره می شود عکس قبلی باقی است چطوری اون عکس را حذف کنم از پوشه ممنون میشم که اگه سریع جواب بدین خیلی ضروری است
روش ساده ای وجود ندارد! باید برنامه نویسی بلد باشید و از توابعی مانند unlink در PHP استفاده کنید، البته تغییر کدها بدون داشتن تسلط لازم توصیه نمی شود.
ebrahim
۱۳:۰۹ ۱۳۹۲/۱۲/۰۵
سلام آموزشهای سایتتون فوق العاده اند. لطفا با قدرت به کارتون ادامه بدین.
من مطالب رو مو به مو خوندم و مثل آموزشهایی که شما توضیح دادین انجام دادم.
ولی گزینه مرا به خاطر بسپارم کار نمی کنه یعنی چه تیک داشته باشه چه تیک نداشته باشه یه کار رو انجام میده
http://isasak.ir/sabtname/New/vorud.php
کد مشکلی ندارد، منتها یک بخش دیگر باید به آن اضافه شود، حالتی که کاربر ابتدا دکمه مرا به خاطر بسپار را می زند و کوکی تنظیم می شود، سپس از سایت خارج شده و مجددا بدون تیک زدن چک باکس وارد می شود، در اینجا باید زمان کوکی 0 یا همان at the end of session تعیین شود، بدین منظور یک دستور شرطی دیگر تعریف کنید:
//بررسی امکان مرا به خاطر بسپار
elseif (isset($_SESSION['remember']) && $_SESSION['remember'] == '1'){
$login_time = 3600;//یک ساعت
setcookie(session_name(),session_id(),time()+$login_time,("/"));
}
else{
$login_time = 0;//صفر
setcookie(session_name(),session_id(),time()+$login_time,("/"));
}
نکته: راه حل تست نشده!
مهران
۱۱:۳۱ ۱۳۹۲/۱۲/۰۳
ماها عادت داریم مطالب خیلی خوب و مجانی که میبینیم تشکر نکنیم ولی بسیار بسیار ممنون از اینکه بدون چشمداشت جهت اشتراک گذاری این مطالب تلاش میکنین.
rasool
۲۰:۰۸ ۱۳۹۲/۰۹/۳۰
چطور میشه بعد از تموم شدن زمان به طور اتومات به یک صفحه حاوی اطلاعیه تموم شدن زمان منتقل بشیم. (از انتخاب مرا به خاطر داشته باش)
elseif (isset($_SESSION['SESS_MEMBER_ID']) || (trim($_SESSION['SESS_MEMBER_ID']) =='1')){
$login_time=10;//یک ساعت
setcookie(session_name(),session_id(),time()+$login_time,("/"));
}
طراحی این موارد نیازمند تسلط کامل شما به مباحث سشن، کوکی و تاریخ و زمان در PHP است، در این صورت به روش های مختلف می توان این کار را انجام داد، به طور مثال، هم زمان با ایجاد کوکی یادآوری، یک کوکی با مدت زمان ماندگاری بیشتر ایجاد کنید و مقادیر value آن به فرض تاریخ انقضای کوکی اصلی باشد، سپس با مقایسه تاریخ فعلی و تاریخ انقضای کوکی می توان به نتیجه مورد نظر رسید (یا هر روش ابتکاری دیگری).
more لطفا پیش از ارسال دیدگاه نکات زیر را مد نظر داشته باشید:
- به سوالات کلی، زمانبر، مبهم و مشکلاتی که تلاشی برای رفع آنها نکرده باشید پاسخ مختصر داده شده یا به بخش برنامه نویسی اختصاصی ارجاع داده می شوند.
- کدها و اسکریپت های طولانی را ترجیحا در یک صفحه وب آنلاین یا به صورت حساب موقت و آزمایشی قرار دهید تا امکان بررسی دقیق مشکل و خطایابی میسر باشد.
- تمام دیدگاه های ارسالی خوانده شده و برای هر کاربر مدت زمان لازم جهت پاسخگویی در نظر گرفته می شود، لطفا از طرح سوالات متعدد در بازه زمانی کوتاه خودداری کنید.



 refresh
10 × 10
9 × 3
20 × 20
=