HeadersIterator.cpp 2.5 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768
  1. /*
  2. * Copyright (c) 2022, Linus Groh <linusg@serenityos.org>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <LibJS/Runtime/Array.h>
  7. #include <LibJS/Runtime/IteratorOperations.h>
  8. #include <LibWeb/Bindings/HeadersIteratorPrototype.h>
  9. #include <LibWeb/Fetch/HeadersIterator.h>
  10. #include <LibWeb/HTML/Window.h>
  11. namespace Web::Fetch {
  12. JS::NonnullGCPtr<HeadersIterator> HeadersIterator::create(Headers const& headers, JS::Object::PropertyKind iteration_kind)
  13. {
  14. return *headers.heap().allocate<HeadersIterator>(headers.realm(), headers, iteration_kind);
  15. }
  16. HeadersIterator::HeadersIterator(Headers const& headers, JS::Object::PropertyKind iteration_kind)
  17. : PlatformObject(headers.realm())
  18. , m_headers(headers)
  19. , m_iteration_kind(iteration_kind)
  20. {
  21. set_prototype(&headers.global_object().ensure_web_prototype<Bindings::HeadersIteratorPrototype>("HeadersIterator"));
  22. }
  23. HeadersIterator::~HeadersIterator() = default;
  24. void HeadersIterator::visit_edges(JS::Cell::Visitor& visitor)
  25. {
  26. Base::visit_edges(visitor);
  27. visitor.visit(&m_headers);
  28. }
  29. // https://webidl.spec.whatwg.org/#es-iterable, Step 2
  30. JS::ThrowCompletionOr<JS::Object*> HeadersIterator::next()
  31. {
  32. // The value pairs to iterate over are the return value of running sort and combine with this’s header list.
  33. auto value_pairs_to_iterate_over = [&]() -> JS::ThrowCompletionOr<Vector<Fetch::Infrastructure::Header>> {
  34. auto headers_or_error = m_headers.m_header_list->sort_and_combine();
  35. if (headers_or_error.is_error())
  36. return vm().throw_completion<JS::InternalError>(JS::ErrorType::NotEnoughMemoryToAllocate);
  37. return headers_or_error.release_value();
  38. };
  39. auto pairs = TRY(value_pairs_to_iterate_over());
  40. if (m_index >= pairs.size())
  41. return create_iterator_result_object(vm(), JS::js_undefined(), true);
  42. auto const& pair = pairs[m_index++];
  43. switch (m_iteration_kind) {
  44. case JS::Object::PropertyKind::Key:
  45. return create_iterator_result_object(vm(), JS::js_string(vm(), StringView { pair.name }), false);
  46. case JS::Object::PropertyKind::Value:
  47. return create_iterator_result_object(vm(), JS::js_string(vm(), StringView { pair.value }), false);
  48. case JS::Object::PropertyKind::KeyAndValue: {
  49. auto* array = JS::Array::create_from(realm(), { JS::js_string(vm(), StringView { pair.name }), JS::js_string(vm(), StringView { pair.value }) });
  50. return create_iterator_result_object(vm(), array, false);
  51. }
  52. default:
  53. VERIFY_NOT_REACHED();
  54. }
  55. }
  56. }