Bagaimana untuk menghalang serangan suntikan SQL dalam PHP?
P粉939473759
2023-08-22 10:23:01
<p>Jika input pengguna dimasukkan ke dalam pertanyaan SQL tanpa pengubahsuaian, aplikasi menjadi terdedah kepada serangan suntikan SQL, seperti dalam contoh berikut: </p>
<pre class="lang-php prettyprint-override"><kod>$unsafe_variable = $_POST['user_input'];
mysql_query("INSERT INTO `table` (`column`) NILAI ('$unsafe_variable')");
</code></pre>
<p>Ini kerana pengguna boleh memasukkan sesuatu seperti <code>nilai'); DROP TABLE;--</code>
<pre class="brush:php;toolbar:false;">INSERT IN TO `table` (`column`) NILAI('value');
<p>Bagaimanakah saya boleh menghalang perkara ini daripada berlaku? </p>
Untuk menggunakan pertanyaan berparameter anda perlu menggunakan Mysqli atau PDO. Untuk menulis semula contoh anda menggunakan mysqli, kami memerlukan kod yang serupa dengan yang berikut.
Fungsi utama yang mungkin anda ingin baca ialah
mysqli::prepare
.Selain itu, seperti yang dicadangkan oleh orang lain, anda mungkin mendapati ia lebih berguna/lebih mudah untuk menggunakan lapisan abstraksi tahap yang lebih tinggi seperti PDO.
Sila ambil perhatian bahawa situasi yang anda nyatakan agak mudah, situasi yang lebih kompleks mungkin memerlukan pendekatan yang lebih canggih. Terutamanya:
mysql_real_escape_string
. Dalam kes ini, adalah lebih baik untuk menghantar input pengguna melalui senarai putih untuk memastikan bahawa hanya nilai "selamat" dibenarkan melaluinya.Tidak kira pangkalan data yang anda gunakan, cara yang betul untuk mengelakkan serangan suntikan SQL adalah dengan mengasingkan data daripada SQL supaya data kekal dalam bentuk data yang tidak pernah ditafsirkan sebagai arahan oleh penghurai SQL . Anda boleh membuat pernyataan SQL dengan bahagian data yang diformat dengan betul, tetapi jika anda tidak sepenuhnya memahami butirannya, anda harus sentiasa menggunakan pernyataan yang disediakan dan pertanyaan berparameter. Ini adalah pernyataan SQL yang dihantar secara berasingan daripada sebarang parameter dan dianalisis oleh pelayan pangkalan data. Dengan cara ini, penyerang tidak boleh menyuntik SQL berniat jahat.
Pada asasnya terdapat dua cara untuk mencapai ini:
Menggunakan PDO (berfungsi dengan mana-mana pemacu pangkalan data yang disokong):
Gunakan MySQLi (untuk MySQL):
Sejak PHP 8.2+, kami boleh menggunakan kaedah
execute_query()
untuk menyediakan, mengikat parameter dan melaksanakan pernyataan SQL:Sebelum PHP8.1:
Jika anda menyambung ke pangkalan data selain MySQL, anda boleh merujuk kepada pilihan khusus pemacu kedua (untuk PostgreSQL, contohnya, anda boleh menggunakan
pg_prepare()
和pg_execute()
). PDO ialah pilihan universal.Sediakan sambungan dengan betul
PDO
Sila ambil perhatian bahawa apabila menggunakan PDO untuk mengakses pangkalan data MySQL, kenyataan sebenar yang disediakan tidak digunakan secara lalai . Untuk menyelesaikan isu ini, anda perlu melumpuhkan simulasi kenyataan yang disediakan. Berikut ialah contoh mencipta sambungan menggunakan PDO:
Dalam contoh di atas, mod ralat tidak diperlukan sepenuhnya, tetapi disyorkan untuk menambahnya . Dengan cara ini PDO akan memberitahu anda tentang semua ralat MySQL dengan membuang
PDOException
.Apa mesti , walau bagaimanapun, ialah baris pertama
setAttribute()
, yang memberitahu PDO untuk melumpuhkan kenyataan simulasi yang disediakan dan menggunakan kenyataan sebenar yang disediakan. Ini memastikan bahawa pernyataan dan nilai tidak dihuraikan oleh PHP sebelum dihantar ke pelayan MySQL (jadi penyerang yang berpotensi tidak boleh menyuntik SQL yang berniat jahat).Walaupun anda boleh menetapkannya
charset
dalam pilihan pembina, adalah penting untuk ambil perhatian bahawa versi "lama" PHP (sebelum 5.3.6) mengabaikan parameter charset secara senyap dalam DSN.Mysqli
Untuk mysqli kita perlu mengikut rutin yang sama:
Penjelasan
Apabila anda lulus
prepare
的SQL语句由数据库服务器解析和编译。通过指定参数(在上面的示例中,可以是?
或命名参数,如:name
),您告诉数据库引擎您要在哪里进行过滤。然后,当您调用execute
, pernyataan yang disediakan akan digabungkan dengan nilai parameter yang ditentukan.Perkara penting di sini ialah nilai parameter digabungkan dengan pernyataan yang disusun, bukan dengan rentetan SQL. Suntikan SQL berfungsi dengan menipu skrip untuk memasukkan rentetan berniat jahat apabila mencipta SQL untuk dihantar ke pangkalan data. Oleh itu, dengan menghantar SQL sebenar secara berasingan daripada parameter, anda mengehadkan risiko hasil yang tidak dijangka.
Sebarang parameter yang dihantar menggunakan pernyataan yang disediakan akan dianggap sebagai rentetan (walaupun enjin pangkalan data mungkin melakukan beberapa pengoptimuman parameter, jadi parameter mungkin berakhir dengan nombor). Dalam contoh di atas, jika
$name
变量包含'Sarah'; DELETE FROM employees
,结果将仅是搜索字符串"'Sarah'; DELETE FROM employees"
anda tidak akan mendapat meja kosong.Faedah lain menggunakan pernyataan yang disediakan ialah jika pernyataan yang sama dilaksanakan beberapa kali dalam sesi yang sama, ia hanya akan dihuraikan dan disusun sekali, sekali gus meningkatkan sedikit kelajuan.
Oh, kerana anda bertanya cara mengendalikan sisipan, berikut adalah contoh (menggunakan PDO):
Adakah pernyataan yang disediakan sesuai untuk pertanyaan dinamik?
Walaupun anda masih boleh menggunakan pernyataan yang disediakan untuk parameter pertanyaan, struktur pertanyaan dinamik itu sendiri tidak boleh diparameterkan dan fungsi pertanyaan tertentu tidak boleh diparameterkan.
Untuk senario khusus ini, amalan terbaik ialah menggunakan penapis senarai putih untuk mengehadkan nilai yang mungkin.