All Downloads are FREE. Search and download functionalities are using the official Maven repository.

ratpack.sse.internal.ServerSentEventEncoder Maven / Gradle / Ivy

/*
 * Copyright 2014 the original author or authors.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *    http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package ratpack.sse.internal;

import io.netty.buffer.ByteBuf;
import io.netty.buffer.ByteBufAllocator;
import io.netty.buffer.Unpooled;
import ratpack.sse.ServerSentEvent;

import java.nio.charset.StandardCharsets;

import static io.netty.util.CharsetUtil.UTF_8;

public class ServerSentEventEncoder {

  public static final ServerSentEventEncoder INSTANCE = new ServerSentEventEncoder();

  private static final byte[] EVENT_TYPE_PREFIX = "event: ".getBytes(UTF_8);
  private static final byte[] EVENT_DATA_PREFIX = "data: ".getBytes(UTF_8);
  private static final byte[] EVENT_ID_PREFIX = "id: ".getBytes(UTF_8);
  private static final byte[] COMMENT_PREFIX = ": ".getBytes(UTF_8);

  private static final byte NEWLINE = '\n';

  public ByteBuf encode(ServerSentEvent event, ByteBufAllocator bufferAllocator) throws Exception {
    String eventId = event.getId();
    String eventType = event.getEvent();
    String eventData = event.getData();
    String comment = event.getComment();

    int initialCapacity = 0;
    if (eventId != null) {
      initialCapacity += EVENT_ID_PREFIX.length + eventId.length() + 1;
    }
    if (eventType != null) {
      initialCapacity += EVENT_TYPE_PREFIX.length + eventType.length() + 1;
    }
    if (eventData != null) {
      initialCapacity += EVENT_DATA_PREFIX.length + eventData.length() + 1;
    }
    if (comment != null) {
      initialCapacity += COMMENT_PREFIX.length + comment.length() + 1;
    }
    if (initialCapacity == 0) {
      return Unpooled.EMPTY_BUFFER;
    }

    ByteBuf buffer = bufferAllocator.buffer(initialCapacity + 4096);

    writeMultiline(buffer, COMMENT_PREFIX, comment);

    if (eventId != null) {
      buffer.writeBytes(EVENT_ID_PREFIX);
      buffer.writeCharSequence(eventId, StandardCharsets.UTF_8);
      buffer.writeByte(NEWLINE);
    }

    if (eventType != null) {
      buffer.writeBytes(EVENT_TYPE_PREFIX);
      buffer.writeCharSequence(eventType, StandardCharsets.UTF_8);
      buffer.writeByte(NEWLINE);
    }

    writeMultiline(buffer, EVENT_DATA_PREFIX, eventData);

    return buffer.writeByte(NEWLINE);
  }

  private void writeMultiline(ByteBuf buffer, byte[] prefix, String value) {
    int from = 0;
    if (value != null) {
      int length = value.length();
      buffer.writeBytes(prefix);
      while (from < length) {
        int to = value.indexOf('\n', from);
        if (to == -1) {
          buffer.writeCharSequence(value.substring(from), StandardCharsets.UTF_8);
          break;
        } else {
          buffer.writeCharSequence(value.substring(from, to), StandardCharsets.UTF_8);
          buffer.writeByte(NEWLINE);
          buffer.writeBytes(prefix);
          from = to + 1;
        }
      }
      buffer.writeByte(NEWLINE);
    }
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy