TransformStream.cpp 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177
  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/Bindings/TransformStreamPrototype.h>
  8. #include <LibWeb/Streams/AbstractOperations.h>
  9. #include <LibWeb/Streams/TransformStream.h>
  10. #include <LibWeb/Streams/TransformStreamDefaultController.h>
  11. #include <LibWeb/Streams/Transformer.h>
  12. #include <LibWeb/Streams/WritableStream.h>
  13. #include <LibWeb/WebIDL/AbstractOperations.h>
  14. #include <LibWeb/WebIDL/ExceptionOr.h>
  15. namespace Web::Streams {
  16. GC_DEFINE_ALLOCATOR(TransformStream);
  17. // https://streams.spec.whatwg.org/#ts-constructor
  18. WebIDL::ExceptionOr<GC::Ref<TransformStream>> TransformStream::construct_impl(JS::Realm& realm, Optional<GC::Root<JS::Object>> transformer_object, QueuingStrategy const& writable_strategy, QueuingStrategy const& readable_strategy)
  19. {
  20. auto& vm = realm.vm();
  21. auto stream = realm.create<TransformStream>(realm);
  22. // 1. If transformer is missing, set it to null.
  23. auto transformer = transformer_object.has_value() ? JS::Value { transformer_object.value() } : JS::js_null();
  24. // 2. Let transformerDict be transformer, converted to an IDL value of type Transformer.
  25. auto transformer_dict = TRY(Transformer::from_value(vm, transformer));
  26. // 3. If transformerDict["readableType"] exists, throw a RangeError exception.
  27. if (transformer_dict.readable_type.has_value())
  28. return WebIDL::SimpleException { WebIDL::SimpleExceptionType::RangeError, "Invalid use of reserved key 'readableType'"sv };
  29. // 4. If transformerDict["writableType"] exists, throw a RangeError exception.
  30. if (transformer_dict.writable_type.has_value())
  31. return WebIDL::SimpleException { WebIDL::SimpleExceptionType::RangeError, "Invalid use of reserved key 'writableType'"sv };
  32. // 5. Let readableHighWaterMark be ? ExtractHighWaterMark(readableStrategy, 0).
  33. auto readable_high_water_mark = TRY(extract_high_water_mark(readable_strategy, 0));
  34. // 6. Let readableSizeAlgorithm be ! ExtractSizeAlgorithm(readableStrategy).
  35. auto readable_size_algorithm = extract_size_algorithm(vm, readable_strategy);
  36. // 7. Let writableHighWaterMark be ? ExtractHighWaterMark(writableStrategy, 1).
  37. auto writable_high_water_mark = TRY(extract_high_water_mark(writable_strategy, 1));
  38. // 8. Let writableSizeAlgorithm be ! ExtractSizeAlgorithm(writableStrategy).
  39. auto writable_size_algorithm = extract_size_algorithm(vm, writable_strategy);
  40. // 9. Let startPromise be a new promise.
  41. auto start_promise = WebIDL::create_promise(realm);
  42. // 10. Perform ! InitializeTransformStream(this, startPromise, writableHighWaterMark, writableSizeAlgorithm, readableHighWaterMark, readableSizeAlgorithm).
  43. 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. 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 = TRY(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. // https://streams.spec.whatwg.org/#transformstream-set-up
  59. void TransformStream::set_up(GC::Ref<TransformAlgorithm> transform_algorithm, GC::Ptr<FlushAlgorithm> flush_algorithm, GC::Ptr<CancelAlgorithm> cancel_algorithm)
  60. {
  61. auto& realm = this->realm();
  62. // 1. Let writableHighWaterMark be 1.
  63. auto writable_high_water_mark = 1.0;
  64. // 2. Let writableSizeAlgorithm be an algorithm that returns 1.
  65. auto writable_size_algorithm = GC::create_function(realm.heap(), [](JS::Value) {
  66. return JS::normal_completion(JS::Value { 1 });
  67. });
  68. // 3. Let readableHighWaterMark be 0.
  69. auto readable_high_water_mark = 0.0;
  70. // 4. Let readableSizeAlgorithm be an algorithm that returns 1.
  71. auto readable_size_algorithm = GC::create_function(realm.heap(), [](JS::Value) {
  72. return JS::normal_completion(JS::Value { 1 });
  73. });
  74. // 5. Let transformAlgorithmWrapper be an algorithm that runs these steps given a value chunk:
  75. auto transform_algorithm_wrapper = GC::create_function(realm.heap(), [&realm, transform_algorithm](JS::Value chunk) -> GC::Ref<WebIDL::Promise> {
  76. // 1. Let result be the result of running transformAlgorithm given chunk. If this throws an exception e, return a promise rejected with e.
  77. GC::Ptr<JS::PromiseCapability> result = nullptr;
  78. result = transform_algorithm->function()(chunk);
  79. // 2. If result is a Promise, then return result.
  80. if (result)
  81. return GC::Ref { *result };
  82. // 3. Return a promise resolved with undefined.
  83. return WebIDL::create_resolved_promise(realm, JS::js_undefined());
  84. });
  85. // 6. Let flushAlgorithmWrapper be an algorithm that runs these steps:
  86. auto flush_algorithm_wrapper = GC::create_function(realm.heap(), [&realm, flush_algorithm]() -> GC::Ref<WebIDL::Promise> {
  87. // 1. Let result be the result of running flushAlgorithm, if flushAlgorithm was given, or null otherwise. If this throws an exception e, return a promise rejected with e.
  88. GC::Ptr<JS::PromiseCapability> result = nullptr;
  89. if (flush_algorithm)
  90. result = flush_algorithm->function()();
  91. // 2. If result is a Promise, then return result.
  92. if (result)
  93. return GC::Ref { *result };
  94. // 3. Return a promise resolved with undefined.
  95. return WebIDL::create_resolved_promise(realm, JS::js_undefined());
  96. });
  97. // 7. Let cancelAlgorithmWrapper be an algorithm that runs these steps given a value reason:
  98. auto cancel_algorithm_wrapper = GC::create_function(realm.heap(), [&realm, cancel_algorithm](JS::Value reason) -> GC::Ref<WebIDL::Promise> {
  99. // 1. Let result be the result of running cancelAlgorithm given reason, if cancelAlgorithm was given, or null otherwise. If this throws an exception e, return a promise rejected with e.
  100. GC::Ptr<JS::PromiseCapability> result = nullptr;
  101. if (cancel_algorithm)
  102. result = cancel_algorithm->function()(reason);
  103. // 2. If result is a Promise, then return result.
  104. if (result)
  105. return GC::Ref { *result };
  106. // 3. Return a promise resolved with undefined.
  107. return WebIDL::create_resolved_promise(realm, JS::js_undefined());
  108. });
  109. // 8. Let startPromise be a promise resolved with undefined.
  110. auto start_promise = WebIDL::create_resolved_promise(realm, JS::js_undefined());
  111. // 9. Perform ! InitializeTransformStream(stream, startPromise, writableHighWaterMark, writableSizeAlgorithm, readableHighWaterMark, readableSizeAlgorithm).
  112. initialize_transform_stream(*this, start_promise, writable_high_water_mark, writable_size_algorithm, readable_high_water_mark, readable_size_algorithm);
  113. // 10. Let controller be a new TransformStreamDefaultController.
  114. auto controller = realm.create<TransformStreamDefaultController>(realm);
  115. // 11. Perform ! SetUpTransformStreamDefaultController(stream, controller, transformAlgorithmWrapper, flushAlgorithmWrapper, cancelAlgorithmWrapper).
  116. set_up_transform_stream_default_controller(*this, controller, transform_algorithm_wrapper, flush_algorithm_wrapper, cancel_algorithm_wrapper);
  117. }
  118. TransformStream::TransformStream(JS::Realm& realm)
  119. : Bindings::PlatformObject(realm)
  120. {
  121. }
  122. TransformStream::~TransformStream() = default;
  123. void TransformStream::initialize(JS::Realm& realm)
  124. {
  125. Base::initialize(realm);
  126. WEB_SET_PROTOTYPE_FOR_INTERFACE(TransformStream);
  127. }
  128. void TransformStream::visit_edges(Cell::Visitor& visitor)
  129. {
  130. Base::visit_edges(visitor);
  131. visitor.visit(m_backpressure_change_promise);
  132. visitor.visit(m_controller);
  133. visitor.visit(m_readable);
  134. visitor.visit(m_writable);
  135. }
  136. }