TrueAsync\HttpResponse
(PHP 8.6+, true_async_server 0.6+)
Об'єкт відповіді з fluent-інтерфейсом. Передається другим параметром в обробник. Створюється сервером — не конструюється користувачем.
namespace TrueAsync;
final class HttpResponse
{
// status
public function setStatusCode(int $code): static;
public function getStatusCode(): int;
public function setReasonPhrase(string $phrase): static;
public function getReasonPhrase(): string;
// headers
public function setHeader(string $name, string|array $value): static;
public function addHeader(string $name, string|array $value): static;
public function hasHeader(string $name): bool;
public function getHeader(string $name): ?string;
public function getHeaderLine(string $name): string;
public function getHeaders(): array;
public function resetHeaders(): static;
// trailers (HTTP/2)
public function setTrailer(string $name, string $value): static;
public function setTrailers(array $trailers): static;
public function resetTrailers(): static;
public function getTrailers(): array;
// protocol introspection
public function getProtocolName(): string;
public function getProtocolVersion(): string;
// body
public function write(string $data): static;
public function send(string $chunk): static;
public function sendable(): bool;
public function setNoCompression(): static;
public function getBody(): string;
public function setBody(string $body): static;
public function getBodyStream(): mixed; // TODO
public function setBodyStream(mixed $stream): static; // TODO
// helpers
public function json(array|string|object|null|int|float|bool $data, int $status = 200, int $flags = 0): static;
public function html(string $html): static;
public function redirect(string $url, int $status = 302): static;
// send / state
public function end(?string $data = null): void;
public function sendFile(string $path, ?SendFileOptions $options = null): void;
public function isHeadersSent(): bool;
public function isClosed(): bool;
}Статус
setStatusCode
public HttpResponse::setStatusCode(int $code): staticHTTP-код 100..599.
getStatusCode
public HttpResponse::getStatusCode(): intsetReasonPhrase / getReasonPhrase
public HttpResponse::setReasonPhrase(string $phrase): static
public HttpResponse::getReasonPhrase(): string"OK", "Not Found" тощо.
Заголовки
setHeader
public HttpResponse::setHeader(string $name, string|array $value): staticВстановити заголовок, замінюючи попередні значення.
addHeader
public HttpResponse::addHeader(string $name, string|array $value): staticДодати значення до наявних (наприклад, Set-Cookie).
hasHeader / getHeader / getHeaderLine / getHeaders
public HttpResponse::hasHeader(string $name): bool
public HttpResponse::getHeader(string $name): ?string
public HttpResponse::getHeaderLine(string $name): string
public HttpResponse::getHeaders(): arrayCase-insensitive читання того, що встановив handler.
resetHeaders
public HttpResponse::resetHeaders(): staticЗняти всі заголовки.
Trailers (HTTP/2)
HEADERS-фрейм, що шлеться після тіла. Канонічний споживач — gRPC (grpc-status). На HTTP/1.1 значення мовчки ігнорується — chunked-encoding trailer emission не в межах Step 5b.
setTrailer
public HttpResponse::setTrailer(string $name, string $value): staticІм'я — lowercase (RFC 9113 §8.2.2); uppercase автоматично приводиться.
setTrailers
public HttpResponse::setTrailers(array $trailers): staticBulk-set. Наявні trailers зберігаються — для clean slate викликайте resetTrailers() спочатку.
resetTrailers
public HttpResponse::resetTrailers(): staticgetTrailers
public HttpResponse::getTrailers(): arrayПротокол
getProtocolName / getProtocolVersion
public HttpResponse::getProtocolName(): string // завжди "HTTP"
public HttpResponse::getProtocolVersion(): string // "1.1", "2", "3"Тіло
write
public HttpResponse::write(string $data): staticAppend у внутрішній body-буфер. Надсилання — на end() / автоматично при поверненні з handler'а.
send
public HttpResponse::send(string $chunk): staticНадіслати чанк клієнту (streaming).
- Перший
send()комітить статус + заголовки — змінити їх уже не можна. - Наступні — append DATA-фреймів (HTTP/2) або chunked-segments (HTTP/1).
- Блокує handler-корутину лише під backpressure (per-stream staging buffer заповнено). Дефолт backpressure-порогу:
setStreamWriteBufferBytes()— 256 KiB. - У звичайному випадку повертається одразу.
sendable
public HttpResponse::sendable(): boolAdvisory non-blocking перевірка:
true—send()прийме чанк без suspend'а корутини.false—send()заблокує на backpressure, або response уже sealedsendFile()'ом / закритий, або це не streaming-capable тип відповіді.
send() завжди безпечно викликати — sendable() лише дає handler'у можливість зайнятись іншою роботою замість блокування на повільному peer'і.
setNoCompression
public HttpResponse::setNoCompression(): staticЗаборона стиснення для цієї відповіді — перебиває Accept-Encoding, MIME-whitelist і size threshold. Застосовуйте на: BREACH-чутливих endpoints (секрети + reflected user input), payload'ах з уже виставленим Content-Encoding, тілах, які сервер не має обгортати. Ідемпотентно.
getBody / setBody
public HttpResponse::getBody(): string
public HttpResponse::setBody(string $body): staticGet/set поточного вмісту буфера.
Helpers
json
public HttpResponse::json(
array|string|object|null|int|float|bool $data,
int $status = 200,
int $flags = 0
): staticJSON-серіалізація через php_json_encode_ex (той самий шлях, що у json_encode()):
array/object/ scalar$data→ encoded.string$data→ шлеться як є (cached JSON, pre-built bytes). Skip re-encoding.
Content-Type: application/json ставиться лише якщо handler не виставив свій — chain setHeader('Content-Type', 'application/problem+json')->json($payload) для іншого media-type.
$flags — JSON_* bitmask. 0 — дефолти сервера з HttpServerConfig::setJsonEncodeFlags() (JSON_UNESCAPED_UNICODE | JSON_UNESCAPED_SLASHES з коробки).
JSON_THROW_ON_ERROR мовчки стрипається: помилка encode дає 500 JSON-помилки, виняток не пробрасується. Handler ніколи не має обгортати json() у try/catch.
html
public HttpResponse::html(string $html): staticСтавить Content-Type: text/html.
redirect
public HttpResponse::redirect(string $url, int $status = 302): staticНадсилання
end
public HttpResponse::end(?string $data = null): voidЗавершити відповідь і надіслати клієнту. Після end() нічого більше писати не можна.
sendFile
public HttpResponse::sendFile(string $path, ?SendFileOptions $options = null): voidHandler-driven доставка файлу. Записує path + options на response і відразу повертається — передача йде в dispose-фазі через ту саму FSM, що і StaticHandler (MIME, ETag, IMF-date, Range, conditional GET, precompressed sidecars).
Після sendFile() response sealed: setHeader / setStatus* / write / send / setBody / json / html / redirect / end / повторний sendFile() кидають HttpServerRuntimeException.
Шлях — довірений (handler прийняв рішення про доступ). Помилки open/fstat (ENOENT, EACCES, oversize, non-regular) — 500, бо заголовки ще не на дроті.
Compression middleware bypasses для sendFile-тіл (своя delivery pipeline).
HTTP/3 path для
sendFile()— у роботі; поки H3 dispose-хук відкидає з 500.
Див. SendFileOptions.
Стан
isHeadersSent
public HttpResponse::isHeadersSent(): boolisClosed
public HttpResponse::isClosed(): boolПриклад
use TrueAsync\HttpResponse;
use TrueAsync\SendFileOptions;
use TrueAsync\SendFileDisposition;
$server->addHttpHandler(function ($req, HttpResponse $res) {
// SSE
if ($req->getPath() === '/events') {
$res
->setStatusCode(200)
->setHeader('Content-Type', 'text/event-stream')
->setHeader('Cache-Control', 'no-store')
->setNoCompression();
foreach (loadEvents() as $event) {
$res->send("data: " . json_encode($event) . "\n\n");
}
return;
}
// sendFile
if ($req->getPath() === '/report.pdf') {
$res->sendFile('/var/reports/q1.pdf', new SendFileOptions(
disposition: SendFileDisposition::ATTACHMENT,
downloadName: 'Q1-Report.pdf',
));
return;
}
// JSON
$res->json(['ok' => true]);
});