شنبه ۰۸ مهر ۱۴۰۲

Saturday, September 30, 2023 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
جستجو در مطالب سایت با استفاده از MySQL Full-Text
آموزش ساخت فرم عضویت در سایت با PHP و MySQL
آموزش ساخت پنل ورود و خروج سایت با PHP و MySQL
دیدگاه
more ۸۵ دیدگاه برای این مطلب ارسال شده است.
more دیدگاه جدید بر اساس تاریخ ارسال در انتهای دیدگاه های موجود نمایش داده می شود.
mahdi
۲۳:۳۵ ۱۳۹۲/۰۴/۲۲
سلام خیلی خیلی ممنون از شما ولی یه مشکل :
الان من کدی که بهم دادید رو به این شکل تغییر دادم :
<?php
$con = mysql_connect("mysql.xzn.ir", "u864181118_rav", "pass")
or die(mysql_error());
//نام دیتابیس
mysql_select_db("u864181118_mahdi", $con)
or die(mysql_error());
//پرس و جوی فرضی
$result = mysql_query ("SELECT country, username, password FROM register WHERE username = '$username' AND password = '$password'")
or die(mysql_error());
while($row = mysql_fetch_array($result)){
$_SESSION['country'] = $row['country'];
$_SESSION['username'] = $row['username'];
$_SESSION['password'] = $row['password'];
}
//گرفتن خروجی فرضی
echo $_SESSION['country'];
echo "<br>";
echo $_SESSION['username'];
echo "<br>";
echo $_SESSION['password'];
mysql_close($con);
?>
اما متاسفانه country رو نمیده و چاپ نمی کنه ؟! باید چیکار کرد ؟ آیا کد اشکال داره ؟ممنون
بحث سشن ها جزء مباحث پیشرفته برنامه نویسی PHP است که به نظر شما با آن هنوز کامل آشنا نیستید، یک نشست با تابع
session_start();
باید شروع شود، یعنی این تابع را در ابتدای هر صفحه ای که می خواهید نشست در آن ادامه داشته باشد باید قرار داده باشید.
mahdi
۱۲:۰۸ ۱۳۹۲/۰۴/۲۳
سلام مجدد ...
خیلی ممنون ولی من این کار رو انجام داده بودم ! این کد صفحه کنترل پنل کاربری که بعد از ورود به آنجا منتقل می شوند :
کد حذف شد
حال می خواهم در قسمتی که مشخص شده است ، وقتی کاربری لوگین کرد فقط اطلاعات مربوط به خودش مثل کشور ، نام کاربری و پسورد رو ببینه !
میشه لطفا راهنمایی کنید یا برای من انجام بدید ؟ ممنون ...
کدها در تئوری مشکلی ندارند، برای خطایابی تابع زیر را در ابتدای صفحه قرار دهید:
error_reporting(-1);
برای اطمینان از اینکه پرس و جوی دیتابیس، یک نتیجه بر می گرداند، در زیر پرس و جو کد زیر را قرار دهید:
echo mysqli_num_rows($query);
اگر نام کاربری و کلمه عبور صحیح باشد، این کد مقدار 1 را چاپ می کند، در ضمن توجه کنید که ستون های country, username, password باید در جدول register موجود باشند.
نکته: به جهت محدودیت هایی که بعضا هاست های رایگان دارند، بهتر است کدها را در لوکال سرور امتحان کنید.
یوسفی
۱۹:۵۶ ۱۳۹۲/۰۴/۳۱
واقعا آموزش هاتون عالیه. انشاالله که تو همه عرصه ها موفق باشید.
نورمند
۱۴:۴۸ ۱۳۹۲/۰۵/۱۷
بسیار عالی بود انشا الله همیشه موفق باشید
رضا
۰۱:۱۱ ۱۳۹۲/۰۶/۱۹
سلام
خیلی آموزش عالی بود
من یه مشکل دارم من توی سایتم دو تا پنل لاگین دارم
میخوام این اتفاق فقط روی یکی از پنل ها بیفته ولی روی هر دوتا پنل انجام میشه
لطفا راهنمایی کنید
پاسخ به سوالتان نیازمند داشتن جزئیات بیشتری از مشکل و هدف شما است، به عبارتی بدون بررسی دقیق سورس کدهای سایت نمی توان خیلی کمک کرد.
رضا
۱۸:۱۱ ۱۳۹۲/۰۶/۱۹
من توی سایتم یه قسمت برای مدیریت وجود داره که فقط مدیران میتونن وارد بشن و یه قسمت هم آزاد هست برای کاربران و نام سشن هایی که نشست می کنن کاملا متفاوت هستن مثلا برای مدیریت من سشن های
$_SESSION['admin_username']
$_SESSION['admin_password']
رو ست کردم و برای کاربران سشن های
$_SESSION['user_username']
$_SESSION['user_password']
رو ست کردم
حالا من میخوام که فقط سشن های مربوط به کاربران به خاطر بسپاره
به روش های مختلف می توانید ست شدن کوکی یادآوری را محدود به یک گروه کاربری کنید، به طور مثال می توانید یک شرط if تعریف کنید که اگر session_name برابر به فرض admin_username بود، قسمت کوکی یادآوری اجرا شود، یا اگر سشن admin_username ست شده بود، این قسمت اجرا شود، روش دیگر این است که در قسمت فرم ورود، یک input از نوع hidden بسازید و مقادیر آن را پیش از ست کردن کوکی بررسی کرده و متناسب با آن کوکی را اجرا کنید.
در واقع نوشتن این امکان نیازمند اندکی آشنایی و تجربه قبلی کار با PHP است.
رضا
۱۰:۴۲ ۱۳۹۲/۰۶/۲۰
اون دو تا پنلی که گفتم، اطلاعاتشون به دوتا صفحه مختلف ارسال میشه.
من کوکی یادآوری رو توی صفحه ای که مربوط به کاربران هست اجرا می کنم یعنی بعد از اینکه سشن ها ست شد خط بعد کوکی اجرا میشه
ولی باز زمانی که وارد پنل مدیریت میشم و مرورگر رو میبندم و دوباره باز می کنم میبینم که پنل مدیریت بدون اینکه صفحه لاگین بیاد باز میشه
ببخشید من خیلی سوال می کنم
کار با کوکی ها نیازمند داشتن تجربه بالا در برنامه نویسی وب و زبان هایی مانند PHP و JavaScript است، در این صورت می توانید تاریخ انقضای کوکی های ست شده را از طریق مرورگر خود بررسی کنید و اگر تاریخ کوکی برای کاربران عادی بیشتر از یک روز بود، مطمئن شوید که ایراد از کدنویسی شما است که کوکی را برای همه ست می کند، ضمن اینکه باید با نحوه حذف کوکی از مرورگر جهت تست و آزمایش و خطا آشنا باشید.
مهنا
۲۱:۰۸ ۱۳۹۲/۰۷/۲۸
سلام خسته نباشید. از آموزشهای بسیار مفیدتون تشکر می کنم. فقط من یه سوال داشتم من وقتی می خواستم وارد شم با اینکه اطلاعات با دیتابیس تطبیق داشت ارور میداد ولی وقتی خط مربوط به md5 رو غیر فعال کردم پیام داد که در حال استفاده از بخش کاربری هستید. اشکال کار از کجاست؟ چرا نمیشه از این تابع استفاده کنم؟ مثلا خودم یه یوزر و پسورد تو دیتابیس insert کردم ولی اونجا که با md5 رمزنگاری نمیشه درسته؟ چطور این مشکل برطرف میشه؟
ممنون میشم اگه راهنمایی بفرمایید
هنگام درج (INSERT) اطلاعات نیز باید از طریق کدهای PHP این کار انجام دهید (به فرض ایجاد فرم ثبت نام) و پیش از درج، مقادیر متغیرها را با تابع md5 رمزی و سپس در دیتابیس ذخیره کنید، به عبارتی اگر به دیتابیس خود توجه کنید، مقادیر رمز باید به صورت کدهای رندوم تبدیل شده باشند (مرحله درج و تطبیق باید هر دو از md5 استفاده کنند).
مهنا
۱۰:۴۰ ۱۳۹۲/۰۸/۰۵
سلام. ممنون از راهنماییتون. یه سوال دیگه داشتم اگه ما بخواهیم این پیغام تو کل صفحات بیاد باید چه کار کنیم؟ چون وقتی آدرسه این صفحات رو به جای user.php میذارم وقتی میخوام برم تو اون صفحه چون وارد نشدم میفرسته به صفحه اول سایت باید از دستور خاصی همراه session استفاده بشه؟ تشکر
باید سشن در تمام صفحاتی که نیاز به برقراری نشست است ادامه داشته باشد (استفاده شود)، تا به این صورت سشن آی دی های ست شده در صفحات قبل، قابل انتقال از صفحه ای به صفحه دیگر و به تبع قابل بررسی باشند (که اگر ست شده بود، ادامه کدها اجرا شود، در غیر این صورت به صفحه ورود برگردد).
ابوالفضل
۱۸:۱۴ ۱۳۹۲/۰۸/۱۷
سلام . با تشکر از اطلاعات بسیار خوبتون . یه سوال دارم. من می خوام تو سایتم امکان این رو بذارم که کارمندا با وارد کردن نام کاربری و پسوردشون هر ماه فیش حقوقیشون رو ببینن . چطوری می تونم این کار رو انجام بدم ؟ بعد از ایجاد دسترسی برای هر کدومشون چطوری می تونم تعریف کنم که هر کارمند فیش حقوقی خودش رو ببینه ؟ لطفا راهنمایی کنید . متشکرم.
سوالتان کلی است، نوشتن این سیستم نیازمند درک مباحثی نسبتا پیشرفته مانند سشن ها و تجربه لازم در کار با MySQL است، در واقع باید برای هر کاربر یک ستون اختصاصی در دیتابیس داشته باشید و در هنگام فراخوانی اطلاعات، متناسب با نام کاربری و کلمه عبور، اطلاعات ستون یا ستون های متناظر در جداول را فراخوانی کنید (برای طراحی این سیستم قاعدتا باید ابتدا مباحث مقدماتی را مطالعه و تمرین و تکرار کافی داشته باشید).
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 آن به فرض تاریخ انقضای کوکی اصلی باشد، سپس با مقایسه تاریخ فعلی و تاریخ انقضای کوکی می توان به نتیجه مورد نظر رسید (یا هر روش ابتکاری دیگری).
مهران
۱۱:۳۱ ۱۳۹۲/۱۲/۰۳
ماها عادت داریم مطالب خیلی خوب و مجانی که میبینیم تشکر نکنیم ولی بسیار بسیار ممنون از اینکه بدون چشمداشت جهت اشتراک گذاری این مطالب تلاش میکنین.
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,("/"));
}
نکته: راه حل تست نشده!
manizhe
۰۴:۰۰ ۱۳۹۳/۰۱/۰۶
سلام من در پنل مدیریت ام وقتی روی دکمه ویرایش یک عکس کلیک می کنم عکس جدید در بانک جایگزین قبلی می شود ولی در پوشه ای که همه ی عکس ها ذخیره می شود عکس قبلی باقی است چطوری اون عکس را حذف کنم از پوشه ممنون میشم که اگه سریع جواب بدین خیلی ضروری است
روش ساده ای وجود ندارد! باید برنامه نویسی بلد باشید و از توابعی مانند unlink در PHP استفاده کنید، البته تغییر کدها بدون داشتن تسلط لازم توصیه نمی شود.
محمد
۱۸:۰۴ ۱۳۹۳/۰۴/۲۳
سلام
دستورات به درستی کار میکنه ولی زمان در 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 کرد، البته الزامی هم وجود ندارد، فرض کنید کاربر مرورگر را ببندد یا برق قطع شود!
more لطفا پیش از ارسال دیدگاه نکات زیر را مد نظر داشته باشید:
- به سوالات کلی، زمانبر، مبهم و مشکلاتی که تلاشی برای رفع آنها نکرده باشید پاسخ مختصر داده شده یا به بخش برنامه نویسی اختصاصی ارجاع داده می شوند.
- کدها و اسکریپت های طولانی را ترجیحا در یک صفحه وب آنلاین یا به صورت حساب موقت و آزمایشی قرار دهید تا امکان بررسی دقیق مشکل و خطایابی میسر باشد.
- تمام دیدگاه های ارسالی خوانده شده و برای هر کاربر مدت زمان لازم جهت پاسخگویی در نظر گرفته می شود، لطفا از طرح سوالات متعدد در بازه زمانی کوتاه خودداری کنید.



 refresh
10 × 10
7 × 9
20 × 20
=
آخرین دیدگاه ها
more برای دسترسی سریع به یادداشت مربوطه می توانید از لینک مطلب در کادر زیر استفاده کنید.
javascript
سلام 🙌 توی جاوا اسکریپت چطور تگ آپشن رو پیش فرض بزاریم مثلا : تگ آپشن _استان محل صدور : همدان -...
۱۴۰۲/۰۷/۰۶

Fateme
سلام مطالب عالی و استفاده می کنم دستتان درد نکنه . سوال من این هست چطور میتونم تاریخ تولد را وارد کنم سن شخص رو...
۱۴۰۲/۰۷/۰۵

مجتهدزاده
سلام من در لاراول سعی دارم تا با یک لینک رو اجرا کنم تا یک اس ام اس به شماره ای که در...
۱۴۰۲/۰۷/۰۳

پـــرتو
سلام جناب خسته نباشید شرمنده مزاحم میشم عه جناب ببخشید من دو سه تا وبلاگ دارم که یکیش یا دوتاشون هک میشن یعنی...
۱۴۰۲/۰۷/۰۳

فاطمه سیداحمدی
با سلام و خسته نباشید من میخوام با استفاده از جاوااسکریپت یه کدی داشته باشم که وقتی کاربر چیزی رو از توی سایت من...
۱۴۰۲/۰۷/۰۲

رضا جهانیان
درود برشما من تو گوگل که اسم سایت حراجستون را سرچ می کنم اسم دامنه نوشته شده به انگلیسی کدهای زیادی هم استفاده کردم...
۱۴۰۲/۰۶/۳۱

سعید
نمای ظاهری من مثل قبل که در اوت لوک میفرستادم در نیو ایمیل نیست یعنی تغییر کرده چطور میتونم برگردم به همون حالت اول خودش...
۱۴۰۲/۰۶/۲۷

سینا
با سلام و خسته نباشید من برای ترجمه متن از گوگل ترانسلیت استفاده می کنم یه مشکلی که دارم اینکه وقتی متنی رو برام...
۱۴۰۲/۰۶/۲۵

امین رحیمی
عالی بود ممنون
۱۴۰۲/۰۶/۲۴

کوروش
سلام خسته نباشید توضیح : سایتی که تگ های select option ها و همه div ها و label ها ( کلاس های...
۱۴۰۲/۰۶/۲۴

عدم اجرای فونت فیس در تب منو
سلام برای تب منو فونت فیس برای تب اول کار میکند اما تب های بعدی اعمال نمیشود چکار باید کرد
۱۴۰۲/۰۶/۲۲

محبوبه قاسم پور
سلام وقتتون بخیر برای منوهای کشویی سایت، اگر خاصیت overflow:hidden باشد، زیرمنوها نمایش داده نمی شود، از طرفی اگر این خاصیت فعال نباشد...
۱۴۰۲/۰۶/۱۸

mehdi
سلام من امتحان کردم نشد
۱۴۰۲/۰۶/۱۵

پـــرتو
سلام فکنم در یادداشت ها ارسال شده بود و تاریخشم یادم نیست ولی من واقعا نمی دونم کامنتش کجاست که برم برش دارم ...
۱۴۰۲/۰۶/۱۴

پـــرتو
سلام خسته نباشید ببخشید مزاحم شدم می خواستم بگم که یه کدی داده بودین قبلا برای بلاگفا که آی پی کسایی که بهمون توهین می...
۱۴۰۲/۰۶/۱۱

امیرحسین رستمی
سلام مجدد وقت بخیر لطفا لینک زیر را ببینید میخوام تمام متن های فارسی که اینجا انتخاب کردم رو با عبارت با قائده انتخاب...
۱۴۰۲/۰۶/۱۱

امیرحسین رستمی
سلام وقت بخیر من این متنو دارم: میخوام فقط متن های فارسی رو انتخاب کنم با preg_match و متن جایگزین براش بزارم قبلا...
۱۴۰۲/۰۶/۱۱

امیرحسین رستمی
در رابطه با دوتا سوال قبلی من این کد رو نوشتم شما بررسی کنید ببینید اگه مشکل دیگه ای نداره ، درست داره کار میکنه...
۱۴۰۲/۰۶/۰۹

امیرحسین رستمی
سلام وقت بخیر من یک عبارات با قائده ای میخوام که این متن رو : من همچین عبارتی در نظرم هست ولی ...
۱۴۰۲/۰۶/۰۹

hossein
سلام مجدد، ممنون از پاسخگویی شما از وردپرس استفاده میکنم و قالب وودمارت، هدر هم با هدر ساز قالب ساخته شده با اینکه...
۱۴۰۲/۰۶/۰۹

امیرحسین
سلام وقت بخیر من یک برنامه نوشتم که از ایپی ای گوگل کمک میگیره و متن میدم بهش و ترجمه میکنه : شما فرض...
۱۴۰۲/۰۶/۰۹

hossein
سلام وقت شما بخیر باشه ابتدا جا داره از شما تشکر کنم میخوام یک آیکن لینک دار توی هدر سایتم را از دید...
۱۴۰۲/۰۶/۰۹

علی
سلام و با تشکر از راهنمایی شما بنده یک وب سرویس ارسال پیامک دارم که از طریق آن برای کابرانم پیامک ارسال می کنم....
۱۴۰۲/۰۶/۰۷

مهدی
سلا چطوری میتونم کدام تو مرورگر دقیق نمایش بدم حتی >< تگ هام نمایش داده بشه در html تگ کد را هم معرف نکنید چون...
۱۴۰۲/۰۶/۰۴

محمدرضا
سلام و خدا قوت حضور شما بزرگوار. اول یه تشکر و قدردانی از شما دارم که زمان میزارید و جواب همه دوستان رو میدید....
۱۴۰۲/۰۶/۰۳

محمدرضا
سلام وقت شما بخیر لطف بفرمائید با switch و default کدی رو بنویسید که: اگه متغیر a یک باشد متغیر b دو...
۱۴۰۲/۰۶/۰۱

پـــرتو
سلام بابت جواب به سوالم در مورد بلاگفا ممنون من خودم با پشتیبانی بلاگفا تماس گرفتم گفنن نمیشه اینکار ولی یه سوال دیگه ازتون داشتم...
۱۴۰۲/۰۵/۲۸

fatima
سلام. خسته نباشید. ما با responsive هم میتوانیم اسکرول افقی را حذف کنیم؟
۱۴۰۲/۰۵/۲۸

عاطفه
ممنون از پاسخگوییتون. اگر ممکنه این سوال رو هم جواب بدید ممنون میشم. معنی این ارور ها و راه حل رفع اونها رو میخواستم بدونم....
۱۴۰۲/۰۵/۲۶

عاطفه
سلام، وقت بخیر. من چندتا مشکل سایتم داره که نمیدونم چطور باید برطرفشون کنم. یکی اینکه وقت لینکای شکسته رو چک میکنم برای لینکدین و...
۱۴۰۲/۰۵/۲۵
  در انتظار بررسی: ۰
 پاسخگویی به سوالات ممکن است تا ۲۴ ساعت زمان ببرد.