Fix a bunch of memory problems in beagle:
[beagle.git] / BeagleClient / Client.cs
blob1cb031ff3ddcd072a8bea070cc48afc14d273b32
1 //
2 // Client.cs
3 //
4 // Copyright (C) 2005 Novell, Inc.
5 //
7 //
8 // Permission is hereby granted, free of charge, to any person obtaining a
9 // copy of this software and associated documentation files (the "Software"),
10 // to deal in the Software without restriction, including without limitation
11 // the rights to use, copy, modify, merge, publish, distribute, sublicense,
12 // and/or sell copies of the Software, and to permit persons to whom the
13 // Software is furnished to do so, subject to the following conditions:
15 // The above copyright notice and this permission notice shall be included in
16 // all copies or substantial portions of the Software.
18 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
19 // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
20 // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
21 // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
22 // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
23 // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
24 // DEALINGS IN THE SOFTWARE.
27 using System;
28 using System.IO;
29 using System.Net.Sockets;
30 using System.Threading;
31 using System.Xml.Serialization;
32 using Mono.Unix;
34 using GLib;
36 using Beagle.Util;
38 namespace Beagle {
41 internal class Client {
43 private class EventThrowingClosure {
45 private Client client;
46 private ResponseMessage response;
48 public EventThrowingClosure (Client client, ResponseMessage response)
50 this.client = client;
51 this.response = response;
54 public bool ThrowEvent ()
56 if (this.client.AsyncResponseEvent != null)
57 this.client.AsyncResponseEvent (this.response);
59 return false;
63 private string socket_name;
65 private UnixClient client;
67 private byte[] network_data = new byte [4096];
68 private MemoryStream buffer_stream = new MemoryStream ();
70 private bool closed = false;
72 public delegate void AsyncResponse (ResponseMessage response);
73 public event AsyncResponse AsyncResponseEvent;
75 public delegate void Closed ();
76 public event Closed ClosedEvent;
78 public Client (string client_name)
80 // use the default socket name when passed null
81 if (client_name == null)
82 client_name = "socket";
84 string storage_dir = PathFinder.GetRemoteStorageDir (false);
85 if (storage_dir == null)
86 throw new System.Net.Sockets.SocketException ();
88 this.socket_name = Path.Combine (storage_dir, client_name);
91 public Client () : this (null)
96 ~Client ()
98 Close ();
101 public void Close ()
103 bool previously_closed = this.closed;
105 // Important to set this before we close the
106 // UnixClient, since that will trigger the
107 // ReadCallback() method, reading 0 bytes off the
108 // wire, and we check this.closed in there.
109 this.closed = true;
111 if (this.client != null)
112 this.client.Close ();
114 if (!previously_closed && this.ClosedEvent != null)
115 this.ClosedEvent ();
118 static XmlSerializer req_serializer = new XmlSerializer (typeof (RequestWrapper), RequestMessage.Types);
120 private void SendRequest (RequestMessage request)
122 this.client = new UnixClient (this.socket_name);
123 NetworkStream stream = this.client.GetStream ();
125 // The socket may be shut down at some point here. It
126 // is the caller's responsibility to handle the error
127 // correctly.
128 #if ENABLE_XML_DUMP
129 MemoryStream mem_stream = new MemoryStream ();
130 XmlFu.SerializeUtf8 (req_serializer, mem_stream, new RequestWrapper (request));
131 mem_stream.Seek (0, SeekOrigin.Begin);
132 StreamReader r = new StreamReader (mem_stream);
133 Logger.Log.Debug ("Sending request:\n{0}\n", r.ReadToEnd ());
134 mem_stream.Seek (0, SeekOrigin.Begin);
135 mem_stream.WriteTo (stream);
136 mem_stream.Close ();
137 #else
138 XmlFu.SerializeUtf8 (req_serializer, stream, new RequestWrapper (request));
139 #endif
140 // Send end of message marker
141 stream.WriteByte (0xff);
142 stream.Flush ();
145 static XmlSerializer resp_serializer = new XmlSerializer (typeof (ResponseWrapper), ResponseMessage.Types);
147 // This function will be called from its own thread
148 private void ReadCallback (IAsyncResult ar)
150 if (this.closed)
151 return;
153 try {
154 NetworkStream stream = this.client.GetStream ();
155 int bytes_read = 0;
157 try {
158 bytes_read = stream.EndRead (ar);
159 } catch (SocketException) {
160 Logger.Log.Debug ("Caught SocketException in ReadCallback");
161 } catch (IOException) {
162 Logger.Log.Debug ("Caught IOException in ReadCallback");
165 // Connection hung up, we're through
166 if (bytes_read == 0) {
167 this.Close ();
168 return;
171 int end_index = -1;
172 int prev_index = 0;
174 do {
175 // 0xff signifies end of message
176 end_index = ArrayFu.IndexOfByte (this.network_data, (byte) 0xff, prev_index);
178 this.buffer_stream.Write (this.network_data, prev_index, (end_index == -1 ? bytes_read : end_index) - prev_index);
180 if (end_index != -1) {
182 MemoryStream deserialize_stream = this.buffer_stream;
183 this.buffer_stream = new MemoryStream ();
185 deserialize_stream.Seek (0, SeekOrigin.Begin);
187 #if ENABLE_XML_DUMP
188 StreamReader r = new StreamReader (deserialize_stream);
189 Logger.Log.Debug ("Received response:\n{0}\n", r.ReadToEnd ());
190 deserialize_stream.Seek (0, SeekOrigin.Begin);
191 #endif
193 ResponseWrapper wrapper;
194 wrapper = (ResponseWrapper) resp_serializer.Deserialize (deserialize_stream);
196 ResponseMessage resp = wrapper.Message;
198 deserialize_stream.Close ();
200 // Run the handler in an idle handler
201 // so that events are thrown in the
202 // main thread instead of this inferior
203 // helper thread.
204 EventThrowingClosure closure = new EventThrowingClosure (this, resp);
205 GLib.Idle.Add (new IdleHandler (closure.ThrowEvent));
207 // Move past the end-of-message marker
208 prev_index = end_index + 1;
210 } while (end_index != -1);
212 // Check to see if we're still connected, and keep
213 // looking for new data if so.
214 if (!this.closed)
215 BeginRead ();
217 } catch (Exception e) {
218 Logger.Log.Error (e, "Got an exception while trying to read data:");
222 private void BeginRead ()
224 NetworkStream stream = this.client.GetStream ();
225 Array.Clear (this.network_data, 0, this.network_data.Length);
226 stream.BeginRead (this.network_data, 0, this.network_data.Length,
227 new AsyncCallback (ReadCallback), null);
230 public void SendAsync (RequestMessage request)
232 Exception ex = null;
234 try {
235 SendRequest (request);
236 } catch (IOException e) {
237 ex = e;
238 } catch (SocketException e) {
239 ex = e;
242 if (ex != null) {
243 ResponseMessage resp = new ErrorResponse (ex);
245 if (this.AsyncResponseEvent != null)
246 this.AsyncResponseEvent (resp);
247 } else
248 BeginRead ();
251 public void SendAsyncBlocking (RequestMessage request)
253 Exception ex = null;
255 try {
256 SendRequest (request);
257 } catch (IOException e) {
258 ex = e;
259 } catch (SocketException e) {
260 ex = e;
263 if (ex != null) {
264 ResponseMessage resp = new ErrorResponse (ex);
266 if (this.AsyncResponseEvent != null)
267 this.AsyncResponseEvent (resp);
268 return;
272 NetworkStream stream = this.client.GetStream ();
273 MemoryStream deserialize_stream = new MemoryStream ();
275 // This buffer is annoyingly small on purpose, to avoid
276 // having to deal with the case of multiple messages
277 // in a single block.
278 byte [] buffer = new byte [32];
280 while (! this.closed) {
282 Array.Clear (buffer, 0, buffer.Length);
284 int bytes_read;
285 bytes_read = stream.Read (buffer, 0, buffer.Length);
286 if (bytes_read == 0)
287 break;
289 int end_index;
290 end_index = ArrayFu.IndexOfByte (buffer, (byte) 0xff);
292 if (end_index == -1) {
293 deserialize_stream.Write (buffer, 0, bytes_read);
294 } else {
295 deserialize_stream.Write (buffer, 0, end_index);
296 deserialize_stream.Seek (0, SeekOrigin.Begin);
298 ResponseMessage resp;
299 try {
300 ResponseWrapper wrapper;
301 wrapper = (ResponseWrapper) resp_serializer.Deserialize (deserialize_stream);
303 resp = wrapper.Message;
304 } catch (Exception e) {
305 resp = new ErrorResponse (e);
308 if (this.AsyncResponseEvent != null)
309 this.AsyncResponseEvent (resp);
311 deserialize_stream.Close ();
312 deserialize_stream = new MemoryStream ();
313 if (bytes_read - end_index - 1 > 0)
314 deserialize_stream.Write (buffer, end_index+1, bytes_read-end_index-1);
320 public ResponseMessage Send (RequestMessage request)
322 if (request.Keepalive)
323 throw new Exception ("A blocking connection on a keepalive request is not allowed");
325 Exception throw_me = null;
327 try {
328 SendRequest (request);
329 } catch (IOException e) {
330 throw_me = e;
331 } catch (SocketException e) {
332 throw_me = e;
335 if (throw_me != null)
336 throw new ResponseMessageException (throw_me);
338 NetworkStream stream = this.client.GetStream ();
339 int bytes_read, end_index = -1;
341 do {
342 bytes_read = stream.Read (this.network_data, 0, 4096);
344 //Logger.Log.Debug ("Read {0} bytes", bytes_read);
346 if (bytes_read > 0) {
347 // 0xff signifies end of message
348 end_index = ArrayFu.IndexOfByte (this.network_data, (byte) 0xff);
350 this.buffer_stream.Write (this.network_data, 0,
351 end_index == -1 ? bytes_read : end_index);
353 } while (bytes_read > 0 && end_index == -1);
355 // It's possible that the server side shut down the
356 // connection before we had a chance to read any data.
357 // If this is the case, throw a rather descriptive
358 // exception.
359 if (this.buffer_stream.Length == 0) {
360 this.buffer_stream.Close ();
361 throw new ResponseMessageException ("Socket was closed before any data could be read");
364 this.buffer_stream.Seek (0, SeekOrigin.Begin);
366 ResponseMessage resp = null;
368 try {
369 ResponseWrapper wrapper;
370 wrapper = (ResponseWrapper) resp_serializer.Deserialize (this.buffer_stream);
372 resp = wrapper.Message;
373 } catch (Exception e) {
374 this.buffer_stream.Seek (0, SeekOrigin.Begin);
375 StreamReader r = new StreamReader (this.buffer_stream);
376 throw_me = new ResponseMessageException (e, "Exception while deserializing response", String.Format ("Message contents: '{0}'", r.ReadToEnd ()));
377 this.buffer_stream.Seek (0, SeekOrigin.Begin);
380 this.buffer_stream.Close ();
382 if (throw_me != null)
383 throw throw_me;
385 return resp;