2026-05-02 01:18:43 +02:00
|
|
|
#include "SingleInstance.h"
|
|
|
|
|
|
|
|
|
|
#include <QDataStream>
|
|
|
|
|
#include <QLocalSocket>
|
|
|
|
|
#include <QThread>
|
|
|
|
|
|
|
|
|
|
namespace PhpQml::Bridge {
|
|
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
constexpr int kForwardConnectTimeoutMs = 200;
|
|
|
|
|
constexpr int kForwardWriteTimeoutMs = 1500;
|
|
|
|
|
constexpr int kBindRetries = 3;
|
|
|
|
|
constexpr int kBindRetryDelayMs = 50;
|
|
|
|
|
} // namespace
|
|
|
|
|
|
|
|
|
|
SingleInstance::SingleInstance(const QString& applicationId, QObject* parent)
|
|
|
|
|
: QObject(parent)
|
|
|
|
|
, m_appId(applicationId)
|
|
|
|
|
{
|
|
|
|
|
connect(&m_server, &QLocalServer::newConnection, this, &SingleInstance::onNewConnection);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QString SingleInstance::endpointName() const
|
|
|
|
|
{
|
|
|
|
|
// QLocalServer maps the name to a per-user path on Unix and a named
|
|
|
|
|
// pipe on Windows. The appId keeps multiple bridge-using apps apart.
|
|
|
|
|
return QStringLiteral("php-qml-bridge.%1").arg(m_appId);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool SingleInstance::acquireOrForward(const QStringList& launchArgs)
|
|
|
|
|
{
|
|
|
|
|
const QString name = endpointName();
|
|
|
|
|
|
Phase 1 sub-commit 6: skeleton wiring — make dev runs end-to-end
Symfony app under framework/skeleton/symfony/: minimal bin/console,
public/index.php, MicroKernel-based src/Kernel.php, services.yaml,
framework/security/mercure config, and a demo App\Controller\PingController
that GETs /api/ping (returning JSON pong) and republishes the same
payload to the Mercure topic app://ping. composer.json uses a path
repository to symlink the bundle from ../../php so local edits are
picked up live.
QML app under framework/skeleton/qml/: top-level CMake that
add_subdirectory's framework/qml, a main.cpp that creates the Qt
process, runs SingleInstance.acquireOrForward before any QML loads,
exposes SingleInstance via context property, and loadFromModule's
Skeleton.Main. Main.qml uses BackendConnection / RestClient /
MercureClient from PhpQml.Bridge and renders status dots, a Ping
button, and an event log.
Caddyfile binds 127.0.0.1:8765, enables in-memory Mercure with a
256-bit dev JWT (matches symfony/.env, lcobucci/jwt requires this).
Makefile wraps build / dev / doctor / clean; scripts/dev.sh starts
FrankenPHP --watch and the Qt host together with explicit PID-based
teardown (process-group `kill 0` proved unreliable when frankenphp's
watch fork reparented).
Bug fixes uncovered in this sub-commit:
- SingleInstance.acquireOrForward: probe-first, then removeServer +
retry-listen. The original loop-with-removeServer-after-failed-bind
silently exited on stale sockets from prior runs.
- Main.qml: MercureClient does NOT inherit BackendConnection.token —
Mercure subscribes anonymously in dev (Caddyfile), and forwarding
the bridge bearer made it 401-loop.
- /api/ping was 500ing because the dev MERCURE_JWT_SECRET was 144 bits;
bumped to 64-char (>=256 bit) to satisfy lcobucci/jwt.
- Linked the framework lib (php_qml_bridge) explicitly in addition to
the QML plugin so SingleInstance.h resolves.
- Auto-generated config/reference.php gitignored.
Smoke verified offscreen: /healthz 200, /api/ping 200, 1 publish, 1
subscriber, zero 401s, clean shutdown with no zombies.
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-02 01:50:16 +02:00
|
|
|
// Probe first: if a live instance answers, forward and exit.
|
|
|
|
|
// This avoids a race where eagerly calling removeServer() would break
|
|
|
|
|
// a running peer.
|
|
|
|
|
{
|
2026-05-02 01:18:43 +02:00
|
|
|
QLocalSocket probe;
|
|
|
|
|
probe.connectToServer(name);
|
|
|
|
|
if (probe.waitForConnected(kForwardConnectTimeoutMs)) {
|
|
|
|
|
QDataStream out(&probe);
|
|
|
|
|
out.setVersion(QDataStream::Qt_6_5);
|
|
|
|
|
out << launchArgs;
|
|
|
|
|
probe.flush();
|
|
|
|
|
probe.waitForBytesWritten(kForwardWriteTimeoutMs);
|
|
|
|
|
probe.disconnectFromServer();
|
|
|
|
|
return false;
|
|
|
|
|
}
|
Phase 1 sub-commit 6: skeleton wiring — make dev runs end-to-end
Symfony app under framework/skeleton/symfony/: minimal bin/console,
public/index.php, MicroKernel-based src/Kernel.php, services.yaml,
framework/security/mercure config, and a demo App\Controller\PingController
that GETs /api/ping (returning JSON pong) and republishes the same
payload to the Mercure topic app://ping. composer.json uses a path
repository to symlink the bundle from ../../php so local edits are
picked up live.
QML app under framework/skeleton/qml/: top-level CMake that
add_subdirectory's framework/qml, a main.cpp that creates the Qt
process, runs SingleInstance.acquireOrForward before any QML loads,
exposes SingleInstance via context property, and loadFromModule's
Skeleton.Main. Main.qml uses BackendConnection / RestClient /
MercureClient from PhpQml.Bridge and renders status dots, a Ping
button, and an event log.
Caddyfile binds 127.0.0.1:8765, enables in-memory Mercure with a
256-bit dev JWT (matches symfony/.env, lcobucci/jwt requires this).
Makefile wraps build / dev / doctor / clean; scripts/dev.sh starts
FrankenPHP --watch and the Qt host together with explicit PID-based
teardown (process-group `kill 0` proved unreliable when frankenphp's
watch fork reparented).
Bug fixes uncovered in this sub-commit:
- SingleInstance.acquireOrForward: probe-first, then removeServer +
retry-listen. The original loop-with-removeServer-after-failed-bind
silently exited on stale sockets from prior runs.
- Main.qml: MercureClient does NOT inherit BackendConnection.token —
Mercure subscribes anonymously in dev (Caddyfile), and forwarding
the bridge bearer made it 401-loop.
- /api/ping was 500ing because the dev MERCURE_JWT_SECRET was 144 bits;
bumped to 64-char (>=256 bit) to satisfy lcobucci/jwt.
- Linked the framework lib (php_qml_bridge) explicitly in addition to
the QML plugin so SingleInstance.h resolves.
- Auto-generated config/reference.php gitignored.
Smoke verified offscreen: /healthz 200, /api/ping 200, 1 publish, 1
subscriber, zero 401s, clean shutdown with no zombies.
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-02 01:50:16 +02:00
|
|
|
}
|
2026-05-02 01:18:43 +02:00
|
|
|
|
Phase 1 sub-commit 6: skeleton wiring — make dev runs end-to-end
Symfony app under framework/skeleton/symfony/: minimal bin/console,
public/index.php, MicroKernel-based src/Kernel.php, services.yaml,
framework/security/mercure config, and a demo App\Controller\PingController
that GETs /api/ping (returning JSON pong) and republishes the same
payload to the Mercure topic app://ping. composer.json uses a path
repository to symlink the bundle from ../../php so local edits are
picked up live.
QML app under framework/skeleton/qml/: top-level CMake that
add_subdirectory's framework/qml, a main.cpp that creates the Qt
process, runs SingleInstance.acquireOrForward before any QML loads,
exposes SingleInstance via context property, and loadFromModule's
Skeleton.Main. Main.qml uses BackendConnection / RestClient /
MercureClient from PhpQml.Bridge and renders status dots, a Ping
button, and an event log.
Caddyfile binds 127.0.0.1:8765, enables in-memory Mercure with a
256-bit dev JWT (matches symfony/.env, lcobucci/jwt requires this).
Makefile wraps build / dev / doctor / clean; scripts/dev.sh starts
FrankenPHP --watch and the Qt host together with explicit PID-based
teardown (process-group `kill 0` proved unreliable when frankenphp's
watch fork reparented).
Bug fixes uncovered in this sub-commit:
- SingleInstance.acquireOrForward: probe-first, then removeServer +
retry-listen. The original loop-with-removeServer-after-failed-bind
silently exited on stale sockets from prior runs.
- Main.qml: MercureClient does NOT inherit BackendConnection.token —
Mercure subscribes anonymously in dev (Caddyfile), and forwarding
the bridge bearer made it 401-loop.
- /api/ping was 500ing because the dev MERCURE_JWT_SECRET was 144 bits;
bumped to 64-char (>=256 bit) to satisfy lcobucci/jwt.
- Linked the framework lib (php_qml_bridge) explicitly in addition to
the QML plugin so SingleInstance.h resolves.
- Auto-generated config/reference.php gitignored.
Smoke verified offscreen: /healthz 200, /api/ping 200, 1 publish, 1
subscriber, zero 401s, clean shutdown with no zombies.
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-02 01:50:16 +02:00
|
|
|
// No live peer answered. The endpoint may not exist, or a stale
|
|
|
|
|
// file is left over from a crashed process — removeServer() handles
|
|
|
|
|
// both cases safely.
|
|
|
|
|
QLocalServer::removeServer(name);
|
|
|
|
|
|
|
|
|
|
for (int attempt = 0; attempt < kBindRetries; ++attempt) {
|
|
|
|
|
if (m_server.listen(name)) {
|
|
|
|
|
return true;
|
|
|
|
|
}
|
2026-05-02 01:18:43 +02:00
|
|
|
QThread::msleep(kBindRetryDelayMs);
|
|
|
|
|
}
|
|
|
|
|
|
Phase 1 sub-commit 6: skeleton wiring — make dev runs end-to-end
Symfony app under framework/skeleton/symfony/: minimal bin/console,
public/index.php, MicroKernel-based src/Kernel.php, services.yaml,
framework/security/mercure config, and a demo App\Controller\PingController
that GETs /api/ping (returning JSON pong) and republishes the same
payload to the Mercure topic app://ping. composer.json uses a path
repository to symlink the bundle from ../../php so local edits are
picked up live.
QML app under framework/skeleton/qml/: top-level CMake that
add_subdirectory's framework/qml, a main.cpp that creates the Qt
process, runs SingleInstance.acquireOrForward before any QML loads,
exposes SingleInstance via context property, and loadFromModule's
Skeleton.Main. Main.qml uses BackendConnection / RestClient /
MercureClient from PhpQml.Bridge and renders status dots, a Ping
button, and an event log.
Caddyfile binds 127.0.0.1:8765, enables in-memory Mercure with a
256-bit dev JWT (matches symfony/.env, lcobucci/jwt requires this).
Makefile wraps build / dev / doctor / clean; scripts/dev.sh starts
FrankenPHP --watch and the Qt host together with explicit PID-based
teardown (process-group `kill 0` proved unreliable when frankenphp's
watch fork reparented).
Bug fixes uncovered in this sub-commit:
- SingleInstance.acquireOrForward: probe-first, then removeServer +
retry-listen. The original loop-with-removeServer-after-failed-bind
silently exited on stale sockets from prior runs.
- Main.qml: MercureClient does NOT inherit BackendConnection.token —
Mercure subscribes anonymously in dev (Caddyfile), and forwarding
the bridge bearer made it 401-loop.
- /api/ping was 500ing because the dev MERCURE_JWT_SECRET was 144 bits;
bumped to 64-char (>=256 bit) to satisfy lcobucci/jwt.
- Linked the framework lib (php_qml_bridge) explicitly in addition to
the QML plugin so SingleInstance.h resolves.
- Auto-generated config/reference.php gitignored.
Smoke verified offscreen: /healthz 200, /api/ping 200, 1 publish, 1
subscriber, zero 401s, clean shutdown with no zombies.
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-02 01:50:16 +02:00
|
|
|
// Exhausted retries — better to continue as a degraded "live" instance
|
|
|
|
|
// than to deadlock-exit (PLAN.md §3 *Edge cases — Single-instance
|
|
|
|
|
// launch race*). Subsequent invocations may not be forwarded, but
|
|
|
|
|
// this process will still run.
|
2026-05-02 01:18:43 +02:00
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void SingleInstance::onNewConnection()
|
|
|
|
|
{
|
|
|
|
|
while (auto* socket = m_server.nextPendingConnection()) {
|
|
|
|
|
connect(socket, &QLocalSocket::readyRead, this, [this, socket]() {
|
|
|
|
|
QDataStream in(socket);
|
|
|
|
|
in.setVersion(QDataStream::Qt_6_5);
|
|
|
|
|
QStringList args;
|
|
|
|
|
in >> args;
|
|
|
|
|
if (in.status() == QDataStream::Ok) {
|
|
|
|
|
emit launchArgsReceived(args);
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
connect(socket, &QLocalSocket::disconnected, socket, &QLocalSocket::deleteLater);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
} // namespace PhpQml::Bridge
|