javax.websocketclient : how to send large binary data from clientendpoint to serverendpoint

标签 java websocket jetty java-websocket jsr356

我正在尝试使用 jetty 构建服务器客户端应用程序。我已经设置了一个jetty服务器并配置了websockets。在客户端和服务器之间发送文本消息工作正常。但是如何从客户端端点发送二进制数据作为输入流。我找不到任何有关 websocket 客户端的片段。以下是我尝试过的

服务器端点:

   @OnMessage
   public void handleBinaryMessage(InputStream input, Session session) {

       logger.info("onMessage::inputstream");

       try {

        byte[] buffer = new byte[2048];
        try (OutputStream output = session.getBasicRemote().getSendStream())
        {
            int read;
            while ((read = input.read(buffer)) >= 0)
                output.write(buffer, 0, read);
        }

    } catch (IOException e) {
      e.printStackTrace();
    }

客户端端点:

@OnOpen 
public void onOpen(Session s) throws IOException {
  logger.info("Client Connected ... " + s.getId());
  this.session=s;

  session.getBasicRemote().sendText("Ping from client");

  // size of the file 200~500MB
  File source= new File("/tmp/Setup.exe");

  try(InputStream input = new FileInputStream(source)) {


              session.getAsyncRemote().sendObject(input);


            }

}        

感谢任何帮助

编辑:

我修改了 clientendpoint 和 serverendpoint。尝试将数据作为 block 发送,但 zip 文件是部分的,有时甚至比源文件小得多。

源大小:1.5GB 使用流从缓冲区写入数据后:20kb

@ClientEndpoint

   private static void sendFileToRemote(File source) throws FileNotFoundException, IOException {
        // TODO Auto-generated method stub
            Session session=null;
            final WsClient wc = new WsClient("ws://localhost:2714/events/","test");

            session=wc.getSession();

             try (
                        InputStream inputStream = new FileInputStream(source);

                    ) {


                    byte[] chunk = new byte[102400];
                    int chunkLen = 0;
                    while ((chunkLen = inputStream.read(chunk)) != -1) {

                        session.getAsyncRemote().sendBinary(ByteBuffer.wrap(chunk, 0, chunkLen));
                    }

                    } catch (IOException ex) {
                        ex.printStackTrace();
                    }

             }

@serverendpoint

@ServerEndpoint("/events/")
public class EventListener {
    static Logger logger = Logger.getLogger(Initservice.class.getName());
   private OutputStream os=null; 
@OnOpen
    public void Init(Session session) throws FileNotFoundException {

        this.user_session = session;
        logger.info("onOpen:: server" + session.getId());     

        this.os = new FileOutputStream(new File("/tmp/silicon_test.zip"));
        logger.info("instantiate zip files");

    }

@OnMessage
    public void onMessage(Session session, ByteBuffer byteBuffer) throws IOException {
        try {

          os.write(byteBuffer.get());

        } catch(Exception e) {
            close();
            logger.log(Level.SEVERE,"Exception occured in onMessage :: ", e);
            throw e;
        }
    }
}

最佳答案

ServerEndpoint 中的代码看起来应该可以正常工作,但是在 ClientEndpoint 中,您仅将文本数据发送到 ServerEndpoint,并且只能由配置为接收文本消息的服务器 onMessage 方法读取。

您应该使用方法 session.getRemoteEndpoint().sendBinary(...),而不是使用 session.getRemoteEndpoint().sendText(...)。这将以二进制帧而不是文本帧的形式发送数据,您将能够在服务器的 handleBinaryMessage 方法中接收它。

对于 session.getAsyncRemote().sendObject(input),要实现此功能,您还需要提供 Encoder.BinaryEncoder .BinaryStream 以便将对象作为二进制数据发送。

编辑:

WebSocket 是一种基于消息的协议(protocol),您通过多个 WebSocket 消息从文件发送数据。您可以使用 session.getBasicRemote().sendBinary(ByteBuffer, boolean) 发送部分消息并在同一消息中发送所有数据。

或者您可以尝试类似这样的代码,这可能会更简单。

try (InputStream inputStream = new FileInputStream(source))
{
    try (OutputStream sendStream = session.getBasicRemote().getSendStream())
    {
        inputStream.transferTo(sendStream);
    }
}

关于javax.websocketclient : how to send large binary data from clientendpoint to serverendpoint,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/61549428/

相关文章:

java - 如何更新文件中包含的序列化 HashMap?

c++ - 如何将超过 65536 字节的二进制数据编码为 C++ 上的 websocket 帧

java - 1 如何捕获使用 Java 代码运行的可执行文件抛出的异常?

java - 从java中的字符串中提取值

http - 配置 Nginx 和 Socket.IO

java - Jetty 日志中的 SSL_NULL_WITH_NULL_NULL 密码套件

java - jetty/context/text.xml 在哪里?

tomcat - 用于小型 Linode 的 Jetty 或 Tomcat

java - 在 MYSQL 中使用 liquibase 删除外键(如果存在)

java - 如何在Linux上部署java vertx服务器?