TransformStream.cpp 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899
  1. /*
  2. * Copyright (c) 2023, Kenneth Myhra <kennethmyhra@serenityos.org>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <LibWeb/Bindings/Intrinsics.h>
  7. #include <LibWeb/Streams/AbstractOperations.h>
  8. #include <LibWeb/Streams/TransformStream.h>
  9. #include <LibWeb/Streams/TransformStreamDefaultController.h>
  10. #include <LibWeb/Streams/Transformer.h>
  11. #include <LibWeb/Streams/WritableStream.h>
  12. #include <LibWeb/WebIDL/AbstractOperations.h>
  13. #include <LibWeb/WebIDL/ExceptionOr.h>
  14. namespace Web::Streams {
  15. JS_DEFINE_ALLOCATOR(TransformStream);
  16. // https://streams.spec.whatwg.org/#ts-construct
  17. WebIDL::ExceptionOr<JS::NonnullGCPtr<TransformStream>> TransformStream::construct_impl(JS::Realm& realm, Optional<JS::Handle<JS::Object>> transformer_object, QueuingStrategy const& writable_strategy, QueuingStrategy const& readable_strategy)
  18. {
  19. auto& vm = realm.vm();
  20. auto stream = realm.heap().allocate<TransformStream>(realm, realm);
  21. // 1. If transformer is missing, set it to null.
  22. auto transformer = transformer_object.has_value() ? JS::Value { transformer_object.value() } : JS::js_null();
  23. // 2. Let transformerDict be transformer, converted to an IDL value of type Transformer.
  24. auto transformer_dict = TRY(Transformer::from_value(vm, transformer));
  25. // 3. If transformerDict["readableType"] exists, throw a RangeError exception.
  26. if (transformer_dict.readable_type.has_value())
  27. return WebIDL::SimpleException { WebIDL::SimpleExceptionType::RangeError, "Invalid use of reserved key 'readableType'"sv };
  28. // 4. If transformerDict["writableType"] exists, throw a RangeError exception.
  29. if (transformer_dict.writable_type.has_value())
  30. return WebIDL::SimpleException { WebIDL::SimpleExceptionType::RangeError, "Invalid use of reserved key 'writableType'"sv };
  31. // 5. Let readableHighWaterMark be ? ExtractHighWaterMark(readableStrategy, 0).
  32. auto readable_high_water_mark = TRY(extract_high_water_mark(readable_strategy, 0));
  33. // 6. Let readableSizeAlgorithm be ! ExtractSizeAlgorithm(readableStrategy).
  34. auto readable_size_algorithm = extract_size_algorithm(vm, readable_strategy);
  35. // 7. Let writableHighWaterMark be ? ExtractHighWaterMark(writableStrategy, 1).
  36. auto writable_high_water_mark = TRY(extract_high_water_mark(writable_strategy, 1));
  37. // 8. Let writableSizeAlgorithm be ! ExtractSizeAlgorithm(writableStrategy).
  38. auto writable_size_algorithm = extract_size_algorithm(vm, writable_strategy);
  39. // 9. Let startPromise be a new promise.
  40. auto start_promise = WebIDL::create_promise(realm);
  41. // 10. Perform ! InitializeTransformStream(this, startPromise, writableHighWaterMark, writableSizeAlgorithm,
  42. // readableHighWaterMark, readableSizeAlgorithm).
  43. TRY(initialize_transform_stream(*stream, start_promise, writable_high_water_mark, move(writable_size_algorithm), readable_high_water_mark, move(readable_size_algorithm)));
  44. // 11. Perform ? SetUpTransformStreamDefaultControllerFromTransformer(this, transformer, transformerDict).
  45. TRY(set_up_transform_stream_default_controller_from_transformer(*stream, transformer, transformer_dict));
  46. // 12. If transformerDict["start"] exists, then resolve startPromise with the result of invoking
  47. // transformerDict["start"] with argument list « this.[[controller]] » and callback this value transformer.
  48. if (transformer_dict.start) {
  49. auto result = MUST_OR_THROW_OOM(WebIDL::invoke_callback(*transformer_dict.start, transformer, stream->controller())).release_value();
  50. WebIDL::resolve_promise(realm, start_promise, result);
  51. }
  52. // 13. Otherwise, resolve startPromise with undefined.
  53. else {
  54. WebIDL::resolve_promise(realm, start_promise, JS::js_undefined());
  55. }
  56. return stream;
  57. }
  58. TransformStream::TransformStream(JS::Realm& realm)
  59. : Bindings::PlatformObject(realm)
  60. {
  61. }
  62. TransformStream::~TransformStream() = default;
  63. void TransformStream::initialize(JS::Realm& realm)
  64. {
  65. Base::initialize(realm);
  66. WEB_SET_PROTOTYPE_FOR_INTERFACE(TransformStream);
  67. }
  68. void TransformStream::visit_edges(Cell::Visitor& visitor)
  69. {
  70. Base::visit_edges(visitor);
  71. visitor.visit(m_backpressure_change_promise);
  72. visitor.visit(m_controller);
  73. visitor.visit(m_readable);
  74. visitor.visit(m_writable);
  75. }
  76. }