[System] Clean up some warnings
[mono.git] / mcs / class / System / System.Net / ResponseStream.cs
1 //
2 // System.Net.ResponseStream
3 //
4 // Author:
5 //      Gonzalo Paniagua Javier (gonzalo@novell.com)
6 //
7 // Copyright (c) 2005 Novell, Inc. (http://www.novell.com)
8 //
9 // Permission is hereby granted, free of charge, to any person obtaining
10 // a copy of this software and associated documentation files (the
11 // "Software"), to deal in the Software without restriction, including
12 // without limitation the rights to use, copy, modify, merge, publish,
13 // distribute, sublicense, and/or sell copies of the Software, and to
14 // permit persons to whom the Software is furnished to do so, subject to
15 // the following conditions:
16 // 
17 // The above copyright notice and this permission notice shall be
18 // included in all copies or substantial portions of the Software.
19 // 
20 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
21 // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
22 // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
23 // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
24 // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
25 // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
26 // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
27 //
28
29 #if SECURITY_DEP
30
31 using System.IO;
32 using System.Net.Sockets;
33 using System.Text;
34 using System.Runtime.InteropServices;
35 namespace System.Net {
36         // FIXME: Does this buffer the response until Close?
37         // Update: we send a single packet for the first non-chunked Write
38         // What happens when we set content-length to X and write X-1 bytes then close?
39         // what if we don't set content-length at all?
40         class ResponseStream : Stream
41         {
42                 HttpListenerResponse response;
43                 bool ignore_errors;
44                 bool disposed;
45                 bool trailer_sent;
46                 Stream stream;
47
48                 internal ResponseStream (Stream stream, HttpListenerResponse response, bool ignore_errors)
49                 {
50                         this.response = response;
51                         this.ignore_errors = ignore_errors;
52                         this.stream = stream;
53                 }
54
55                 public override bool CanRead {
56                         get { return false; }
57                 }
58
59                 public override bool CanSeek {
60                         get { return false; }
61                 }
62
63                 public override bool CanWrite {
64                         get { return true; }
65                 }
66
67                 public override long Length {
68                         get { throw new NotSupportedException (); }
69                 }
70
71                 public override long Position {
72                         get { throw new NotSupportedException (); }
73                         set { throw new NotSupportedException (); }
74                 }
75
76
77                 public override void Close ()
78                 {
79                         if (disposed == false) {
80                                 disposed = true;
81                                 byte [] bytes = null;
82                                 MemoryStream ms = GetHeaders (true);
83                                 bool chunked = response.SendChunked;
84                                 if (stream.CanWrite) {
85                                         try {
86                                                 if (ms != null) {
87                                                         long start = ms.Position;
88                                                         if (chunked && !trailer_sent) {
89                                                                 bytes = GetChunkSizeBytes (0, true);
90                                                                 ms.Position = ms.Length;
91                                                                 ms.Write (bytes, 0, bytes.Length);
92                                                         }
93                                                         InternalWrite (ms.GetBuffer (), (int) start, (int) (ms.Length - start));
94                                                         trailer_sent = true;
95                                                 } else if (chunked && !trailer_sent) {
96                                                         bytes = GetChunkSizeBytes (0, true);
97                                                         InternalWrite (bytes, 0, bytes.Length);
98                                                         trailer_sent = true;
99                                                 }
100                                         } catch (IOException) {
101                                                 // Ignore error due to connection reset by peer
102                                         }
103                                 }
104                                 response.Close ();
105                         }
106                 }
107
108                 MemoryStream GetHeaders (bool closing)
109                 {
110                         // SendHeaders works on shared headers
111                         lock (response.headers_lock) {
112                                 if (response.HeadersSent)
113                                         return null;
114                                 MemoryStream ms = new MemoryStream ();
115                                 response.SendHeaders (closing, ms);
116                                 return ms;
117                         }
118                 }
119
120                 public override void Flush ()
121                 {
122                 }
123
124                 static byte [] crlf = new byte [] { 13, 10 };
125                 static byte [] GetChunkSizeBytes (int size, bool final)
126                 {
127                         string str = String.Format ("{0:x}\r\n{1}", size, final ? "\r\n" : "");
128                         return Encoding.ASCII.GetBytes (str);
129                 }
130
131                 internal void InternalWrite (byte [] buffer, int offset, int count)
132                 {
133                         if (ignore_errors) {
134                                 try {
135                                         stream.Write (buffer, offset, count);
136                                 } catch { }
137                         } else {
138                                 stream.Write (buffer, offset, count);
139                         }
140                 }
141
142                 public override void Write (byte [] buffer, int offset, int count)
143                 {
144                         if (disposed)
145                                 throw new ObjectDisposedException (GetType ().ToString ());
146
147                         byte [] bytes = null;
148                         MemoryStream ms = GetHeaders (false);
149                         bool chunked = response.SendChunked;
150                         if (ms != null) {
151                                 long start = ms.Position; // After the possible preamble for the encoding
152                                 ms.Position = ms.Length;
153                                 if (chunked) {
154                                         bytes = GetChunkSizeBytes (count, false);
155                                         ms.Write (bytes, 0, bytes.Length);
156                                 }
157
158                                 int new_count = Math.Min (count, 16384 - (int) ms.Position + (int) start);
159                                 ms.Write (buffer, offset, new_count);
160                                 count -= new_count;
161                                 offset += new_count;
162                                 InternalWrite (ms.GetBuffer (), (int) start, (int) (ms.Length - start));
163                                 ms.SetLength (0);
164                                 ms.Capacity = 0; // 'dispose' the buffer in ms.
165                         } else if (chunked) {
166                                 bytes = GetChunkSizeBytes (count, false);
167                                 InternalWrite (bytes, 0, bytes.Length);
168                         }
169
170                         if (count > 0)
171                                 InternalWrite (buffer, offset, count);
172                         if (chunked)
173                                 InternalWrite (crlf, 0, 2);
174                 }
175
176                 public override IAsyncResult BeginWrite (byte [] buffer, int offset, int count,
177                                                         AsyncCallback cback, object state)
178                 {
179                         if (disposed)
180                                 throw new ObjectDisposedException (GetType ().ToString ());
181
182                         byte [] bytes = null;
183                         MemoryStream ms = GetHeaders (false);
184                         bool chunked = response.SendChunked;
185                         if (ms != null) {
186                                 long start = ms.Position;
187                                 ms.Position = ms.Length;
188                                 if (chunked) {
189                                         bytes = GetChunkSizeBytes (count, false);
190                                         ms.Write (bytes, 0, bytes.Length);
191                                 }
192                                 ms.Write (buffer, offset, count);
193                                 buffer = ms.GetBuffer ();
194                                 offset = (int) start;
195                                 count = (int) (ms.Position - start);
196                         } else if (chunked) {
197                                 bytes = GetChunkSizeBytes (count, false);
198                                 InternalWrite (bytes, 0, bytes.Length);
199                         }
200
201                         return stream.BeginWrite (buffer, offset, count, cback, state);
202                 }
203
204                 public override void EndWrite (IAsyncResult ares)
205                 {
206                         if (disposed)
207                                 throw new ObjectDisposedException (GetType ().ToString ());
208
209                         if (ignore_errors) {
210                                 try {
211                                         stream.EndWrite (ares);
212                                         if (response.SendChunked)
213                                                 stream.Write (crlf, 0, 2);
214                                 } catch { }
215                         } else {
216                                 stream.EndWrite (ares);
217                                 if (response.SendChunked)
218                                         stream.Write (crlf, 0, 2);
219                         }
220                 }
221
222                 public override int Read ([In,Out] byte[] buffer, int offset, int count)
223                 {
224                         throw new NotSupportedException ();
225                 }
226
227                 public override IAsyncResult BeginRead (byte [] buffer, int offset, int count,
228                                                         AsyncCallback cback, object state)
229                 {
230                         throw new NotSupportedException ();
231                 }
232
233                 public override int EndRead (IAsyncResult ares)
234                 {
235                         throw new NotSupportedException ();
236                 }
237
238                 public override long Seek (long offset, SeekOrigin origin)
239                 {
240                         throw new NotSupportedException ();
241                 }
242
243                 public override void SetLength (long value)
244                 {
245                         throw new NotSupportedException ();
246                 }
247         }
248 }
249 #endif
250