diff options
author | AnotherTest <ali.mpfard@gmail.com> | 2020-07-26 06:38:33 +0430 |
---|---|---|
committer | Andreas Kling <kling@serenityos.org> | 2020-07-26 14:55:47 +0200 |
commit | ec0315883beca84ab143f094d10703034b6dc896 (patch) | |
tree | bf981d7b43b4d248967243b0ee43ad659f2e8cb0 /Libraries | |
parent | 49c5acaa3d36db94fff37c4c4aca299c49420868 (diff) | |
download | serenity-ec0315883beca84ab143f094d10703034b6dc896.zip |
LibHTTP: Be more tolerant about bad chunked encoding trailers
Some servers (*glares at cloudflare*) like to send two last chunks,
which is strictly against the spec. Let's be more tolerant of this
behaviour.
Diffstat (limited to 'Libraries')
-rw-r--r-- | Libraries/LibHTTP/HttpJob.h | 1 | ||||
-rw-r--r-- | Libraries/LibHTTP/Job.cpp | 18 |
2 files changed, 19 insertions, 0 deletions
diff --git a/Libraries/LibHTTP/HttpJob.h b/Libraries/LibHTTP/HttpJob.h index 9eff924e34..4cd306c102 100644 --- a/Libraries/LibHTTP/HttpJob.h +++ b/Libraries/LibHTTP/HttpJob.h @@ -51,6 +51,7 @@ public: virtual void shutdown() override; protected: + virtual bool should_fail_on_empty_payload() const override { return false; } virtual void register_on_ready_to_read(Function<void()>) override; virtual void register_on_ready_to_write(Function<void()>) override; virtual bool can_read_line() const override; diff --git a/Libraries/LibHTTP/Job.cpp b/Libraries/LibHTTP/Job.cpp index c1d5e8bae6..36d47f9455 100644 --- a/Libraries/LibHTTP/Job.cpp +++ b/Libraries/LibHTTP/Job.cpp @@ -122,6 +122,12 @@ void Job::on_socket_connected() return; auto line = read_line(PAGE_SIZE); if (line.is_null()) { + if (m_state == State::AfterChunkedEncodingTrailer) { + // Some servers like to send two ending chunks + // use this fact as an excuse to ignore anything after the last chunk + // that is not a valid trailing header. + return finish_up(); + } fprintf(stderr, "Job: Expected HTTP header\n"); return did_fail(Core::NetworkJob::Error::ProtocolFailed); } @@ -136,11 +142,23 @@ void Job::on_socket_connected() } auto parts = chomped_line.split(':'); if (parts.is_empty()) { + if (m_state == State::AfterChunkedEncodingTrailer) { + // Some servers like to send two ending chunks + // use this fact as an excuse to ignore anything after the last chunk + // that is not a valid trailing header. + return finish_up(); + } fprintf(stderr, "Job: Expected HTTP header with key/value\n"); return deferred_invoke([this](auto&) { did_fail(Core::NetworkJob::Error::ProtocolFailed); }); } auto name = parts[0]; if (chomped_line.length() < name.length() + 2) { + if (m_state == State::AfterChunkedEncodingTrailer) { + // Some servers like to send two ending chunks + // use this fact as an excuse to ignore anything after the last chunk + // that is not a valid trailing header. + return finish_up(); + } fprintf(stderr, "Job: Malformed HTTP header: '%s' (%zu)\n", chomped_line.characters(), chomped_line.length()); return deferred_invoke([this](auto&) { did_fail(Core::NetworkJob::Error::ProtocolFailed); }); } |