2026-05-02 01:18:43 +02:00
|
|
|
#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();
|
|
|
|
|
|
Phase 2 sub-commit 3: full Update Semantics + ReactiveListModel + AppShell
BackendConnection's ConnectionState enum is now Connecting / Online /
Reconnecting / Offline (PLAN.md §5). The probe loop tracks the first
failure since the last Online and transitions to Reconnecting on any
failed probe, then to Offline once the configurable threshold (30 s
default) is exceeded. The Error state is gone; Reconnecting + the
exposed `error` string subsume its UI role.
ReactiveListModel is the headline QML type:
- QAbstractListModel that GETs `baseUrl + source` for an initial JSON
array and then keeps in sync via an internal MercureClient subscribed
to `topic`.
- Role names are derived dynamically from the first row's keys plus an
internal `pending` boolean role used by optimistic mutations.
- Diff application: upsert (insert-or-update), delete, replace; gap
detection via the envelope `version` field with auto re-fetch.
- `invoke(method, path, body, optimistic)` is the optimistic command
primitive. Generates an Idempotency-Key, applies the local diff,
POST/PATCH/DELETEs with that key, and resolves on the matching
Mercure echo (correlation-key matched in ModelPublisher's envelope).
Rolls back and emits commandFailed on 4xx/5xx, commandTimedOut after
10 s without an echo. Phase 4 packaging will surface configuration
for the timeout.
AppShell.qml is the optional convenience root:
- Reads BackendConnection.connectionState.
- Reconnecting → top banner.
- Offline → modal overlay with the error string and a Retry button
(calls BackendConnection.restart()).
- Wraps user content via `default property alias content`.
Apps that want full chrome control can skip AppShell entirely; the
skeleton's Main.qml keeps its own status display for demonstration
and is unaffected.
ReactiveObject (single-entity twin of ReactiveListModel) is intentionally
deferred — same envelope handling, smaller surface; will land in Phase 2
follow-up or Phase 3 alongside the todo example. Cursor pagination is
similarly deferred (the Phase 2 done criterion uses small lists).
Smoke tested: /healthz + /api/ping round-trip cleanly, zero Mercure 401s,
clean shutdown. composer quality stays green (16 tests, 45 assertions).
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-02 02:40:12 +02:00
|
|
|
bool ok = false;
|
2026-05-02 01:18:43 +02:00
|
|
|
if (reply->error() == QNetworkReply::NoError) {
|
|
|
|
|
const int status = reply->attribute(QNetworkRequest::HttpStatusCodeAttribute).toInt();
|
|
|
|
|
if (status == 200) {
|
Phase 2 sub-commit 3: full Update Semantics + ReactiveListModel + AppShell
BackendConnection's ConnectionState enum is now Connecting / Online /
Reconnecting / Offline (PLAN.md §5). The probe loop tracks the first
failure since the last Online and transitions to Reconnecting on any
failed probe, then to Offline once the configurable threshold (30 s
default) is exceeded. The Error state is gone; Reconnecting + the
exposed `error` string subsume its UI role.
ReactiveListModel is the headline QML type:
- QAbstractListModel that GETs `baseUrl + source` for an initial JSON
array and then keeps in sync via an internal MercureClient subscribed
to `topic`.
- Role names are derived dynamically from the first row's keys plus an
internal `pending` boolean role used by optimistic mutations.
- Diff application: upsert (insert-or-update), delete, replace; gap
detection via the envelope `version` field with auto re-fetch.
- `invoke(method, path, body, optimistic)` is the optimistic command
primitive. Generates an Idempotency-Key, applies the local diff,
POST/PATCH/DELETEs with that key, and resolves on the matching
Mercure echo (correlation-key matched in ModelPublisher's envelope).
Rolls back and emits commandFailed on 4xx/5xx, commandTimedOut after
10 s without an echo. Phase 4 packaging will surface configuration
for the timeout.
AppShell.qml is the optional convenience root:
- Reads BackendConnection.connectionState.
- Reconnecting → top banner.
- Offline → modal overlay with the error string and a Retry button
(calls BackendConnection.restart()).
- Wraps user content via `default property alias content`.
Apps that want full chrome control can skip AppShell entirely; the
skeleton's Main.qml keeps its own status display for demonstration
and is unaffected.
ReactiveObject (single-entity twin of ReactiveListModel) is intentionally
deferred — same envelope handling, smaller surface; will land in Phase 2
follow-up or Phase 3 alongside the todo example. Cursor pagination is
similarly deferred (the Phase 2 done criterion uses small lists).
Smoke tested: /healthz + /api/ping round-trip cleanly, zero Mercure 401s,
clean shutdown. composer quality stays green (16 tests, 45 assertions).
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-02 02:40:12 +02:00
|
|
|
ok = true;
|
|
|
|
|
} else {
|
|
|
|
|
setError(QStringLiteral("/healthz returned HTTP %1").arg(status));
|
2026-05-02 01:18:43 +02:00
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
setError(reply->errorString());
|
|
|
|
|
}
|
Phase 2 sub-commit 3: full Update Semantics + ReactiveListModel + AppShell
BackendConnection's ConnectionState enum is now Connecting / Online /
Reconnecting / Offline (PLAN.md §5). The probe loop tracks the first
failure since the last Online and transitions to Reconnecting on any
failed probe, then to Offline once the configurable threshold (30 s
default) is exceeded. The Error state is gone; Reconnecting + the
exposed `error` string subsume its UI role.
ReactiveListModel is the headline QML type:
- QAbstractListModel that GETs `baseUrl + source` for an initial JSON
array and then keeps in sync via an internal MercureClient subscribed
to `topic`.
- Role names are derived dynamically from the first row's keys plus an
internal `pending` boolean role used by optimistic mutations.
- Diff application: upsert (insert-or-update), delete, replace; gap
detection via the envelope `version` field with auto re-fetch.
- `invoke(method, path, body, optimistic)` is the optimistic command
primitive. Generates an Idempotency-Key, applies the local diff,
POST/PATCH/DELETEs with that key, and resolves on the matching
Mercure echo (correlation-key matched in ModelPublisher's envelope).
Rolls back and emits commandFailed on 4xx/5xx, commandTimedOut after
10 s without an echo. Phase 4 packaging will surface configuration
for the timeout.
AppShell.qml is the optional convenience root:
- Reads BackendConnection.connectionState.
- Reconnecting → top banner.
- Offline → modal overlay with the error string and a Retry button
(calls BackendConnection.restart()).
- Wraps user content via `default property alias content`.
Apps that want full chrome control can skip AppShell entirely; the
skeleton's Main.qml keeps its own status display for demonstration
and is unaffected.
ReactiveObject (single-entity twin of ReactiveListModel) is intentionally
deferred — same envelope handling, smaller surface; will land in Phase 2
follow-up or Phase 3 alongside the todo example. Cursor pagination is
similarly deferred (the Phase 2 done criterion uses small lists).
Smoke tested: /healthz + /api/ping round-trip cleanly, zero Mercure 401s,
clean shutdown. composer quality stays green (16 tests, 45 assertions).
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-02 02:40:12 +02:00
|
|
|
|
|
|
|
|
if (ok) {
|
|
|
|
|
setError(QString());
|
|
|
|
|
m_firstFailureSinceOnline.invalidate();
|
|
|
|
|
setState(ConnectionState::Online);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Probe failed. Decide between Reconnecting and Offline based on how
|
|
|
|
|
// long we've been failing since the last Online. (PLAN.md §5.)
|
|
|
|
|
if (!m_firstFailureSinceOnline.isValid()) {
|
|
|
|
|
m_firstFailureSinceOnline.start();
|
|
|
|
|
}
|
|
|
|
|
if (m_firstFailureSinceOnline.hasExpired(m_offlineThresholdMs)) {
|
|
|
|
|
setState(ConnectionState::Offline);
|
|
|
|
|
} else {
|
|
|
|
|
setState(ConnectionState::Reconnecting);
|
|
|
|
|
}
|
2026-05-02 01:18:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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
|