/*
Copyright (C) 2004 Christopher E. Miller
This software is provided 'as-is', without any express or implied
warranty. In no event will the authors be held liable for any damages
arising from the use of this software.
Permission is granted to anyone to use this software for any purpose,
including commercial applications, and to alter it and redistribute it
freely, subject to the following restrictions:
1. The origin of this software must not be misrepresented; you must not
claim that you wrote the original software. If you use this software
in a product, an acknowledgment in the product documentation would be
appreciated but is not required.
2. Altered source versions must be plainly marked as such, and must not be
misrepresented as being the original software.
3. This notice may not be removed or altered from any source distribution.
*/
/**************
* SocketStream is a stream for a blocking,
* connected Socket.
*
* For Win32 systems, link with ws2_32.lib.
*
* Example:
* See /dmd/samples/d/htmlget.d
* Authors: Christopher E. Miller
* References:
* $(LINK2 std_stream.html, std.stream)
* Macros: WIKI=Phobos/StdSocketstream
*/
module std.socketstream;
private import std.stream;
private import std.socket;
/**************
* SocketStream is a stream for a blocking,
* connected Socket.
*/
class SocketStream: Stream
{
private:
Socket sock;
public:
/**
* Constructs a SocketStream with the specified Socket and FileMode flags.
*/
this(Socket sock, FileMode mode)
{
if(mode & FileMode.In)
readable = true;
if(mode & FileMode.Out)
writeable = true;
this.sock = sock;
}
/**
* Uses mode FileMode.In | FileMode.Out.
*/
this(Socket sock)
{
writeable = readable = true;
this.sock = sock;
}
/**
* Property to get the Socket that is being streamed.
*/
Socket socket()
{
return sock;
}
/**
* Attempts to read the entire block, waiting if necessary.
*/
override uint readBlock(void* _buffer, uint size)
{
ubyte* buffer = cast(ubyte*)_buffer;
int len;
uint need = size;
assertReadable();
if(size == 0)
return size;
len = sock.receive(buffer[0 .. size]);
readEOF = cast(bool)(len == 0);
if(len < 0)
len = 0;
return len;
}
/**
* Attempts to write the entire block, waiting if necessary.
*/
override uint writeBlock(void* _buffer, uint size)
{
ubyte* buffer = cast(ubyte*)_buffer;
int len;
assertWriteable();
if(size == 0)
return size;
len = sock.send(buffer[0 .. size]);
readEOF = cast(bool)(len == 0);
if(len < 0)
len = 0;
return len;
}
/**
*
*/
override ulong seek(long offset, SeekPos whence)
{
throw new SeekException("Cannot seek a socket.");
return 0;
}
/**
* Does not return the entire stream because that would
* require the remote connection to be closed.
*/
override char[] toString()
{
return sock.toString();
}
/**
* Close the Socket.
*/
override void close()
{
sock.close();
}
}