/* * Copyright (c) 2020-2022, the SerenityOS developers. * * SPDX-License-Identifier: BSD-2-Clause */ #pragma once #include #include #include #include #include #include #include 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(Core::NetworkJob::response()); } HttpResponse const* response() const { return static_cast(Core::NetworkJob::response()); } protected: void finish_up(); void on_socket_connected(); void flush_received_buffers(); void register_on_ready_to_read(Function); ErrorOr read_line(size_t); ErrorOr 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 m_headers; Vector 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 m_received_buffers; size_t m_buffered_size { 0 }; size_t m_received_size { 0 }; Optional m_content_length; Optional m_current_chunk_remaining_size; Optional 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 }; }; }