Арифметическая операция переполнена при чтении сокета в C # - PullRequest
2 голосов
/ 07 ноября 2011

Поэтому я решил начать изучать его, дав себе интересное задание. Как веб-разработчик, я хотел иметь свой собственный сервер WebSocket. Итак, я написал это, но он принимает только первый запрос. После этого происходит арифметическая операция onverflow.
Вот код для вас, чтобы увидеть, что я делаю неправильно: S У меня действительно нет идей.

using System;
using System.Collections.Generic;
using System.Text;
using System.Net;
using System.Net.Sockets;
using System.Threading;
using System.Text.RegularExpressions;
using System.Security.Cryptography;

using WebSocket.Utils;

namespace WebSocket
{
    class SocketReader
    {
        public EndPoint ipAddr { get; set; }
        private Socket userSocket;
        private byte[] buffer;
        private string SOCKET_GUID = "258EAFA5-E914-47DA-95CA-C5AB0DC85B11";

        public SocketReader(Socket socket)
        {
            userSocket = socket;
            ipAddr = socket.RemoteEndPoint;
            Read();
        }

        private void Read()
        {
            //Read packet size
            buffer = new byte[2];
            userSocket.BeginReceive(buffer, 0, 2, SocketFlags.None, ReadCallbackStatic, null);
        }

        private void ReadCallbackStatic(IAsyncResult ar)
        {
            try
            {
                if (userSocket.EndReceive(ar) >= 1)
                {
                    int bufferSize = BitConverter.ToInt16(buffer, 0);
                    buffer = new byte[bufferSize - 2];
                    //Read Packet
                    userSocket.BeginReceive(buffer, 0, buffer.Length, SocketFlags.None, ReadCallback, null);
                }
            }
            catch (Exception se)
            {
                Console.WriteLine("Something blew on ReadCallbackStatic");
                Console.WriteLine(se.Message);
                Console.WriteLine(se.StackTrace);
                Disconnect();
            }
        }

        private void ReadCallback(IAsyncResult ar)
        {
            try
            {
                //Copy the buffer so we can receive the next packet ASAP
                byte[] buff = new byte[buffer.Length];
                Array.Copy(buffer, buff, buffer.Length);
                Read();

                string handshakeStr = System.Text.Encoding.UTF8.GetString(buff);
                string[] list = Regex.Split(handshakeStr, "\r\n");

                //Sec-WebSocket-Key: S5o6fCVLRMJhdXTF3H9w3Q==
                //Sec-WebSocket-Version: 8

                string key = "";
                string clientProtocol = "0";

                foreach (string str in list)
                {
                    if (String.IsNullOrEmpty(str)) { continue; }
                    if (str.Length > 20 && str.Substring(0, 19) == "Sec-WebSocket-Key: ")
                    {
                        key = str.Substring(19);
                        continue;
                    }

                    if (str.Length > 20 && str.Substring(0, 23) == "Sec-WebSocket-Version: ")
                    {
                        clientProtocol = str.Substring(23);
                        continue;
                    }
                }

                if (String.IsNullOrEmpty(key))
                {
                    Disconnect();
                }

                SHA1 shaEnc = new SHA1CryptoServiceProvider();

                byte[] byteString = ASCIIEncoding.ASCII.GetBytes(key + SOCKET_GUID);
                byte[] hash = shaEnc.ComputeHash(byteString, 0, byteString.Length);
                string acceptKey = Convert.ToBase64String(hash);

                List<string> headers = new List<string>();
                headers.Add("HTTP/1.1 101 Switching Protocols");
                headers.Add("Upgrade: websocket");
                headers.Add("Connection: Upgrade");
                headers.Add("Sec-WebSocket-Accept: " + acceptKey);

                foreach (string header in headers)
                {
                    SendString(header + "\r\n");
                }

                Console.WriteLine(acceptKey);

                SendString("\r\n");

            }
            catch (SocketException se)
            {
                Console.WriteLine("Something blew on ReadCallback");
                Console.WriteLine(se.Message);
                Disconnect();
            }
        }

        private void SendString(string str)
        {
            userSocket.Send(Encoding.UTF8.GetBytes(str));
        }

        private void Disconnect()
        {
            userSocket.Disconnect(false);
            Console.WriteLine("Client with ip {0} Disconnected", ipAddr);
        }
    }
}

Это сокращенная версия моего класса, но проблема, которая мне мешает, появляется в "ReadCallbackStatic" в этой строке:

buffer = new byte[bufferSize - 2];

Я действительно не знаю, что я делаю неправильно: S.
Дело в том, что ... я на самом деле рукопожатие правильно, но затем, когда я отправил некоторую информацию от клиента на мой сервер, это исключение выдается

Я сделал некоторую отладку, и похоже, что переменная буфера каким-то образом становится отрицательным числом O.O

Ответы [ 2 ]

2 голосов
/ 07 ноября 2011

Я думаю, что вы взорвали, когда MSB установлен, то есть, возможно, клиент отправляет число> = 32768 и <65536, которое (через ваш <code>ToInt16) становится большим отрицательным 16-разрядным число (или, альтернативно, выбрасывая проблему арифметического переполнения). Я бы попробовал использовать ToUInt16 вместо.

Если честно, вам действительно не нужно BitConverter здесь; в зависимости от порядка байтов это либо:

int bufferSize = (buffer[0] << 8) | buffer[1];

или

int bufferSize = buffer[0] | (buffer[1] << 8);
0 голосов
/ 07 ноября 2011

Я считаю, что проблема может быть в том случае, если буфер содержит нулевые числа, поэтому:

  int bufferSize = BitConverter.ToInt16(buffer, 0); // 0
  buffer = new byte[bufferSize - 2];  // 0 - 2 == -2

, что приводит к исключению переполнения при выполнении new byte[-2]

Я не знаюлогика вашего кода, но я считаю, что вы должны выделить новый буфер, учитывая buffer.Length значение

...