sse_parse_util.dart 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142
  1. import 'dart:async';
  2. import 'dart:convert';
  3. import 'dart:typed_data';
  4. class SSEParseUtil {
  5. static Stream<Message> parse(Stream<Uint8List> stream) {
  6. return stream.transform(SSETransformer());
  7. }
  8. }
  9. class Message {
  10. final String id;
  11. final String event;
  12. final String data;
  13. final int? retry;
  14. Message(
  15. {required this.id,
  16. required this.event,
  17. required this.data,
  18. required this.retry});
  19. @override
  20. String toString() {
  21. return 'Message{id: $id, event: $event, data: $data, retry: $retry}';
  22. }
  23. }
  24. class SSETransformer extends StreamTransformerBase<Uint8List, Message> {
  25. @override
  26. Stream<Message> bind(Stream<Uint8List> stream) {
  27. return Stream.eventTransformed(
  28. stream.map((uint8List) => List<int>.from(uint8List)),
  29. (sink) => SSESink(sink),
  30. );
  31. }
  32. }
  33. class SSESink implements EventSink<List<int>> {
  34. static final _eventSeparator = utf8.encode("\n\n");
  35. static const _fieldSeparator = "\n";
  36. static const _dataPrefix = "data:";
  37. static const _dataPrefixR = "data: ";
  38. static const _idPrefix = "id:";
  39. static const _idPrefixR = "id: ";
  40. static const _eventPrefix = "event:";
  41. static const _eventPrefixR = "event: ";
  42. static const _retryPrefix = "retry:";
  43. static const _retryPrefixR = "retry: ";
  44. static const _commentPrefix = ":";
  45. static const _commentPrefixR = ": ";
  46. final EventSink<Message> _eventSink;
  47. final List<int> _buffer = [];
  48. SSESink(this._eventSink);
  49. @override
  50. void add(List<int> event) {
  51. _buffer.addAll(event);
  52. final endIndex = _indexOf(_buffer, _eventSeparator);
  53. if (endIndex == -1) {
  54. return;
  55. }
  56. final completedEvent = _buffer.sublist(0, endIndex);
  57. _buffer.removeRange(0, endIndex + _eventSeparator.length);
  58. parseEvent(completedEvent);
  59. }
  60. @override
  61. void addError(Object error, [StackTrace? stackTrace]) {
  62. _eventSink.addError(error, stackTrace);
  63. }
  64. @override
  65. void close() {
  66. _eventSink.close();
  67. }
  68. int _indexOf(List<int> origin, List<int> target) {
  69. for (var i = 0; i < origin.length - target.length; i++) {
  70. var found = true;
  71. for (var j = 0; j < target.length; j++) {
  72. if (origin[i + j] != target[j]) {
  73. found = false;
  74. break;
  75. }
  76. }
  77. if (found) {
  78. return i;
  79. }
  80. }
  81. return -1;
  82. }
  83. void parseEvent(List<int> completedEvent) {
  84. final eventString = utf8.decode(completedEvent);
  85. final fields = eventString.split(_fieldSeparator);
  86. String? id;
  87. String? event;
  88. String data = "";
  89. int? retry;
  90. for (final field in fields) {
  91. final trimmedField = field.trim();
  92. if (trimmedField.isEmpty) {
  93. continue;
  94. }
  95. if (trimmedField.startsWith(_commentPrefix) ||
  96. trimmedField.startsWith(_commentPrefixR)) {
  97. continue;
  98. }
  99. if (trimmedField.startsWith(_retryPrefixR)) {
  100. retry = int.tryParse(trimmedField.substring(_retryPrefixR.length));
  101. } else if (trimmedField.startsWith(_dataPrefixR)) {
  102. data += trimmedField.substring(_dataPrefixR.length);
  103. } else if (trimmedField.startsWith(_eventPrefixR)) {
  104. event = trimmedField.substring(_eventPrefixR.length);
  105. } else if (trimmedField.startsWith(_idPrefixR)) {
  106. id = trimmedField.substring(_idPrefixR.length);
  107. } else if (trimmedField.startsWith(_idPrefix)) {
  108. id = trimmedField.substring(_idPrefix.length);
  109. } else if (trimmedField.startsWith(_eventPrefix)) {
  110. event = trimmedField.substring(_eventPrefix.length);
  111. } else if (trimmedField.startsWith(_dataPrefix)) {
  112. data += trimmedField.substring(_dataPrefix.length);
  113. } else if (trimmedField.startsWith(_retryPrefix)) {
  114. retry = int.tryParse(trimmedField.substring(_retryPrefix.length));
  115. }
  116. }
  117. _eventSink.add(Message(
  118. id: id ?? "", event: event ?? "", data: data, retry: retry));
  119. }
  120. }