mirror of
https://github.com/RGBCube/serenity
synced 2025-10-24 16:52:33 +00:00

Unlike HTTP/1.1 and above, the default behaviour for HTTP/1.0 servers is to close the connection after sending the response.
86 lines
2.3 KiB
C++
86 lines
2.3 KiB
C++
/*
|
|
* Copyright (c) 2020-2022, the SerenityOS developers.
|
|
*
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
*/
|
|
|
|
#pragma once
|
|
|
|
#include <AK/FileStream.h>
|
|
#include <AK/HashMap.h>
|
|
#include <AK/NonnullOwnPtrVector.h>
|
|
#include <AK/Optional.h>
|
|
#include <LibCore/NetworkJob.h>
|
|
#include <LibHTTP/HttpRequest.h>
|
|
#include <LibHTTP/HttpResponse.h>
|
|
|
|
namespace HTTP {
|
|
|
|
class Job : public Core::NetworkJob {
|
|
C_OBJECT(Job);
|
|
|
|
public:
|
|
explicit Job(HttpRequest&&, Core::Stream::Stream&);
|
|
virtual ~Job() override = default;
|
|
|
|
virtual void start(Core::Stream::Socket&) override;
|
|
virtual void shutdown(ShutdownMode) override;
|
|
|
|
Core::Stream::Socket const* socket() const { return m_socket; }
|
|
URL url() const { return m_request.url(); }
|
|
|
|
HttpResponse* response() { return static_cast<HttpResponse*>(Core::NetworkJob::response()); }
|
|
HttpResponse const* response() const { return static_cast<HttpResponse const*>(Core::NetworkJob::response()); }
|
|
|
|
protected:
|
|
void finish_up();
|
|
void on_socket_connected();
|
|
void flush_received_buffers();
|
|
void register_on_ready_to_read(Function<void()>);
|
|
ErrorOr<String> read_line(size_t);
|
|
ErrorOr<ByteBuffer> receive(size_t);
|
|
void timer_event(Core::TimerEvent&) override;
|
|
|
|
enum class State {
|
|
InStatus,
|
|
InHeaders,
|
|
InBody,
|
|
Trailers,
|
|
Finished,
|
|
};
|
|
|
|
HttpRequest m_request;
|
|
State m_state { State::InStatus };
|
|
Core::Stream::BufferedSocketBase* m_socket { nullptr };
|
|
bool m_legacy_connection { false };
|
|
int m_code { -1 };
|
|
HashMap<String, String, CaseInsensitiveStringTraits> m_headers;
|
|
Vector<String> m_set_cookie_headers;
|
|
|
|
struct ReceivedBuffer {
|
|
ReceivedBuffer(ByteBuffer d)
|
|
: data(move(d))
|
|
, pending_flush(data.bytes())
|
|
{
|
|
}
|
|
|
|
// The entire received buffer.
|
|
ByteBuffer data;
|
|
|
|
// The bytes we have yet to flush. (This is a slice of `data`)
|
|
ReadonlyBytes pending_flush;
|
|
};
|
|
|
|
NonnullOwnPtrVector<ReceivedBuffer> m_received_buffers;
|
|
|
|
size_t m_buffered_size { 0 };
|
|
size_t m_received_size { 0 };
|
|
Optional<u32> m_content_length;
|
|
Optional<ssize_t> m_current_chunk_remaining_size;
|
|
Optional<size_t> m_current_chunk_total_size;
|
|
bool m_can_stream_response { true };
|
|
bool m_should_read_chunk_ending_line { false };
|
|
bool m_has_scheduled_finish { false };
|
|
};
|
|
|
|
}
|