**README** for PLIB3.IO 0.10.5

:Author:        Peter A. Donis
:Release Date:  20 Oct 2021

**NOTE: This package is deprecated, and will not be updated
further. For a more up to date I/O toolkit, see the NET
sub-package of PLIB3.STDLIB.**

The PLIB3.IO package contains classes that encapsulate various forms
of client/server I/O channels.

Note: PLIB3.IO works with Python 3. If you are using Python
2.7, see the PLIB.IO package, available at
https://pypi.org/project/plib.io.

The ``setup.py`` script for PLIB3.IO uses the ``setuputils``
helper module, which helps to automate away much of the
boilerplate in Python setup scripts. This module is available
as a separate release at https://pypi.org/project/setuputils3.

The PLIB3.IO Packages and Modules
---------------------------------

The ``comm`` sub-package provides utilities for managing and
communicating with child threads and processes. This is shipped
as a separate ``plib`` sub-package from ``io``, so it can be
used independently of the ``io`` code.

The ``base`` sub-package contains base classes that implement
common basic functionality that is built on by the rest of PLIB3.IO.

Most of the remaining sub-packages fall into four main categories:

- **Device Types**: ``socket`` and ``serial``. Each device type has
  a ``BaseClient`` and ``BaseServer`` class; the ``socket`` type
  also has a ``BaseRequest`` class. These will usually not need to
  be used directly; they are used by the I/O mode classes, and are
  factored out so that each I/O mode sees the same API for a given
  device type.

- **Channel Types**: The ``comm`` sub-package contains classes that
  implement the basic functionality of the three types of I/O
  channels: clients, servers, and "persistent" (the latter is only
  available with the nonblocking I/O mode--see below). Each I/O mode
  then builds on these base classes to implement its specific channels.

- **I/O modes**: ``nonblocking`` and ``blocking`` (the latter does not
  just mean synchronous: it includes a forking TCP socket server). Each
  I/O mode has a client and server class for both device types, and
  a request class for the ``socket`` device type: the class names are
  ``SerialClient``, ``SerialServer``, ``SocketClient``, ``SocketServer``,
  and ``BaseRequestHandler``. The ``nonblocking`` type also has "persistent"
  classes, which support full-duplex asynchronous communication; these
  are the ``PersistentSerial``, ``PersistentSocket``, and
  ``PersistentRequestHandler`` classes. Mixin versions of these classes
  (class names with ``Mixin`` at the end) are also provided, for use
  if alternate data handling is desired (see next bullet), but it is
  normally not necessary to use these "by hand"--see "automatic mixins"
  below.

- **Data Handling**: the I/O mode classes given above include basic
  data handling, but it is *very* basic: the only way it can detect
  that a "message" has been fully received is to detect a closed
  channel. For some applications this is enough, but often more
  sophisticated and robust data handling is needed. The ``data``
  sub-package provides three mixin classes for this purpose,
  ``ShutdownReadWrite``, ``TerminatorReadWrite`` and ``ReadWrite``.
  The first of these detects the end of a received message by a
  shutdown of the other end of the data channel, but keeps the channel
  open to allow further writes (all the other classes default to
  closing the channel when the other end closes). The other two
  classes allow the detection of multiple "messages" in the data
  stream, either by detecting a "terminator" string or by having
  each message include its length at the beginning. These classes
  also format outgoing messages the same way.

There is also a ``mixins`` sub-package containing classes that
are used as mixins by the other sub-packages, and a ``classes``
sub-package containing higher-level classes that use the API.
Finally, there is a ``utils`` module that implements the automatic
mixin functionality described below.

Automatic Mixins
~~~~~~~~~~~~~~~~

To derive your own client or server classes with alternate data
handling "by hand", you would need to use the "mixin" versions of
the appropriate I/O mode classes, and splice the data handling
class into the middle of the base class list; for example::

    from plib.io.nonblocking import SerialClientMixin, SerialBase
    from plib.io.data import TerminatorReadWrite
    
    class AsyncSerialClientWithTerminator(SerialClientMixin,
        TerminatorReadWrite, SerialBase): pass

This is a bit clumsy, but necessary since the read/write handling has
to be *before* the client/server class in the MRO, but *after* the
base device type, for the cooperative ``super`` calls that underlie
the functionality to work properly. However, since the pattern is the
same in each case, it can be automated, and this has been done in the
``nonblocking`` and ``blocking`` sub-package namespaces, so that instead
of doing the above class construction "by hand", you can just append a
suffix to your desired class name, thus::

    from plib.io.nonblocking import SerialClientWithTerminator

The ``WithTerminator`` suffix (or, alternately, ``WithShutdown``
or ``WithReadWrite``) will cause the equivalent of the above class
definition to occur on the fly, so that the resulting class appears
in the ``plib.io.nonblocking`` namespace (of course the
``plib.io.blocking`` namespace has the same capability).
Once this has happened the first time, however, the class definition
is stored in the appropriate namespace, so additional imports of the
same class name (in different modules of your application) will not
re-do the "on the fly" construction; they will just retrieve the
same class object that was previously constructed.

The above machinery is also made available for use with your own custom
read/write handling classes; the ``nonblocking`` and ``blocking``
sub-packages each export a ``get_readwrite_class`` function that does the
same on-the-fly class definition as above, but with your custom read/write
class instead of one of the built-in ones. All you have to do is pass
the function the name of your desired I/O class and your custom
read/write class object::

    from plib.io import nonblocking
    
    class CustomReadWrite(object):
        # class definition
    
    MyAsyncSerialClient = nonblocking.get_readwrite_class('SerialClient',
        CustomReadWrite)

*API Notes*: One of the goals of this sub-package is to provide a
common, consistent API for all the different types of I/O, so that
switching one specific implementation of a certain functionality
for another can be done transparently to the rest of your application's
code. Thus, all of the usable classes follow the same basic pattern of
mixing in the various pieces of functionality: from left to right
in a class's MRO, one finds the type of endpoint (a client or
server mixin class, which may be specialized to the type of I/O),
the type of data formatting, if any (a mixin class from the
``ReadWrite`` module), and the type of I/O, including device type
(socket, serial port, etc.), mode (non-blocking/asynchronous vs.
blocking), and basic data handling. Also, each endpoint type has
a common API independent of the specific type of I/O and mode; a
client can always use the ``client_communicate`` method to send
data to the server and receive a response; a server can always use
the ``serve_forever`` method to start itself; and all I/O objects
override the same methods to implement application-specific
functionality: ``process_data``, to deal with data as it comes in,
and ``query_done``, to determine when the I/O channel should be
closed. (To see examples of all this in action, look at the test
suite in ``test_io.py`` and the library module for it,
``io_testlib.py``; the library module can use the same
mixin classes to implement test functionality for *all* of the
different mixes of I/O classes in the test suite.)

Installation
------------

To install PLIB3.IO, you can simply run::

    $ python setup.py install

at a shell prompt from the directory into which you
unzipped the source tarball (the same directory that this
README file is in). This will install PLIB3 and then
run each of the post-install scripts in the scripts
directory.

Example Programs
----------------

PLIB3.IO comes with example programs that illustrate key features
of the package. After installation, these can be found in the
``$PREFIX/share/plib/examples`` directory. If you have a
POSIX system (Linux or Mac OSX), the ``plib-setup-examples``
post-install script will install symlinks to the example
programs in the ``$PREFIX/bin`` directory.

The Zen of PLIB3
----------------

There is no single unifying purpose or theme to PLIB3, but
like Python itself, it does have a 'Zen' of sorts:

- Express everything possible in terms of built-in Python
  data structures.

- Once you've expressed it that way, what the code is
  going to do with it should be obvious.

- Avoid boilerplate code, *and* boilerplate data. Every
  piece of data your program needs should have one and
  only one source.

Copyright and License
---------------------

PLIB3.IO is Copyright (C) 2008-2019 by Peter A. Donis.

This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2 of the License, or
(at your option) any later version. (See the LICENSE file for a
copy of version 2 of the License.)

This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
GNU General Public License for more details.

You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
