Phase 1 sub-commit 4: Qt foundation types
All checks were successful
CI / Quality (push) Successful in 5s
All checks were successful
CI / Quality (push) Successful in 5s
BackendConnection (QML singleton via create() factory) reads BRIDGE_URL and BRIDGE_TOKEN from env, periodically probes <url>/healthz with a 2s transfer timeout, and exposes a Connecting/Online/Error state machine plus error/token properties to QML. Bundled-mode startup (spawning the embedded FrankenPHP child) is a Phase 4 deliverable; restart() is a no-op for now. tokenRotated signal is reserved for the per-session secret rotation described in PLAN.md §3. SingleInstance is C++-only — main() must call acquireOrForward() before the QML engine boots, so it's exposed via context property rather than QML_SINGLETON. QLocalServer-based lock with stale-socket detection, launch-arg forwarding via QDataStream, and the deadlock-avoiding race fallback specified in §3 *Edge cases*. CMakeLists.txt declares the PhpQml.Bridge static QML module with both sources and is dual-mode: stands alone for sanity builds, integrates via add_subdirectory from the skeleton's top-level CMake (Phase 1 sub-commit 6). Standalone build verified clean. Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
104
framework/qml/src/BackendConnection.cpp
Normal file
104
framework/qml/src/BackendConnection.cpp
Normal file
@@ -0,0 +1,104 @@
|
||||
#include "BackendConnection.h"
|
||||
|
||||
#include <QNetworkAccessManager>
|
||||
#include <QNetworkReply>
|
||||
#include <QNetworkRequest>
|
||||
#include <QQmlEngine>
|
||||
#include <QTimer>
|
||||
#include <QUrl>
|
||||
|
||||
namespace PhpQml::Bridge {
|
||||
|
||||
namespace {
|
||||
constexpr int kInitialProbeMs = 0;
|
||||
constexpr int kProbeIntervalMs = 5000;
|
||||
constexpr int kProbeTimeoutMs = 2000;
|
||||
} // namespace
|
||||
|
||||
BackendConnection::BackendConnection(QObject* parent)
|
||||
: QObject(parent)
|
||||
, m_nam(new QNetworkAccessManager(this))
|
||||
, m_retryTimer(new QTimer(this))
|
||||
{
|
||||
m_url = QString::fromUtf8(qgetenv("BRIDGE_URL"));
|
||||
m_token = QString::fromUtf8(qgetenv("BRIDGE_TOKEN"));
|
||||
|
||||
if (m_url.isEmpty()) {
|
||||
// Dev-mode fallback: matches the spike's hardcoded port and
|
||||
// documents the convention. See PLAN.md §11 *Open Questions*
|
||||
// (system FrankenPHP collision on :8080).
|
||||
m_url = QStringLiteral("http://127.0.0.1:8765");
|
||||
}
|
||||
|
||||
m_retryTimer->setSingleShot(false);
|
||||
m_retryTimer->setInterval(kProbeIntervalMs);
|
||||
connect(m_retryTimer, &QTimer::timeout, this, &BackendConnection::probe);
|
||||
|
||||
QTimer::singleShot(kInitialProbeMs, this, &BackendConnection::probe);
|
||||
m_retryTimer->start();
|
||||
}
|
||||
|
||||
BackendConnection::~BackendConnection() = default;
|
||||
|
||||
BackendConnection* BackendConnection::create(QQmlEngine* engine, QJSEngine*)
|
||||
{
|
||||
return new BackendConnection(engine);
|
||||
}
|
||||
|
||||
void BackendConnection::restart()
|
||||
{
|
||||
// No-op in dev mode (Phase 1). Phase 4 re-spawns the bundled child.
|
||||
probe();
|
||||
}
|
||||
|
||||
void BackendConnection::probe()
|
||||
{
|
||||
if (m_pendingReply) return;
|
||||
|
||||
QNetworkRequest req;
|
||||
req.setUrl(QUrl(m_url + QStringLiteral("/healthz")));
|
||||
req.setTransferTimeout(kProbeTimeoutMs);
|
||||
if (!m_token.isEmpty()) {
|
||||
req.setRawHeader("Authorization", "Bearer " + m_token.toUtf8());
|
||||
}
|
||||
|
||||
m_pendingReply = m_nam->get(req);
|
||||
connect(m_pendingReply, &QNetworkReply::finished, this, &BackendConnection::onProbeFinished);
|
||||
}
|
||||
|
||||
void BackendConnection::onProbeFinished()
|
||||
{
|
||||
QNetworkReply* reply = m_pendingReply;
|
||||
m_pendingReply = nullptr;
|
||||
if (!reply) return;
|
||||
reply->deleteLater();
|
||||
|
||||
if (reply->error() == QNetworkReply::NoError) {
|
||||
const int status = reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt();
|
||||
if (status == 200) {
|
||||
setError(QString());
|
||||
setState(ConnectionState::Online);
|
||||
return;
|
||||
}
|
||||
setError(QStringLiteral("/healthz returned HTTP %1").arg(status));
|
||||
} else {
|
||||
setError(reply->errorString());
|
||||
}
|
||||
setState(ConnectionState::Error);
|
||||
}
|
||||
|
||||
void BackendConnection::setState(ConnectionState s)
|
||||
{
|
||||
if (m_state == s) return;
|
||||
m_state = s;
|
||||
emit connectionStateChanged();
|
||||
}
|
||||
|
||||
void BackendConnection::setError(const QString& msg)
|
||||
{
|
||||
if (m_error == msg) return;
|
||||
m_error = msg;
|
||||
emit errorChanged();
|
||||
}
|
||||
|
||||
} // namespace PhpQml::Bridge
|
||||
Reference in New Issue
Block a user