HeadersIterator.cpp 2.0 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556
  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/HeadersIteratorWrapper.h>
  9. #include <LibWeb/Bindings/Wrapper.h>
  10. #include <LibWeb/Fetch/HeadersIterator.h>
  11. namespace Web::Fetch {
  12. // https://webidl.spec.whatwg.org/#es-iterable, Step 2
  13. JS::ThrowCompletionOr<JS::Object*> HeadersIterator::next()
  14. {
  15. auto& global_object = wrapper()->global_object();
  16. auto& vm = global_object.vm();
  17. auto& realm = *global_object.associated_realm();
  18. // The value pairs to iterate over are the return value of running sort and combine with this’s header list.
  19. auto value_pairs_to_iterate_over = [&]() -> JS::ThrowCompletionOr<Vector<Fetch::Infrastructure::Header>> {
  20. auto headers_or_error = m_headers.m_header_list.sort_and_combine();
  21. if (headers_or_error.is_error())
  22. return vm.throw_completion<JS::InternalError>(JS::ErrorType::NotEnoughMemoryToAllocate);
  23. return headers_or_error.release_value();
  24. };
  25. auto pairs = TRY(value_pairs_to_iterate_over());
  26. if (m_index >= pairs.size())
  27. return create_iterator_result_object(vm, JS::js_undefined(), true);
  28. auto const& pair = pairs[m_index++];
  29. switch (m_iteration_kind) {
  30. case JS::Object::PropertyKind::Key:
  31. return create_iterator_result_object(vm, JS::js_string(vm, StringView { pair.name }), false);
  32. case JS::Object::PropertyKind::Value:
  33. return create_iterator_result_object(vm, JS::js_string(vm, StringView { pair.value }), false);
  34. case JS::Object::PropertyKind::KeyAndValue: {
  35. auto* array = JS::Array::create_from(realm, { JS::js_string(vm, StringView { pair.name }), JS::js_string(vm, StringView { pair.value }) });
  36. return create_iterator_result_object(vm, array, false);
  37. }
  38. default:
  39. VERIFY_NOT_REACHED();
  40. }
  41. }
  42. void HeadersIterator::visit_edges(JS::Cell::Visitor& visitor)
  43. {
  44. visitor.visit(m_headers.wrapper());
  45. }
  46. }