Node:Stream, Next:Iterator, Previous:Attribute, Up:libmailbox
#include <mailutils/stream.h>
These generic flags are interpreted as appropriate to the specific streams.
MU_STREAM_READ
MU_STREAM_WRITE
MU_STREAM_RDWR
MU_STREAM_APPEND
MU_STREAM_CREAT
MU_STREAM_NONBLOCK
MU_STREAM_NO_CHECK
MU_STREAM_NO_CLOSE
| int file_stream_create (stream_t *pstream, const char *filename, int flags) | Function |
| int tcp_stream_create (stream_t *pstream, const char *host, int port, int flags) | Function |
| int mapfile_stream_create (stream_t *pstream, const char *filename, int flags) | Function |
| int memory_stream_create (stream_t *pstream, const char *filename, int flags) | Function |
| int encoder_stream_create (stream_t *pstream, stream_t iostream, const char *encoding) | Function |
| int decoder_stream_create (stream_t *pstream, stream_t iostream, const char *encoding) | Function |
| int stdio_stream_create (stream_t *pstream, FILE* stdio, int flags) | Function |
| If MU_STREAM_NO_CLOSE is specified, fclose() will not be called on stdio when the stream is closed. |
| void stream_destroy (stream_t *pstream, void *owner) | Function |
| int stream_open (stream_t stream) | Function |
| int stream_close (stream_t stream) | Function |
| int stream_is_seekable (stream_t stream) | Function |
| int stream_get_fd (stream_t stream, int *pfd) | Function |
| int stream_read (stream_t stream, char *buffer, size_t buflen, off_t offset, size_t *pwriten) | Function |
| int stream_readline (stream_t stream, char *buffer, size_t buflen, off_t offset, size_t *pwriten) | Function |
| int stream_size (stream_t stream, off_t *psize) | Function |
| int stream_truncate (stream_t stream, off_t size) | Function |
| int stream_write (stream_t stream, const char *buffer, size_t buflen, off_t offset, size_t *pwriten) | Function |
| int stream_setbufsiz (stream_t stream, size_t size) | Function |
| int stream_flush (stream_t stream) | Function |
These functions will typically only be useful to implementors of streams.
| int stream_create (stream_t *pstream, int flags, void *owner) | Function |
| Used to implement a new kind of stream. |
| int stream_get_flags (stream_t stream, int *pflags) | Function |
| int stream_get_state (stream_t stream, int *pstate) | Function |
|
An example using tcp_stream_create to make a simple web client:
/* This is an example program to illustrate the use of stream functions.
It connects to a remote HTTP server and prints the contents of its
index page */
#include <stdlib.h>
#include <stdio.h>
#include <errno.h>
#include <string.h>
#include <unistd.h>
#include <mailutils/mailutils.h>
const char *wbuf = "GET / HTTP/1.0\r\n\r\n";
char rbuf[1024];
int
main ()
{
int ret, off = 0, fd;
stream_t stream;
size_t nb;
fd_set fds;
ret = tcp_stream_create (&stream, "www.gnu.org", 80, MU_STREAM_NONBLOCK);
if (ret != 0)
{
mu_error ( "tcp_stream_create: %s\n", mu_errstring (ret));
exit (EXIT_FAILURE);
}
connect_again:
ret = stream_open (stream);
if (ret != 0)
{
if (ret == EAGAIN)
{
ret = stream_get_fd (stream, &fd);
if (ret != 0)
{
mu_error ( "stream_get_fd: %s\n", mu_errstring (ret));
exit (EXIT_FAILURE);
}
FD_ZERO (&fds);
FD_SET (fd, &fds);
select (fd + 1, NULL, &fds, NULL, NULL);
goto connect_again;
}
mu_error ( "stream_open: %s\n", mu_errstring (ret));
exit (EXIT_FAILURE);
}
ret = stream_get_fd (stream, &fd);
if (ret != 0)
{
mu_error ( "stream_get_fd: %s\n", mu_errstring (ret));
exit (EXIT_FAILURE);
}
write_again:
ret = stream_write (stream, wbuf + off, strlen (wbuf), 0, &nb);
if (ret != 0)
{
if (ret == EAGAIN)
{
FD_ZERO (&fds);
FD_SET (fd, &fds);
select (fd + 1, NULL, &fds, NULL, NULL);
off += nb;
goto write_again;
}
mu_error ( "stream_write: %s\n", mu_errstring (ret));
exit (EXIT_FAILURE);
}
if (nb != strlen (wbuf))
{
mu_error ( "stream_write: %s\n", "nb != wbuf length");
exit (EXIT_FAILURE);
}
do
{
ret = stream_read (stream, rbuf, sizeof (rbuf), 0, &nb);
if (ret != 0)
{
if (ret == EAGAIN)
{
FD_ZERO (&fds);
FD_SET (fd, &fds);
select (fd + 1, &fds, NULL, NULL, NULL);
}
else
{
mu_error ( "stream_read: %s\n", mu_errstring (ret));
exit (EXIT_FAILURE);
}
}
write (2, rbuf, nb);
}
while (nb || ret == EAGAIN);
ret = stream_close (stream);
if (ret != 0)
{
mu_error ( "stream_close: %s\n", mu_errstring (ret));
exit (EXIT_FAILURE);
}
stream_destroy (&stream, NULL);
exit (EXIT_SUCCESS);
}