OLD | NEW |
1 // Copyright (c) 2010 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2010 The Chromium Authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #ifndef REMOTING_PROTOCOL_MESSAGE_READER_H_ | 5 #ifndef REMOTING_PROTOCOL_MESSAGE_READER_H_ |
6 #define REMOTING_PROTOCOL_MESSAGE_READER_H_ | 6 #define REMOTING_PROTOCOL_MESSAGE_READER_H_ |
7 | 7 |
8 #include "base/callback.h" | 8 #include "base/callback.h" |
9 #include "base/ref_counted.h" | 9 #include "base/ref_counted.h" |
10 #include "base/scoped_ptr.h" | 10 #include "base/scoped_ptr.h" |
11 #include "base/task.h" | 11 #include "base/task.h" |
12 #include "net/base/completion_callback.h" | 12 #include "net/base/completion_callback.h" |
13 #include "net/base/io_buffer.h" | 13 #include "remoting/base/compound_buffer.h" |
14 #include "remoting/protocol/message_decoder.h" | 14 #include "remoting/protocol/message_decoder.h" |
15 | 15 |
16 namespace net { | 16 namespace net { |
| 17 class IOBuffer; |
17 class Socket; | 18 class Socket; |
18 } // namespace net | 19 } // namespace net |
19 | 20 |
20 namespace remoting { | 21 namespace remoting { |
21 namespace protocol { | 22 namespace protocol { |
22 | 23 |
23 class MessageReader; | 24 // MessageReader reads data from the socket asynchronously and calls |
24 | 25 // callback for each message it receives |
25 namespace internal { | |
26 | |
27 template <class T> | |
28 class MessageReaderPrivate { | |
29 private: | |
30 friend class remoting::protocol::MessageReader; | |
31 | |
32 typedef typename Callback1<T*>::Type MessageReceivedCallback; | |
33 | |
34 MessageReaderPrivate(MessageReceivedCallback* callback) | |
35 : message_received_callback_(callback) { | |
36 } | |
37 | |
38 ~MessageReaderPrivate() { } | |
39 | |
40 void OnDataReceived(net::IOBuffer* buffer, int data_size) { | |
41 typedef typename std::list<T*>::iterator MessageListIterator; | |
42 | |
43 std::list<T*> message_list; | |
44 message_decoder_.ParseMessages(buffer, data_size, &message_list); | |
45 for (MessageListIterator it = message_list.begin(); | |
46 it != message_list.end(); ++it) { | |
47 message_received_callback_->Run(*it); | |
48 } | |
49 } | |
50 | |
51 void Destroy() { | |
52 delete this; | |
53 } | |
54 | |
55 // Message decoder is used to decode bytes into protobuf message. | |
56 MessageDecoder message_decoder_; | |
57 | |
58 // Callback is called when a message is received. | |
59 scoped_ptr<MessageReceivedCallback> message_received_callback_; | |
60 }; | |
61 | |
62 } // namespace internal | |
63 | |
64 // MessageReader reads data from the socket asynchronously and uses | |
65 // MessageReaderPrivate to decode the data received. | |
66 class MessageReader { | 26 class MessageReader { |
67 public: | 27 public: |
| 28 typedef Callback1<CompoundBuffer*>::Type MessageReceivedCallback; |
| 29 |
68 MessageReader(); | 30 MessageReader(); |
69 virtual ~MessageReader(); | 31 virtual ~MessageReader(); |
70 | 32 |
71 // Stops reading. Must be called on the same thread as Init(). | |
72 void Close(); | |
73 | |
74 // Initialize the MessageReader with a socket. If a message is received | 33 // Initialize the MessageReader with a socket. If a message is received |
75 // |callback| is called. | 34 // |callback| is called. |
76 template <class T> | 35 void Init(net::Socket* socket, MessageReceivedCallback* callback); |
77 void Init(net::Socket* socket, typename Callback1<T*>::Type* callback) { | |
78 internal::MessageReaderPrivate<T>* reader = | |
79 new internal::MessageReaderPrivate<T>(callback); | |
80 data_received_callback_.reset( | |
81 ::NewCallback( | |
82 reader, &internal::MessageReaderPrivate<T>::OnDataReceived)); | |
83 destruction_callback_.reset( | |
84 ::NewCallback(reader, &internal::MessageReaderPrivate<T>::Destroy)); | |
85 Init(socket); | |
86 } | |
87 | 36 |
88 private: | 37 private: |
89 void Init(net::Socket* socket); | |
90 | |
91 void DoRead(); | 38 void DoRead(); |
92 void OnRead(int result); | 39 void OnRead(int result); |
93 void HandleReadResult(int result); | 40 void HandleReadResult(int result); |
| 41 void OnDataReceived(net::IOBuffer* data, int data_size); |
94 | 42 |
95 net::Socket* socket_; | 43 net::Socket* socket_; |
96 | 44 |
97 bool closed_; | 45 bool closed_; |
98 scoped_refptr<net::IOBuffer> read_buffer_; | 46 scoped_refptr<net::IOBuffer> read_buffer_; |
99 net::CompletionCallbackImpl<MessageReader> read_callback_; | 47 net::CompletionCallbackImpl<MessageReader> read_callback_; |
100 | 48 |
101 scoped_ptr<Callback2<net::IOBuffer*, int>::Type> data_received_callback_; | 49 MessageDecoder message_decoder_; |
102 scoped_ptr<Callback0::Type> destruction_callback_; | 50 |
| 51 // Callback is called when a message is received. |
| 52 scoped_ptr<MessageReceivedCallback> message_received_callback_; |
| 53 }; |
| 54 |
| 55 template <class T> |
| 56 class ProtobufMessageReader { |
| 57 public: |
| 58 typedef typename Callback1<T*>::Type MessageReceivedCallback; |
| 59 |
| 60 ProtobufMessageReader() { }; |
| 61 ~ProtobufMessageReader() { }; |
| 62 |
| 63 void Init(net::Socket* socket, MessageReceivedCallback* callback) { |
| 64 message_received_callback_.reset(callback); |
| 65 message_reader_.Init( |
| 66 socket, NewCallback(this, &ProtobufMessageReader<T>::OnNewData)); |
| 67 } |
| 68 |
| 69 private: |
| 70 void OnNewData(CompoundBuffer* buffer) { |
| 71 T* message = new T(); |
| 72 CompoundBufferInputStream stream(buffer); |
| 73 bool ret = message->ParseFromZeroCopyStream(&stream); |
| 74 if (!ret) { |
| 75 delete message; |
| 76 } else { |
| 77 message_received_callback_->Run(message); |
| 78 } |
| 79 } |
| 80 |
| 81 MessageReader message_reader_; |
| 82 scoped_ptr<MessageReceivedCallback> message_received_callback_; |
103 }; | 83 }; |
104 | 84 |
105 } // namespace protocol | 85 } // namespace protocol |
106 } // namespace remoting | 86 } // namespace remoting |
107 | 87 |
108 #endif // REMOTING_PROTOCOL_MESSAGE_READER_H_ | 88 #endif // REMOTING_PROTOCOL_MESSAGE_READER_H_ |
OLD | NEW |