2022-07-16 17:55:29 +01:00
/*
2023-03-02 23:26:35 +00:00
* Copyright ( c ) 2022 - 2023 , Linus Groh < linusg @ serenityos . org >
2022-07-16 17:55:29 +01:00
*
* SPDX - License - Identifier : BSD - 2 - Clause
*/
2022-10-23 22:16:14 +01:00
# include <AK/Debug.h>
2022-11-23 13:28:01 +01:00
# include <AK/TypeCasts.h>
2022-10-30 01:52:07 +00:00
# include <LibJS/Heap/Heap.h>
2023-01-07 12:14:54 -05:00
# include <LibJS/Runtime/Completion.h>
2022-10-30 01:52:07 +00:00
# include <LibJS/Runtime/VM.h>
2022-10-24 18:46:27 +01:00
# include <LibWeb/Bindings/MainThreadVM.h>
2022-10-23 22:50:38 +01:00
# include <LibWeb/Fetch/Infrastructure/FetchParams.h>
2022-09-25 19:23:35 +01:00
# include <LibWeb/Fetch/Infrastructure/HTTP/Bodies.h>
2022-07-16 17:55:29 +01:00
# include <LibWeb/Fetch/Infrastructure/HTTP/Responses.h>
2023-07-15 14:24:58 +12:00
# include <LibWeb/URL/URL.h>
2022-07-16 17:55:29 +01:00
2022-07-17 23:52:02 +01:00
namespace Web : : Fetch : : Infrastructure {
2022-07-16 17:55:29 +01:00
2022-10-30 01:52:07 +00:00
Response : : Response ( JS : : NonnullGCPtr < HeaderList > header_list )
: m_header_list ( header_list )
2022-09-25 20:52:51 +01:00
{
}
2022-10-30 01:52:07 +00:00
void Response : : visit_edges ( JS : : Cell : : Visitor & visitor )
2022-10-04 23:45:47 +01:00
{
2022-10-30 01:52:07 +00:00
Base : : visit_edges ( visitor ) ;
visitor . visit ( m_header_list ) ;
}
JS : : NonnullGCPtr < Response > Response : : create ( JS : : VM & vm )
{
2022-12-14 17:40:33 +00:00
return vm . heap ( ) . allocate_without_realm < Response > ( HeaderList : : create ( vm ) ) ;
2022-10-04 23:45:47 +01:00
}
2022-07-16 17:55:29 +01:00
// https://fetch.spec.whatwg.org/#ref-for-concept-network-error%E2%91%A3
// A network error is a response whose status is always 0, status message is always
// the empty byte sequence, header list is always empty, and body is always null.
2023-03-03 18:02:43 +00:00
JS : : NonnullGCPtr < Response > Response : : aborted_network_error ( JS : : VM & vm )
2022-07-16 17:55:29 +01:00
{
2023-03-03 18:02:43 +00:00
auto response = network_error ( vm , " Fetch has been aborted " sv ) ;
2022-09-24 00:28:11 +01:00
response - > set_aborted ( true ) ;
2022-07-16 17:55:29 +01:00
return response ;
}
2023-03-03 18:02:43 +00:00
JS : : NonnullGCPtr < Response > Response : : network_error ( JS : : VM & vm , Variant < String , StringView > message )
2022-07-16 17:55:29 +01:00
{
2023-03-03 18:02:43 +00:00
dbgln_if ( WEB_FETCH_DEBUG , " Fetch: Creating network error response with message: {} " , message . visit ( [ ] ( auto const & s ) - > StringView { return s ; } ) ) ;
2022-10-30 01:52:07 +00:00
auto response = Response : : create ( vm ) ;
2022-09-24 00:28:11 +01:00
response - > set_status ( 0 ) ;
response - > set_type ( Type : : Error ) ;
VERIFY ( ! response - > body ( ) . has_value ( ) ) ;
2022-10-23 19:10:17 +02:00
response - > m_network_error_message = move ( message ) ;
2022-07-16 17:55:29 +01:00
return response ;
}
2022-10-23 22:50:38 +01:00
// https://fetch.spec.whatwg.org/#appropriate-network-error
2023-03-03 18:02:43 +00:00
JS : : NonnullGCPtr < Response > Response : : appropriate_network_error ( JS : : VM & vm , FetchParams const & fetch_params )
2022-10-23 22:50:38 +01:00
{
// 1. Assert: fetchParams is canceled.
VERIFY ( fetch_params . is_canceled ( ) ) ;
// 2. Return an aborted network error if fetchParams is aborted; otherwise return a network error.
return fetch_params . is_aborted ( )
2023-03-03 18:02:43 +00:00
? aborted_network_error ( vm )
: network_error ( vm , " Fetch has been terminated " sv ) ;
2022-10-23 22:50:38 +01:00
}
2022-07-16 17:55:29 +01:00
// https://fetch.spec.whatwg.org/#concept-aborted-network-error
bool Response : : is_aborted_network_error ( ) const
{
// A response whose type is "error" and aborted flag is set is known as an aborted network error.
2022-10-24 18:51:25 +01:00
// NOTE: We have to use the virtual getter here to not bypass filtered responses.
return type ( ) = = Type : : Error & & aborted ( ) ;
2022-07-16 17:55:29 +01:00
}
// https://fetch.spec.whatwg.org/#concept-network-error
bool Response : : is_network_error ( ) const
{
// A response whose type is "error" is known as a network error.
2022-10-24 18:51:25 +01:00
// NOTE: We have to use the virtual getter here to not bypass filtered responses.
return type ( ) = = Type : : Error ;
2022-07-16 17:55:29 +01:00
}
// https://fetch.spec.whatwg.org/#concept-response-url
Optional < AK : : URL const & > Response : : url ( ) const
{
// A response has an associated URL. It is a pointer to the last URL in response’ s URL list and null if response’ s URL list is empty.
2022-10-24 18:51:25 +01:00
// NOTE: We have to use the virtual getter here to not bypass filtered responses.
if ( url_list ( ) . is_empty ( ) )
2022-07-16 17:55:29 +01:00
return { } ;
2022-10-24 18:51:25 +01:00
return url_list ( ) . last ( ) ;
2022-07-16 17:55:29 +01:00
}
// https://fetch.spec.whatwg.org/#concept-response-location-url
2023-03-02 23:26:35 +00:00
ErrorOr < Optional < AK : : URL > > Response : : location_url ( Optional < String > const & request_fragment ) const
2022-07-16 17:55:29 +01:00
{
// The location URL of a response response, given null or an ASCII string requestFragment, is the value returned by the following steps. They return null, failure, or a URL.
// 1. If response’ s status is not a redirect status, then return null.
2022-10-24 18:51:25 +01:00
// NOTE: We have to use the virtual getter here to not bypass filtered responses.
if ( ! is_redirect_status ( status ( ) ) )
2022-07-16 17:55:29 +01:00
return Optional < AK : : URL > { } ;
2022-10-25 23:03:39 +01:00
// 2. Let location be the result of extracting header list values given `Location` and response’ s header list.
2023-02-08 23:32:44 +00:00
auto location_values_or_failure = TRY ( extract_header_list_values ( " Location " sv . bytes ( ) , m_header_list ) ) ;
if ( location_values_or_failure . has < Infrastructure : : ExtractHeaderParseFailure > ( ) | | location_values_or_failure . has < Empty > ( ) )
return Optional < AK : : URL > { } ;
auto const & location_values = location_values_or_failure . get < Vector < ByteBuffer > > ( ) ;
if ( location_values . size ( ) ! = 1 )
2022-10-25 23:03:39 +01:00
return Optional < AK : : URL > { } ;
2022-07-16 17:55:29 +01:00
// 3. If location is a header value, then set location to the result of parsing location with response’ s URL.
2023-07-15 14:24:58 +12:00
auto location = URL : : parse ( location_values . first ( ) , url ( ) ) ;
2022-07-16 17:55:29 +01:00
if ( ! location . is_valid ( ) )
return Error : : from_string_view ( " Invalid 'Location' header URL " sv ) ;
// 4. If location is a URL whose fragment is null, then set location’ s fragment to requestFragment.
if ( location . fragment ( ) . is_null ( ) )
2023-03-02 23:26:35 +00:00
location . set_fragment ( request_fragment . has_value ( ) ? request_fragment - > to_deprecated_string ( ) : DeprecatedString { } ) ;
2022-07-16 17:55:29 +01:00
// 5. Return location.
return location ;
}
2022-09-25 19:23:35 +01:00
// https://fetch.spec.whatwg.org/#concept-response-clone
2023-02-28 17:45:49 +00:00
WebIDL : : ExceptionOr < JS : : NonnullGCPtr < Response > > Response : : clone ( JS : : Realm & realm ) const
2022-09-25 19:23:35 +01:00
{
// To clone a response response, run these steps:
2023-02-28 17:45:49 +00:00
auto & vm = realm . vm ( ) ;
2022-09-25 19:23:35 +01:00
2022-10-24 18:46:27 +01:00
// 1. If response is a filtered response, then return a new identical filtered response whose internal response is a clone of response’ s internal response.
if ( is < FilteredResponse > ( * this ) ) {
2023-02-28 17:45:49 +00:00
auto internal_response = TRY ( static_cast < FilteredResponse const & > ( * this ) . internal_response ( ) - > clone ( realm ) ) ;
2022-10-24 18:46:27 +01:00
if ( is < BasicFilteredResponse > ( * this ) )
2023-01-07 12:14:54 -05:00
return TRY_OR_THROW_OOM ( vm , BasicFilteredResponse : : create ( vm , internal_response ) ) ;
2022-10-24 18:46:27 +01:00
if ( is < CORSFilteredResponse > ( * this ) )
2023-01-07 12:14:54 -05:00
return TRY_OR_THROW_OOM ( vm , CORSFilteredResponse : : create ( vm , internal_response ) ) ;
2022-10-24 18:46:27 +01:00
if ( is < OpaqueFilteredResponse > ( * this ) )
2022-10-30 01:52:07 +00:00
return OpaqueFilteredResponse : : create ( vm , internal_response ) ;
2022-10-24 18:46:27 +01:00
if ( is < OpaqueRedirectFilteredResponse > ( * this ) )
2022-10-30 01:52:07 +00:00
return OpaqueRedirectFilteredResponse : : create ( vm , internal_response ) ;
2022-10-24 18:46:27 +01:00
VERIFY_NOT_REACHED ( ) ;
}
2022-09-25 19:23:35 +01:00
// 2. Let newResponse be a copy of response, except for its body.
2022-10-30 01:52:07 +00:00
auto new_response = Infrastructure : : Response : : create ( vm ) ;
2022-10-04 23:45:47 +01:00
new_response - > set_type ( m_type ) ;
new_response - > set_aborted ( m_aborted ) ;
new_response - > set_url_list ( m_url_list ) ;
new_response - > set_status ( m_status ) ;
new_response - > set_status_message ( m_status_message ) ;
for ( auto const & header : * m_header_list )
MUST ( new_response - > header_list ( ) - > append ( header ) ) ;
new_response - > set_cache_state ( m_cache_state ) ;
new_response - > set_cors_exposed_header_name_list ( m_cors_exposed_header_name_list ) ;
new_response - > set_range_requested ( m_range_requested ) ;
new_response - > set_request_includes_credentials ( m_request_includes_credentials ) ;
new_response - > set_timing_allow_passed ( m_timing_allow_passed ) ;
new_response - > set_body_info ( m_body_info ) ;
// FIXME: service worker timing info
2022-09-25 19:25:53 +01:00
// 3. If response’ s body is non-null, then set newResponse’ s body to the result of cloning response’ s body.
if ( m_body . has_value ( ) )
2023-08-13 13:05:26 +02:00
new_response - > set_body ( m_body - > clone ( realm ) ) ;
2022-09-25 19:23:35 +01:00
// 4. Return newResponse.
return new_response ;
}
2023-05-10 16:57:23 -04:00
// https://html.spec.whatwg.org/multipage/urls-and-fetching.html#unsafe-response
JS : : NonnullGCPtr < Response > Response : : unsafe_response ( )
{
// A response's unsafe response is its internal response if it has one, and the response itself otherwise.
if ( is < FilteredResponse > ( this ) )
return static_cast < FilteredResponse & > ( * this ) . internal_response ( ) ;
return * this ;
}
2023-05-10 16:27:55 -04:00
// https://html.spec.whatwg.org/multipage/urls-and-fetching.html#cors-cross-origin
bool Response : : is_cors_cross_origin ( ) const
{
// A response whose type is "opaque" or "opaqueredirect" is CORS-cross-origin.
return type ( ) = = Type : : Opaque | | type ( ) = = Type : : OpaqueRedirect ;
}
2023-03-03 18:02:43 +00:00
// Non-standard
Optional < StringView > Response : : network_error_message ( ) const
{
if ( ! m_network_error_message . has_value ( ) )
return { } ;
return m_network_error_message - > visit ( [ ] ( auto const & s ) - > StringView { return s ; } ) ;
}
2022-10-30 01:52:07 +00:00
FilteredResponse : : FilteredResponse ( JS : : NonnullGCPtr < Response > internal_response , JS : : NonnullGCPtr < HeaderList > header_list )
: Response ( header_list )
, m_internal_response ( internal_response )
2022-07-16 17:55:29 +01:00
{
}
FilteredResponse : : ~ FilteredResponse ( )
{
}
2022-10-30 01:52:07 +00:00
void FilteredResponse : : visit_edges ( JS : : Cell : : Visitor & visitor )
{
Base : : visit_edges ( visitor ) ;
visitor . visit ( m_internal_response ) ;
}
ErrorOr < JS : : NonnullGCPtr < BasicFilteredResponse > > BasicFilteredResponse : : create ( JS : : VM & vm , JS : : NonnullGCPtr < Response > internal_response )
2022-07-16 17:55:29 +01:00
{
// A basic filtered response is a filtered response whose type is "basic" and header list excludes
// any headers in internal response’ s header list whose name is a forbidden response-header name.
2022-10-30 01:52:07 +00:00
auto header_list = HeaderList : : create ( vm ) ;
2022-10-04 23:45:47 +01:00
for ( auto const & header : * internal_response - > header_list ( ) ) {
2022-07-16 17:55:29 +01:00
if ( ! is_forbidden_response_header_name ( header . name ) )
2022-09-25 20:52:51 +01:00
TRY ( header_list - > append ( header ) ) ;
2022-07-16 17:55:29 +01:00
}
2022-12-14 17:40:33 +00:00
return vm . heap ( ) . allocate_without_realm < BasicFilteredResponse > ( internal_response , header_list ) ;
2022-07-16 17:55:29 +01:00
}
2022-10-30 01:52:07 +00:00
BasicFilteredResponse : : BasicFilteredResponse ( JS : : NonnullGCPtr < Response > internal_response , JS : : NonnullGCPtr < HeaderList > header_list )
: FilteredResponse ( internal_response , header_list )
, m_header_list ( header_list )
2022-07-16 17:55:29 +01:00
{
}
2022-10-30 01:52:07 +00:00
void BasicFilteredResponse : : visit_edges ( JS : : Cell : : Visitor & visitor )
{
Base : : visit_edges ( visitor ) ;
visitor . visit ( m_header_list ) ;
}
ErrorOr < JS : : NonnullGCPtr < CORSFilteredResponse > > CORSFilteredResponse : : create ( JS : : VM & vm , JS : : NonnullGCPtr < Response > internal_response )
2022-07-16 17:55:29 +01:00
{
// A CORS filtered response is a filtered response whose type is "cors" and header list excludes
// any headers in internal response’ s header list whose name is not a CORS-safelisted response-header
// name, given internal response’ s CORS-exposed header-name list.
Vector < ReadonlyBytes > cors_exposed_header_name_list ;
2022-10-04 23:45:47 +01:00
for ( auto const & header_name : internal_response - > cors_exposed_header_name_list ( ) )
2022-07-16 17:55:29 +01:00
cors_exposed_header_name_list . append ( header_name . span ( ) ) ;
2022-10-30 01:52:07 +00:00
auto header_list = HeaderList : : create ( vm ) ;
2022-10-04 23:45:47 +01:00
for ( auto const & header : * internal_response - > header_list ( ) ) {
2022-07-16 17:55:29 +01:00
if ( is_cors_safelisted_response_header_name ( header . name , cors_exposed_header_name_list ) )
2022-09-25 20:52:51 +01:00
TRY ( header_list - > append ( header ) ) ;
2022-07-16 17:55:29 +01:00
}
2022-12-14 17:40:33 +00:00
return vm . heap ( ) . allocate_without_realm < CORSFilteredResponse > ( internal_response , header_list ) ;
2022-07-16 17:55:29 +01:00
}
2022-10-30 01:52:07 +00:00
CORSFilteredResponse : : CORSFilteredResponse ( JS : : NonnullGCPtr < Response > internal_response , JS : : NonnullGCPtr < HeaderList > header_list )
: FilteredResponse ( internal_response , header_list )
, m_header_list ( header_list )
2022-07-16 17:55:29 +01:00
{
}
2022-10-30 01:52:07 +00:00
void CORSFilteredResponse : : visit_edges ( JS : : Cell : : Visitor & visitor )
2022-07-16 17:55:29 +01:00
{
2022-10-30 01:52:07 +00:00
Base : : visit_edges ( visitor ) ;
visitor . visit ( m_header_list ) ;
}
JS : : NonnullGCPtr < OpaqueFilteredResponse > OpaqueFilteredResponse : : create ( JS : : VM & vm , JS : : NonnullGCPtr < Response > internal_response )
{
// An opaque filtered response is a filtered response whose type is "opaque", URL list is the empty list,
2022-07-16 17:55:29 +01:00
// status is 0, status message is the empty byte sequence, header list is empty, and body is null.
2022-12-14 17:40:33 +00:00
return vm . heap ( ) . allocate_without_realm < OpaqueFilteredResponse > ( internal_response , HeaderList : : create ( vm ) ) ;
2022-10-30 01:52:07 +00:00
}
OpaqueFilteredResponse : : OpaqueFilteredResponse ( JS : : NonnullGCPtr < Response > internal_response , JS : : NonnullGCPtr < HeaderList > header_list )
: FilteredResponse ( internal_response , header_list )
, m_header_list ( header_list )
{
2022-07-16 17:55:29 +01:00
}
2022-10-30 01:52:07 +00:00
void OpaqueFilteredResponse : : visit_edges ( JS : : Cell : : Visitor & visitor )
2022-07-16 17:55:29 +01:00
{
2022-10-30 01:52:07 +00:00
Base : : visit_edges ( visitor ) ;
visitor . visit ( m_header_list ) ;
}
JS : : NonnullGCPtr < OpaqueRedirectFilteredResponse > OpaqueRedirectFilteredResponse : : create ( JS : : VM & vm , JS : : NonnullGCPtr < Response > internal_response )
{
// An opaque-redirect filtered response is a filtered response whose type is "opaqueredirect",
// status is 0, status message is the empty byte sequence, header list is empty, and body is null.
2022-12-14 17:40:33 +00:00
return vm . heap ( ) . allocate_without_realm < OpaqueRedirectFilteredResponse > ( internal_response , HeaderList : : create ( vm ) ) ;
2022-07-16 17:55:29 +01:00
}
2022-10-30 01:52:07 +00:00
OpaqueRedirectFilteredResponse : : OpaqueRedirectFilteredResponse ( JS : : NonnullGCPtr < Response > internal_response , JS : : NonnullGCPtr < HeaderList > header_list )
: FilteredResponse ( internal_response , header_list )
, m_header_list ( header_list )
2022-07-16 17:55:29 +01:00
{
}
2022-10-30 01:52:07 +00:00
void OpaqueRedirectFilteredResponse : : visit_edges ( JS : : Cell : : Visitor & visitor )
2022-07-16 17:55:29 +01:00
{
2022-10-30 01:52:07 +00:00
Base : : visit_edges ( visitor ) ;
visitor . visit ( m_header_list ) ;
2022-07-16 17:55:29 +01:00
}
}