Commit 06df2dac authored by Allan Sandfeld Jensen's avatar Allan Sandfeld Jensen Committed by Allan Sandfeld Jensen
Browse files

Make the websocket handshake statefull


The websocket handshake could not handle being split over multiple
TCP packets since it was entirely handled in on function. This patch
splits it into a socket state making it possible to process over
multiple incoming packaets.

Change-Id: I3c45892ee6f1bb67062d561e9fbd2d7296f1208e
Task-number: QTBUG-40878
Reviewed-by: default avatarJocelyn Turcotte <jocelyn.turcotte@digia.com>
Showing with 82 additions and 39 deletions
...@@ -137,7 +137,8 @@ QWebSocketPrivate::QWebSocketPrivate(QTcpSocket *pTcpSocket, QWebSocketProtocol: ...@@ -137,7 +137,8 @@ QWebSocketPrivate::QWebSocketPrivate(QTcpSocket *pTcpSocket, QWebSocketProtocol:
m_dataProcessor(), m_dataProcessor(),
m_configuration(), m_configuration(),
m_pMaskGenerator(&m_defaultMaskGenerator), m_pMaskGenerator(&m_defaultMaskGenerator),
m_defaultMaskGenerator() m_defaultMaskGenerator(),
m_handshakeState(NothingDoneState)
{ {
} }
...@@ -889,46 +890,66 @@ void QWebSocketPrivate::processHandshake(QTcpSocket *pSocket) ...@@ -889,46 +890,66 @@ void QWebSocketPrivate::processHandshake(QTcpSocket *pSocket)
Q_Q(QWebSocket); Q_Q(QWebSocket);
if (Q_UNLIKELY(!pSocket)) if (Q_UNLIKELY(!pSocket))
return; return;
// Reset handshake on a new connection.
if (m_handshakeState == AllDoneState)
m_handshakeState = NothingDoneState;
bool ok = false;
QString errorDescription; QString errorDescription;
const QByteArray statusLine = pSocket->readLine(); switch (m_handshakeState) {
int httpMajorVersion, httpMinorVersion; case NothingDoneState:
int httpStatusCode; m_headers.clear();
QString httpStatusMessage; m_handshakeState = ReadingStatusState;
if (Q_UNLIKELY(!parseStatusLine(statusLine, &httpMajorVersion, &httpMinorVersion, // no break
&httpStatusCode, &httpStatusMessage))) { case ReadingStatusState:
errorDescription = QWebSocket::tr("Invalid statusline in response: %1.").arg(QString::fromLatin1(statusLine)); if (!pSocket->canReadLine())
} else { return;
QString headerLine = readLine(pSocket); m_statusLine = pSocket->readLine();
QMap<QString, QString> headers; if (Q_UNLIKELY(!parseStatusLine(m_statusLine, &m_httpMajorVersion, &m_httpMinorVersion, &m_httpStatusCode, &m_httpStatusMessage))) {
while (!headerLine.isEmpty()) { errorDescription = QWebSocket::tr("Invalid statusline in response: %1.").arg(QString::fromLatin1(m_statusLine));
break;
}
m_handshakeState = ReadingHeaderState;
// no break
case ReadingHeaderState:
while (pSocket->canReadLine()) {
QString headerLine = readLine(pSocket);
const QStringList headerField = headerLine.split(QStringLiteral(": "), const QStringList headerField = headerLine.split(QStringLiteral(": "),
QString::SkipEmptyParts); QString::SkipEmptyParts);
if (headerField.size() == 2) { if (headerField.size() == 2) {
headers.insertMulti(headerField[0].toLower(), headerField[1]); m_headers.insertMulti(headerField[0].toLower(), headerField[1]);
}
if (headerField.isEmpty()) {
m_handshakeState = ParsingHeaderState;
break;
} }
headerLine = readLine(pSocket);
} }
const QString acceptKey = headers.value(QStringLiteral("sec-websocket-accept"), if (m_handshakeState != ParsingHeaderState) {
QString()); if (pSocket->atEnd()) {
const QString upgrade = headers.value(QStringLiteral("upgrade"), QString()); errorDescription = QWebSocket::tr("QWebSocketPrivate::processHandshake: Connection closed while reading header.");
const QString connection = headers.value(QStringLiteral("connection"), QString()); break;
}
return;
}
// no break
case ParsingHeaderState: {
const QString acceptKey = m_headers.value(QStringLiteral("sec-websocket-accept"), QString());
const QString upgrade = m_headers.value(QStringLiteral("upgrade"), QString());
const QString connection = m_headers.value(QStringLiteral("connection"), QString());
// unused for the moment // unused for the moment
// const QString extensions = headers.value(QStringLiteral("sec-websocket-extensions"), // const QString extensions = m_headers.value(QStringLiteral("sec-websocket-extensions"),
// QString()); // QString());
// const QString protocol = headers.value(QStringLiteral("sec-websocket-protocol"), // const QString protocol = m_headers.value(QStringLiteral("sec-websocket-protocol"),
// QString()); // QString());
const QString version = headers.value(QStringLiteral("sec-websocket-version"), const QString version = m_headers.value(QStringLiteral("sec-websocket-version"), QString());
QString());
if (Q_LIKELY(httpStatusCode == 101)) { bool ok = false;
if (Q_LIKELY(m_httpStatusCode == 101)) {
//HTTP/x.y 101 Switching Protocols //HTTP/x.y 101 Switching Protocols
//TODO: do not check the httpStatusText right now //TODO: do not check the httpStatusText right now
ok = !(acceptKey.isEmpty() || ok = !(acceptKey.isEmpty() ||
(httpMajorVersion < 1 || httpMinorVersion < 1) || (m_httpMajorVersion < 1 || m_httpMinorVersion < 1) ||
(upgrade.toLower() != QStringLiteral("websocket")) || (upgrade.toLower() != QStringLiteral("websocket")) ||
(connection.toLower() != QStringLiteral("upgrade"))); (connection.toLower() != QStringLiteral("upgrade")));
if (ok) { if (ok) {
...@@ -941,9 +962,9 @@ void QWebSocketPrivate::processHandshake(QTcpSocket *pSocket) ...@@ -941,9 +962,9 @@ void QWebSocketPrivate::processHandshake(QTcpSocket *pSocket)
} else { } else {
errorDescription = errorDescription =
QWebSocket::tr("QWebSocketPrivate::processHandshake: Invalid statusline in response: %1.") QWebSocket::tr("QWebSocketPrivate::processHandshake: Invalid statusline in response: %1.")
.arg(QString::fromLatin1(statusLine)); .arg(QString::fromLatin1(m_statusLine));
} }
} else if (httpStatusCode == 400) { } else if (m_httpStatusCode == 400) {
//HTTP/1.1 400 Bad Request //HTTP/1.1 400 Bad Request
if (!version.isEmpty()) { if (!version.isEmpty()) {
const QStringList versions = version.split(QStringLiteral(", "), const QStringList versions = version.split(QStringLiteral(", "),
...@@ -954,29 +975,38 @@ void QWebSocketPrivate::processHandshake(QTcpSocket *pSocket) ...@@ -954,29 +975,38 @@ void QWebSocketPrivate::processHandshake(QTcpSocket *pSocket)
errorDescription = errorDescription =
QWebSocket::tr("Handshake: Server requests a version that we don't support: %1.") QWebSocket::tr("Handshake: Server requests a version that we don't support: %1.")
.arg(versions.join(QStringLiteral(", "))); .arg(versions.join(QStringLiteral(", ")));
ok = false;
} else { } else {
//we tried v13, but something different went wrong //we tried v13, but something different went wrong
errorDescription = errorDescription =
QWebSocket::tr("QWebSocketPrivate::processHandshake: Unknown error condition encountered. Aborting connection."); QWebSocket::tr("QWebSocketPrivate::processHandshake: Unknown error condition encountered. Aborting connection.");
ok = false;
} }
} else {
errorDescription =
QWebSocket::tr("QWebSocketPrivate::processHandshake: Unknown error condition encountered. Aborting connection.");
} }
} else { } else {
errorDescription = errorDescription =
QWebSocket::tr("QWebSocketPrivate::processHandshake: Unhandled http status code: %1 (%2).") QWebSocket::tr("QWebSocketPrivate::processHandshake: Unhandled http status code: %1 (%2).")
.arg(httpStatusCode).arg(httpStatusMessage); .arg(m_httpStatusCode).arg(m_httpStatusMessage);
ok = false;
} }
if (ok)
m_handshakeState = AllDoneState;
break;
}
case AllDoneState:
Q_UNREACHABLE();
break;
}
if (!ok) { if (m_handshakeState == AllDoneState) {
setErrorString(errorDescription); // handshake succeeded
Q_EMIT q->error(QAbstractSocket::ConnectionRefusedError); setSocketState(QAbstractSocket::ConnectedState);
} else { Q_EMIT q->connected();
//handshake succeeded } else {
setSocketState(QAbstractSocket::ConnectedState); // handshake failed
Q_EMIT q->connected(); m_handshakeState = AllDoneState;
} setErrorString(errorDescription);
Q_EMIT q->error(QAbstractSocket::ConnectionRefusedError);
} }
} }
......
...@@ -224,6 +224,19 @@ private: ...@@ -224,6 +224,19 @@ private:
QMaskGenerator *m_pMaskGenerator; QMaskGenerator *m_pMaskGenerator;
QDefaultMaskGenerator m_defaultMaskGenerator; QDefaultMaskGenerator m_defaultMaskGenerator;
enum HandshakeState {
NothingDoneState,
ReadingStatusState,
ReadingHeaderState,
ParsingHeaderState,
AllDoneState
} m_handshakeState;
QByteArray m_statusLine;
int m_httpStatusCode;
int m_httpMajorVersion, m_httpMinorVersion;
QString m_httpStatusMessage;
QMap<QString, QString> m_headers;
friend class QWebSocketServerPrivate; friend class QWebSocketServerPrivate;
}; };
......
Supports Markdown
0% or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment