private async void _CommentSessionWebSocket_MessageReceived(MessageWebSocket sender, MessageWebSocketMessageReceivedEventArgs args) { string recievedText = string.Empty; using (var releaser = await _CommentSessionLock.LockAsync()) { using (var reader = new StreamReader(args.GetDataStream().AsStreamForRead())) { recievedText = reader.ReadToEnd(); Debug.WriteLine($"<CommentSession Message> {args.MessageType}: {recievedText}"); } var jsonObject = (JObject)Newtonsoft.Json.JsonConvert.DeserializeObject(recievedText); if (jsonObject.TryGetValue("chat", out var chatJson)) { var chat = chatJson.ToObject <Mntone.Nico2.Videos.Comment.NMSG_Chat>(); var liveChatData = new LiveChatData() { Thread = chat.Thread, No = chat.No, Content = chat.Content, Date = chat.Date, DateUsec = chat.DateUsec, IsAnonymity = chat.Anonymity == 1, __Premium = chat.Premium, IsYourPost = chat.Yourpost == 1, Mail = chat.Mail, Score = chat.Score, UserId = chat.UserId, Vpos = chat.Vpos }; CommentRecieved?.Invoke(this, new CommentRecievedEventArgs() { Chat = liveChatData }); } else if (jsonObject.TryGetValue("chat_result", out var chatResultJson)) { var chatResult = chatResultJson.ToObject <Mntone.Nico2.Videos.Comment.Chat_result>(); CommentPosted?.Invoke(this, new CommentPostedEventArgs() { Thread = chatResult.Thread, ChatResult = chatResult.Status, No = chatResult.No }); } else if (jsonObject.TryGetValue("thread", out var threadJson)) { var thread = threadJson.ToObject <Mntone.Nico2.Videos.Comment.NGMS_Thread_Response>(); _Thread = thread; // // コメントの受信開始を受け取ってからハートビートを開始する // StartHeartbeatTimer(); Connected?.Invoke(this, new CommentServerConnectedEventArgs() { LastRes = thread.LastRes, Resultcode = thread.Resultcode, Revision = thread.Revision, ServerTime = thread.ServerTime, Thread = thread.Thread, Ticket = thread.Ticket, }); } else if (jsonObject.TryGetValue("ping", out var pingJson)) { var ping = pingJson.ToObject <Mntone.Nico2.Videos.Comment.Ping>(); // {"ping":{"content":"rs:1"}} // { "ping":{ "content":"ps:5"} } // {"chat_result":{...}} // {"chat":{...}} // { "ping":{ "content":"pf:5"} } // { "ping":{ "content":"rf:1"} } if (ping.Content.StartsWith("pf")) { _MessageSendCount += 1; } } else { Debug.WriteLine("Not Support"); } } }
private void ParseChatXml(string xml) { var xmlDoc = XDocument.Parse(xml); var xmlRoot = xmlDoc.Root; var elementName = xmlRoot.Name.LocalName; Debug.Write(elementName); Debug.Write(" -> "); if (elementName == "thread") { Debug.Write("connect success"); IsConnected = true; // <thread ticket="{チケット}" server_time="{サーバー時刻}" last_res="{送信される過去のコメント数?(NECOで使用してないので不明)}"> var serverTimeText = xmlRoot.Attribute(XName.Get("server_time")).Value; long serverTime; if (long.TryParse(serverTimeText, out serverTime)) { _ServerTime = serverTime; } var ticketText = xmlRoot.Attribute(XName.Get("ticket")).Value; _Ticket = ticketText; Connected?.Invoke(this, new CommentServerConnectedEventArgs() { Ticket = _Ticket, ServerTime = (int)_ServerTime, Thread = this.ThreadId }); } else if (elementName == "chat_result") { // <chat_result status="{コメント投稿要求の返答}" /> var result = xmlRoot.Attribute(XName.Get("status")).Value; /* * 0 = 投稿に成功した * 1 = 投稿に失敗した(短時間に同じ内容のコメントを投稿しようとした、パラメータが間違っている、他) * 4 = 投稿に失敗した(ごく短時間にコメントを連投しようとした、パラメータが間違っている、他) */ Debug.Write(result); Debug.Write(result == "0" ? " (success)" : " (failed)"); CommentPosted?.Invoke(this, new CommentPostedEventArgs() { ChatResult = (ChatResult)int.Parse(result), }); _LastPostChat = null; } else if (elementName == "chat") { // _LiveComments.Add(chat); // <chat anonymity="{184か}" no="{コメントの番号}" date="{コメントが投稿されたリアル時間?}" mail="{コマンド}" premium="{プレミアムID}" thread="{スレッドID}" user_id="{ユーザーID}" vpos="{コメントが投稿された生放送の時間}" score="{NGスコア}">{コメント}</chat>\0 try { var chatSerializer = new XmlSerializer(typeof(Chat)); using (var readerStream = new StringReader(xml)) { var chat = chatSerializer.Deserialize(readerStream) as Chat; if (chat != null) { Debug.Write(chat.Text); OperationCommnad officialCommand = null; string[] officialCommandArguments = null; if (ChcekOfficialOperationComment(chat, out officialCommand, out officialCommandArguments)) { var args = new NicoLiveOperationCommandEventArgs() { CommandType = officialCommand.CommandType, Arguments = officialCommandArguments, Chat = chat }; // OperationCommandRecieved?.Invoke(this, args); } else { var liveChatData = new LiveChatData() { No = (int)chat.GetCommentNo(), Thread = chat.Thread, IsAnonymity = chat.GetAnonymity(), Content = chat.Text, Mail = chat.Mail, UserId = chat.User_id, Vpos = (int)chat.GetVpos(), Date = int.Parse(chat.Date), }; CommentRecieved?.Invoke(this, new CommentRecievedEventArgs() { Chat = liveChatData }); } } } } catch { } } else { Debug.WriteLine($"not supproted"); Debug.Write(" -> "); Debug.Write(xml); } }
private void ParseLiveCommentServerResponse(string recievedString) { if (string.IsNullOrWhiteSpace(recievedString)) { Debug.Write($"IGNORE"); return; } if (!recievedString.StartsWith("<") || !recievedString.EndsWith(">")) { // Note: 寄り厳密にXMLフォーマットチェックをやるなら // <>の数が同数であることをチェックする Debug.Write($"illigal format, required XML"); Debug.Write($" -> "); Debug.Write(recievedString); return; } var xmlDoc = XDocument.Parse(recievedString); var xmlRoot = xmlDoc.Root; var elementName = xmlRoot.Name.LocalName; Debug.Write(elementName); Debug.Write(" -> "); if (elementName == "thread") { Debug.Write("connect success"); IsCommentServerConnected = true; // <thread ticket="{チケット}" server_time="{サーバー時刻}" last_res="{送信される過去のコメント数?(NECOで使用してないので不明)}"> var serverTimeText = xmlRoot.Attribute(XName.Get("server_time")).Value; long serverTime; if (long.TryParse(serverTimeText, out serverTime)) { _ServerTime = serverTime; } var ticketText = xmlRoot.Attribute(XName.Get("ticket")).Value; _Ticket = ticketText; CommentServerConnected?.Invoke(); } else if (elementName == "chat_result") { // <chat_result status="{コメント投稿要求の返答}" /> var result = xmlRoot.Attribute(XName.Get("status")).Value; /* * 0 = 投稿に成功した * 1 = 投稿に失敗した(短時間に同じ内容のコメントを投稿しようとした、パラメータが間違っている、他) * 4 = 投稿に失敗した(ごく短時間にコメントを連投しようとした、パラメータが間違っている、他) */ Debug.Write(result); Debug.Write(result == "0" ? " (success)" : " (failed)"); CommentPosted?.Invoke(result == "0", _LastPostChat); _LastPostChat = null; } else if (elementName == "chat") { // _LiveComments.Add(chat); // <chat anonymity="{184か}" no="{コメントの番号}" date="{コメントが投稿されたリアル時間?}" mail="{コマンド}" premium="{プレミアムID}" thread="{スレッドID}" user_id="{ユーザーID}" vpos="{コメントが投稿された生放送の時間}" score="{NGスコア}">{コメント}</chat>\0 try { var chatSerializer = new XmlSerializer(typeof(Chat)); using (var readerStream = new StringReader(recievedString)) { var chat = chatSerializer.Deserialize(readerStream) as Chat; if (chat != null) { Debug.Write(chat.Text); OperationCommnad officialCommand = null; string[] officialCommandArguments = null; if (ChcekOfficialOperationComment(chat, out officialCommand, out officialCommandArguments)) { var args = new NicoLiveOperationCommandEventArgs() { CommandType = officialCommand.CommandType, Arguments = officialCommandArguments, Chat = chat }; OperationCommandRecieved?.Invoke(this, args); } else { CommentRecieved?.Invoke(chat); } } } } catch { } } else { Debug.WriteLine($"not supproted"); Debug.Write(" -> "); Debug.Write(recievedString); } }