


Detaillierte Erläuterung von Sitzungssperre, Parallelität und Abdeckung in PHP
Dieser Artikel führt Sie hauptsächlich in die Sperrung und Parallelität von PHP-Sitzungen ein. Zu den damit verbundenen Phänomenen gehören das Blockieren von Anforderungen, der Verlust von Sitzungsdaten und das Nicht-Lesen von Sitzungsdaten .
Ich kann mich nicht anmelden
Eines Tages wollte ich mich bei einem unserer Backend-Systeme anmelden, um einen Fehler zu beheben, und trat ein In diesem Fall kann ich mich nicht anmelden. Nach vielen Experimenten habe ich festgestellt, dass es zwei Hauptfehlermeldungen gibt:
CSRF-Verifizierung fehlgeschlagen
Der Bestätigungscode ist falsch [Ich schwöre bei den Codegöttern, dass ich den Bestätigungscode, den ich gesehen habe, mit halber Breite eingegeben habe, und in der gleichen Reihenfolge, ohne zusätzliche Zeichen]
Unser SystemUnser System wurde auf Basis von Phalcon 2.0 entwickelt .8. Wie Sie sehen können, haben wir im Formularfeld Domäne hinzugefügt, um CSRF-Angriffe zu verhindern. Captcha ist ebenfalls aktiviert.
<input type="hidden" name="{{ security.getTokenKey() }}" value="{{ security.getToken() }}"/> <img src="/login/getCaptcha" id="img-captcha"/>
Ich habe zuerst diese beiden Komponenten überprüft und festgestellt, dass beide Daten in der Sitzung speichern:
# phalcon/security.zep # Security::getToken() let session = <SessionInterface> dependencyInjector->getShared("session"); session->set(this->_tokenValueSessionID, token); $this->session->set('admin_get_captcha_action', $captcha);
Dann habe ich unsere Sitzungsimplementierung überprüft um Daten in Redis zu speichern.
Schauen und suchenWas ist das Problem, das mich daran hindert, mich einzuloggen? Da es ein Problem mit der Datenüberprüfung gibt, beginnen wir mit den Daten. Ich habe mich in unserer Testumgebung beim Redis-Computer angemeldet, den Redis-Cli-Monitor ausgeführt und dann den Anmeldevorgang durchlaufen und die Ausgabe wie folgt gefunden (Bedeutung):
GET sessionId
GET sessionId
SETEX sessionId 3600 csrf=xxxx 🎜>
- Wir können sehen:
1. Hier gibt es zwei Anfragen: Eine dient zum Laden des Formulars und die andere zum Generieren des Bestätigungscodes. 2. Es liegt eine „Parallelitätssituation“ vor. Diese beiden Anforderungen sollten den Bestätigungscode anfordern, nachdem das Formular geladen und gerendert wurde. Das heißt, die Sitzungssequenz sollte get->set->get->set sein . Es scheint, warum gibt es gleichzeitige Anfragen? 3. Letzteres SETEX hat keinen CSRF-Inhalt, was bedeutet, dass es die vorherigen Daten überschreibt. Die ganze Welt ist nicht gut, aber ich verstehe ein wenig, wo das Problem liegt. Was ist das Problem? Es ist eine lange Geschichte, angefangen beim Zugriff auf PHP-Sitzungsdaten.
Zugriff auf PHP-Sitzungsdaten
Die Sitzungsdaten werden in eine Zeichenfolge codiert und im Speicher [Datei, Datenbank, Redis, Memcache usw.] gespeichert Wir verwenden Sitzungen. Wann gehen wir zum Speicher, um Daten abzurufen? Wann werden die Daten in den Speicher geschrieben?
Die Antwort auf diese Frage unterscheidet sich möglicherweise von dem, was einige Freunde denken. Bei einer Anfrage liest PHP den Speicher nur einmal, wenn session_start, und schreibt dann nur einmal in den Speicher, wenn die Anfrage endet . oder wenn session_write_close aufgerufen wird, werden die Daten zurück in den Speicher geleert und die Sitzung geschlossen.
1、如果一个会话,同时出现两个读写session请求,没有保证获取1-写入1-获取2-写入2,同时没有cas版本管理机制的情况下,这些并发请求就会彼此读取不到对方的写入,最后写入的会把前面请求写入的session覆盖掉。
2、如果请求是串行的,像登录页面的表单和验证码,也有可能前面的请求已经输出内容了,但是session还没写入,后面的请求就已经发起了。
锁与不锁
解决这种资源的并发一般会通过锁或版本管理来处理。但是版本管理我看不到好的方法。就聊聊锁吧。
其实锁是不大适合,有弊端的。
php的session,默认是用文件存储的,在打开session的时候,会对文件加独占锁,这样,其它请求就无法获取锁了,只能等待直到前面的锁解了。
这样保证了 读取-写入,读取-写入的顺序。
其它存储器,例如mysql,可以借助select for update进行行锁。redis可以通过一个自增键,返回1的获取到锁等来实现。
这个实现的话,对数据流来说很理想,但是,对于目前这种页面大量应用ajax的情况,所有请求排队处理,将大大加大页面展现的耗时,甚至出现请求超时等不可用故障。
没有解决的解决不建议过多使用session,其一次读取一次写入的机制所引发的问题,会造成坑的存在。
在模版渲染前,或请求输出前调用session_write_close
# 立刻回写session,避免session覆盖 $eventManager = $this->view->getEventsManager(); if (!$eventManager) { $eventManager = new Manager(); $this->view->setEventsManager($eventManager); } $eventManager->attach("view:afterRender",function(){ session_write_close(); }); return $this->view; if($login) { # 立刻回写session,避免session读取不到 $eventManager = $this->dispatcher->getEventsManager(); if (!$eventManager) { $eventManager = new Manager(); $this->dispatcher->setEventsManager($eventManager); } $eventManager->attach('dispatch:afterDispatchLoop',function(){ session_write_close(); }); return $this->response->setHeader('Location', '/'); }
相关推荐:
Das obige ist der detaillierte Inhalt vonDetaillierte Erläuterung von Sitzungssperre, Parallelität und Abdeckung in PHP. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Heiße KI -Werkzeuge

Undresser.AI Undress
KI-gestützte App zum Erstellen realistischer Aktfotos

AI Clothes Remover
Online-KI-Tool zum Entfernen von Kleidung aus Fotos.

Undress AI Tool
Ausziehbilder kostenlos

Clothoff.io
KI-Kleiderentferner

AI Hentai Generator
Erstellen Sie kostenlos Ai Hentai.

Heißer Artikel

Heiße Werkzeuge

Notepad++7.3.1
Einfach zu bedienender und kostenloser Code-Editor

SublimeText3 chinesische Version
Chinesische Version, sehr einfach zu bedienen

Senden Sie Studio 13.0.1
Leistungsstarke integrierte PHP-Entwicklungsumgebung

Dreamweaver CS6
Visuelle Webentwicklungstools

SublimeText3 Mac-Version
Codebearbeitungssoftware auf Gottesniveau (SublimeText3)

Heiße Themen



PHP 8.4 bringt mehrere neue Funktionen, Sicherheitsverbesserungen und Leistungsverbesserungen mit einer beträchtlichen Menge an veralteten und entfernten Funktionen. In dieser Anleitung wird erklärt, wie Sie PHP 8.4 installieren oder auf PHP 8.4 auf Ubuntu, Debian oder deren Derivaten aktualisieren. Obwohl es möglich ist, PHP aus dem Quellcode zu kompilieren, ist die Installation aus einem APT-Repository wie unten erläutert oft schneller und sicherer, da diese Repositorys in Zukunft die neuesten Fehlerbehebungen und Sicherheitsupdates bereitstellen.

Um in cakephp4 mit Datum und Uhrzeit zu arbeiten, verwenden wir die verfügbare FrozenTime-Klasse.

CakePHP ist ein Open-Source-Framework für PHP. Es soll die Entwicklung, Bereitstellung und Wartung von Anwendungen erheblich vereinfachen. CakePHP basiert auf einer MVC-ähnlichen Architektur, die sowohl leistungsstark als auch leicht zu verstehen ist. Modelle, Ansichten und Controller gu

Um am Datei-Upload zu arbeiten, verwenden wir den Formular-Helfer. Hier ist ein Beispiel für den Datei-Upload.

Der Validator kann durch Hinzufügen der folgenden zwei Zeilen im Controller erstellt werden.

Die Anmeldung bei CakePHP ist eine sehr einfache Aufgabe. Sie müssen nur eine Funktion verwenden. Sie können Fehler, Ausnahmen, Benutzeraktivitäten und von Benutzern durchgeführte Aktionen für jeden Hintergrundprozess wie Cronjob protokollieren. Das Protokollieren von Daten in CakePHP ist einfach. Die Funktion log() wird bereitgestellt

Visual Studio Code, auch bekannt als VS Code, ist ein kostenloser Quellcode-Editor – oder eine integrierte Entwicklungsumgebung (IDE) –, die für alle gängigen Betriebssysteme verfügbar ist. Mit einer großen Sammlung von Erweiterungen für viele Programmiersprachen kann VS Code c

CakePHP ist ein Open-Source-MVC-Framework. Es erleichtert die Entwicklung, Bereitstellung und Wartung von Anwendungen erheblich. CakePHP verfügt über eine Reihe von Bibliotheken, um die Überlastung der häufigsten Aufgaben zu reduzieren.
