2022-07-16 17:55:29 +01:00
/*
* Copyright ( c ) 2022 , Linus Groh < linusg @ serenityos . org >
*
* 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-25 23:03:54 +01:00
# include <AK/URLParser.h>
2022-10-30 01:52:07 +00:00
# include <LibJS/Heap/Heap.h>
# 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>
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.
2022-10-30 01:52:07 +00:00
JS : : NonnullGCPtr < Response > Response : : aborted_network_error ( JS : : VM & vm )
2022-07-16 17:55:29 +01:00
{
2022-10-23 19:10:17 +02: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 ;
}
2022-12-04 18:02:33 +00:00
JS : : NonnullGCPtr < Response > Response : : network_error ( JS : : VM & vm , DeprecatedString message )
2022-07-16 17:55:29 +01:00
{
2022-10-23 22:16:14 +01:00
dbgln_if ( WEB_FETCH_DEBUG , " Fetch: Creating network error response with message: {} " , message ) ;
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
JS : : NonnullGCPtr < Response > Response : : appropriate_network_error ( JS : : VM & vm , FetchParams const & fetch_params )
{
// 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 ( )
? aborted_network_error ( vm )
2022-10-23 19:10:17 +02:00
: 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
2022-12-04 18:02:33 +00:00
ErrorOr < Optional < AK : : URL > > Response : : location_url ( Optional < DeprecatedString > 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.
auto location_values = TRY ( extract_header_list_values ( " Location " sv . bytes ( ) , m_header_list ) ) ;
if ( ! location_values . has_value ( ) | | location_values - > size ( ) ! = 1 )
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.
2022-10-25 23:03:54 +01:00
auto base_url = * url ( ) ;
auto location = AK : : URLParser : : parse ( location_values - > first ( ) , & base_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 ( ) )
location . set_fragment ( request_fragment . value_or ( { } ) ) ;
// 5. Return location.
return location ;
}
2022-09-25 19:23:35 +01:00
// https://fetch.spec.whatwg.org/#concept-response-clone
2022-10-30 01:52:07 +00:00
WebIDL : : ExceptionOr < JS : : NonnullGCPtr < Response > > Response : : clone ( JS : : VM & vm ) const
2022-09-25 19:23:35 +01:00
{
// To clone a response response, run these steps:
2022-10-24 18:46:27 +01:00
auto & realm = * vm . current_realm ( ) ;
// 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 ) ) {
2022-10-30 01:52:07 +00:00
auto internal_response = TRY ( static_cast < FilteredResponse const & > ( * this ) . internal_response ( ) - > clone ( vm ) ) ;
2022-10-24 18:46:27 +01:00
if ( is < BasicFilteredResponse > ( * this ) )
2022-10-30 01:52:07 +00:00
return TRY_OR_RETURN_OOM ( realm , BasicFilteredResponse : : create ( vm , internal_response ) ) ;
2022-10-24 18:46:27 +01:00
if ( is < CORSFilteredResponse > ( * this ) )
2022-10-30 01:52:07 +00:00
return TRY_OR_RETURN_OOM ( realm , 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 ( ) )
new_response - > set_body ( TRY ( m_body - > clone ( ) ) ) ;
2022-09-25 19:23:35 +01:00
// 4. Return newResponse.
return new_response ;
}
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
}
}