The network Library#

Overview#

The Network library provides internet address protocols and TCP/IP server and client sockets. It exports a single module, called Sockets.

Utilities#

This section covers the start-sockets function, which all libraries using the Network library must call before any other call to the Network library API. It also covers the with-socket-thread macro which registers the current thread as a thread that will call a socket function that blocks.

start-sockets Function#
Signature:

start-sockets () => ()

Discussion:

Applications must call this function before using any other function or variable from the Network library.

This function is necessary because the Win32 API library Winsock2, which the Network library calls to get native socket functionality, requires applications to call an initialization function before calling any Winsock2 API functions. The call to start-sockets calls this underlying Winsock2 function.

Note that you must not call start-sockets from a top-level form in any DLL project. The combination of this, and the restriction that you must call start-sockets before calling anything else, implies that no Network library function or variable can be called (directly or indirectly) from a top-level form in any DLL project. Instead, the DLL project should define a start function that calls start-sockets (directly or indirectly) or re-export start-sockets so that their clients can arrange to have it called from a top-level form in an appropriate EXE project.

Applications using the Network library must arrange for start-sockets to be called (directly or indirectly) before any other sockets API functions. A good place to do this is at the beginning of your startup function (usually called main, or similar). For example:

define method main () => ()
  start-sockets();
  let the-server = make(<tcp-server-socket>, port: 7);
  ...
end;

begin
  main();
end;

New start functions that call start-sockets and that are defined for DLL projects that use the Network library will inherit all of the restrictions described above for start-sockets.

Calling a Network library function before calling start-sockets results in a <sockets-not-initialized> error. Calling start-sockets from a top-level form in a DLL project will result in unpredictable failures — probably access violations during initialization.

with-socket-thread Statement Macro#
Macro Call:

with-socket-thread (#key server?) body end

Discussion:

Registers the current thread as a blocking socket thread, that is, a thread which will call a socket function that blocks, such as read-element or accept.

The reason for the registration is that Network library shutdown can then synchronize with these threads. The early part of the shutdown sequence should cause the threads to unblock with an <end-of-stream-error> so that they can do whatever application cleanup is necessary. Once these threads have exited, the rest of the shutdown sequence can be executed.

A server socket thread (blocking on accept rather than read-element) notices that the shutdown sequence is underway slightly later, with a <blocking-call-interrupted> condition.

Internet addresses#

This section covers Internet address protocols.

Basic Internet address protocol#

This section covers the class <internet-address> and related generic functions and constants.

<internet-address> Open Primary Abstract Instantiable Class#
Superclasses:

<object>

Init-Keywords:
  • name – An instance of <string> representing a symbolic internet address.

  • address – An instance of <string> representing a presentation (dotted) form Internet address or an instance of <numeric-address> (see below).

Discussion:

The class of objects representing Internet addresses used as endpoints for peer-to-peer socket connections.

To construct an <internet-address> object you must supply either the name: or address: keyword. For example:

make (<internet-address>, name: "www.whatever.com")

or

make (<internet-address>, address: "9.74.122.0")

make on <internet-address> returns an instance of <ipv4-address>.

host-name Open Generic function#
Signature:

host-name internet-address => name

Discussion:

Returns an instance of <string> containing a symbolic host name. The internet-address argument must be an instance of <internet-address>.

Usually the name returned is the canonical host name. Note, however, that the implementation is conservative about making DNS calls. Suppose that the <internet-address> instance was created with the name: keyword and no other information. If the application has not made any other requests that would require a DNS call, such as to host-address or aliases, the name that this function returns will be the one specified with the name: keyword, regardless of whether that is the canonical name or not.

host-address Open Generic function#
Signature:

host-address internet-address => address

Discussion:

Returns an instance of <string> containing the presentation form of the host address. In the case of multi-homed hosts this will usually be the same as:

multi-homed-internet-address.all-addresses.first.host-address

In the case of an Internet address created using the address: keyword it will be either the keyword value or all-addresses.first.host-address.

numeric-host-address Open Generic function#

Returns the host address as a <numeric-address>.

Signature:

numeric-host-address internet-address => numeric-address

all-addresses Open Generic function#
Signature:

all-addresses internet-address => sequence

Discussion:

Returns an instance of <sequence> whose elements are <internet-address> objects containing all known addresses for the host.

aliases Open Generic function#
Signature:

aliases internet-address => sequence

Discussion:

Returns an instance of <sequence> whose elements are instances of <string> representing alternative names for the host.

local-host-name Function#
Signature:

local-host-name () => name

Discussion:

Returns an instance of <string> containing a symbolic host name.

The <ipv6-address> class#

This name is reserved for future development.

The <numeric-address> class#

This section describes numeric Internet representation and associated protocols.

<numeric-address> Primary Abstract Sealed Class#
Superclasses:

<object>

Discussion:

The class of objects representing the numeric form of an Internet addresses.

Currently only ipv4 (32-bit) addresses are supported. Ipv6 addresses will be added when they are supported by Winsock2. In general <numeric-address> objects are accessed using the functions host-order or network-order, depending on the context in which they are employed.

network-order Sealed Generic function#
Signature:

network-order address => network-order-address

Discussion:

Returns the value of the numeric address in network order. The argument is a general instance of <numeric-address>. The class of the object returned depends upon the particular subclass of the argument; the network-order method for <ipv4-numeric-address> returns an instance of <machine-word>.

Network order is big-endian byte order.

host-order Sealed Generic function#
Signature:

host-order address => host-order-address

Discussion:

Like network-order but returns the value in host order.

Host order is big-endian byte order on a big-endian host machine and little-endian on a little-endian host machine.

IPV4 addresses#

<ipv4-numeric-address> Open Primary Abstract Instantiable Class#
Superclasses:

<numeric-address>

Init-Keywords:
  • value – An instance of <machine-word>. Required.

  • order – One of #"network-order" or #"host-order". Required.

Discussion:

The single slot of this class contains a 32-bit value representing a ipv4 address. This slot is accessed by the generic functions network-order and host-order described above. <ipv4-numeric-address> has two concrete subclasses <ipv4-network-order-address> and <ipv4-host-order-address>. Making a <ipv4-numeric-address> returns one or the other of these depending upon the value of the order: keyword.

host-order(<ipv4-numeric-address>) Method#
Signature:

host-order ipv4-numeric-address => machine-word

Discussion:

Returns the numeric address in host order as an instance of <machine-word>. The argument is an instance of <ipv4-numeric-address>.

network-order(<ipv4-numeric-address>) Method#
Signature:

network-order ipv4-numeric-address => machine-word

Discussion:

Returns the numeric address in network order as an instance of <machine-word>. The argument is an instance of <ipv4-numeric-address>.

as(<string>, <ipv4-numeric-address>) Method#

Returns the presentation (dotted string) form of an instance of <ipv4-numeric-address>.

Signature:

as string ipv4-numeric-address => string

<ipv4-network-order-address> Concrete Sealed Class#

Concrete subclass for network-order numeric addresses.

Superclasses:

<ipv4-numeric-address>

Discussion:

make(<ipv4-network-order-address>)

is equivalent to

make(<ipv4-numeric-address>, order: network-order)

<ipv4-host-order-address> Concrete Sealed Class#

Concrete subclass for host order numeric addresses.

Superclasses:

<ipv4-numeric-address>

Sockets#

This section describes socket classes and protocols.

The <abstract-socket> class#

<abstract-socket> Open Free Abstract Uninstantiable Class#
Superclasses:

<object>

Init-Keywords:
  • socket-descriptor – A Windows handle or UNIX file descriptor for the socket. In general users of the sockets API should not need to use this keyword. Only implementors of new socket classes should be interested.

Discussion:

The common superclass of all socket objects including <socket> (IP client socket), <server-socket> and <socket-accessor>.

Each subclass of <abstract-socket> must provide methods for close and for the following generic functions:

local-port Open Generic function#

Returns the local port number.

Signature:

local-port socket => port-number

Parameters:
Values:
socket-descriptor Open Generic function#

Returns the descriptor (handle or fd) for the socket.

Signature:

socket-descriptor socket => descriptor

Parameters:
Values:
  • descriptor – An instance of <accessor-socket-descriptor>.

local-host Open Generic function#

Returns the address of the local host.

Signature:

local-host socket => host-address

Parameters:
Values:

The <server-socket> class#

<server-socket> Open Primary Abstract Instantiable Class#
Superclasses:

<abstract-socket>

Init-Keywords:
  • service – An instance of <string> containing an abstract name for a service with a “well-known” port, such as "ftp" or "daytime". Valid names depend on the configuration of the DNS. Required unless port: is supplied.

  • port – An instance of <integer> identifying the port on which the <server-socket> should listen for connection requests. Required unless service: is supplied.

  • protocol – An instance of <string> naming the protocol. Currently "tcp" is the only supported protocol. You can create instances of protocol-specific subclasses as an alternative to using the protocol: keyword. For example, make(<server-socket>, protocol: "tcp", ...) is equivalent to make(<tcp-server-socket>, ...).

Discussion:

Server-sockets listen on a specified port for connection requests which come in over the network. Either the port: or service: keyword must be supplied.

make on <server-socket> returns an instance of <tcp-server-socket> by default.

accept Open Generic function#
Signature:

accept server-socket #rest args #key => result

Discussion:

Blocks until a connect request is received, then it returns a connected instance of <socket>. The particular subclass of <socket> returned depends on the actual class of the server-socket argument, which must be a general instance of <server-socket>. Calling accept on <tcp-server-socket> returns a connected <tcp-socket>. The keyword arguments are passed to the creation of the <socket> instance.

For UDP sockets accept returns immediately with an instance of <udp-socket>. No blocking happens for UDP sockets because they are connectionless. After reading from a UDP socket returned from accept the socket can be interrogated for the location of the sender using remote-host and remote-port.

with-server-socket Macro#
Macro Call:

with-server-socket (server-var [:: server-class ], keywords)
  body
end

Discussion:

Creates an instance of <server-socket>, using the (optional) server-class argument and keyword arguments to make the <server-socket>, and binds it to the local variable named by server-var. The body is evaluated in the context of the binding and the <server-socket> is closed after the body is executed.

start-server Macro#
Macro Call:

start-server ([server-var = ] socket-server-instance, socket-var [, keywords ])
  body
end

Discussion:

Enters an infinite while(#t) accept loop on the server socket. Each time accept succeeds the <socket> returned from accept is bound to socket-var and body is evaluated in the context of the binding. When body exits, accept is called again producing a new binding for socket-var. The optional keywords are passed to the call to accept.

The <tcp-server-socket> class#

<tcp-server-socket> Class#
Superclasses:

<server-socket>

Init-Keywords:
  • element-type – Establishes a new default for the element-type of <tcp-socket> instances returned by calling accept on this server socket. This default element-type may be overridden for any particular call to accept by using the element-type: keyword to accept. If no element-type: is specified when the server socket is created, <byte-character> is used as the default element-type.

Discussion:

The class of TCP server sockets. A server socket is an object which listens for requests for connections from the network. When accept is called on the server socket and a request for connection is detected, accept returns a connected <socket>.

accept(<tcp-server-socket>) Method#
Signature:

accept server-socket #rest args #key element-type => connected-socket

Parameters:
  • server-socket – An instance of <tcp-server-socket>.

  • element-type (#key) – Controls the element type of the <tcp-socket> (stream) returned. If not supplied, defaults to #f.

Values:
  • connected-socket – A connected instance of <tcp-socket>.

Discussion:

The other keyword arguments are passed directly to the creation of the <tcp-socket> instance.

The <socket> class#

<socket> Open Free Abstract Instantiable Class#

The class of general client sockets. All client sockets are streams.

Superclasses:

<abstract-socket>, <external-stream>

Init-Keywords:
  • direction – Specifies the direction of the stream. It must be one of #"input", #"output", and #"input-output". This keyword is an inherited streams class keyword. See The streams Module for a full description.

  • element-type – An instance of <class>. Useful values are <byte-character> and <byte>. This keyword is an inherited streams class keyword. See The streams Module for a full description.

The <buffered-socket> class#

<buffered-socket> Class#
Superclasses:

<socket>, <double-buffered-stream>

Init-Keywords:
  • force-output-before-read? – An instance of <boolean>. Defaults to #t. The methods which implement the stream reading protocols (read, read-line, read-element and so on) for instances of <socket> call force-output by default before blocking. This is to ensure that any pending output has been sent to the peer before the socket blocks waiting to read data sent by the peer. This corresponds to the expected, usual behavior of single-threaded client sockets and avoids deadlock in usual cases. Multi-threaded applications, particularly applications where one thread is reading and another thread is writing to the same socket, may wish to inhibit the default force-output. If the socket is created with force-output-before-read?: #f, force-output will not be called before the read functions block.

Discussion:

Socket streams whose elements are bytes or characters. These inherit buffering protocols and the implications of read, write, read-element, write-element, force-output and suchlike methods from <double-buffered-stream>.

The <tcp-socket> class#

The class of TCP client sockets.

<tcp-socket> Class#

The class of TCP client sockets.

Superclasses:

<buffered-socket>

Init-Keywords:
  • host – An instance of <internet-address> or <string>. The remote host to connect to. The <string> may be either a host name or a presentation-form Internet address. Required.

  • service – An instance of <string>. A <string> containing an abstract name for a service with a “well-known” port, such as "ftp" or "daytime". Valid names depend on the configuration of the DNS. Required unless port: is supplied.

  • protocol – An instance of <string> naming the protocol. Currently #"tcp" and #"udp" are the only supported protocols. You can create instances of protocol-specific subclasses as an alternative to using the protocol: keyword. For example make(<socket>, protocol: #"tcp", ...) is equivalent to make(<tcp-socket>, ...). make on <socket> returns an instance of <tcp-socket> by default.

  • port – An instance of <integer> representing the remote port to connect to. Required unless service: is supplied.

  • element-type – An instance of <class>. Useful values for <tcp-streams> are <byte-character> and <byte>. This keyword is an inherited streams class keyword. See The streams Module for a full description.

remote-port Open Generic function#

Returns the remote port number for a <socket>.

Signature:

remote-port socket => port-number

Parameters:
Values:
remote-host Open Generic function#

Returns the remote host for a <socket>.

Signature:

remote-host socket => remote-host-address

Parameters:
Values:

The <udp-socket> class#

The class of UDP client sockets.

<udp-socket> Class#

The class of UDP client sockets.

Superclasses:

<buffered-socket>

Init-Keywords:
  • host – An instance of <internet-address> or <string>. The remote host to connect to. The <string> may be either a host name or a presentation-form Internet address. Required.

  • service – An instance of <string>. A <string> containing an abstract name for a service with a “well-known port”, such as "ftp" or "daytime". Valid names depend on the configuration of the DNS. Required unless port: is supplied.

  • protocol – An instance of <string> naming the protocol. Currently #"tcp" and #"udp" are the only supported protocols. You can create instances of protocol-specific subclasses as an alternative to using the protocol: keyword. For example make(<socket>, protocol: "udp", ...) is equivalent to make(<UDP-socket>, ...). make on <socket> returns an instance of <tcp-socket> by default.

  • port – An instance of <integer> representing the remote port to connect to. Required unless service: is supplied.

  • element-type – An instance of <class>. Useful values for <udp-socket> s are <byte-character> and <byte>. This keyword is an inherited streams class keyword. See The streams Module for a full description.

Discussion:

Of the keywords, host: and one of either service: or port: are required.

The <udp-server-socket> class#

The class of UDP server sockets.

<udp-server-socket> Class#
Superclasses:

<server-socket>

Init-Keywords:
  • element-type – Establishes a new default for the element-type of <UDP-socket> instances returned by calling accept with this server socket as the argument to accept. This default element-type may be overridden for any particular call to accept by using the element-type: keyword to accept. If no element-type: is specified when the server socket is created, <byte-character> is used as the default element-type.

Discussion:

The class of UDP server sockets. A server socket is an object that listens for requests from the network. When accept is called on the UDP server socket, accept returns a <udp-socket>.

Socket conditions#

This section lists the socket condition classes in the Network library.

<socket-condition> Class#

All socket conditions are general instances of <socket-condition>. Some are recoverable and others are not.

Superclasses:

<simple-condition>

Discussion:

The class of socket conditions. It inherits the format-string: and format-arguments: keywords from <simple-condition>.

Slots:

socket-condition-details

  • Most socket conditions originate in error return codes from the underlying host operating system’s network library.

  • The socket-condition-details slot provides information about the low-level failure which was the source for the condition. In most cases this slot will hold an instance of <socket-accessor-condition>.

  • When creating general instances of <socket-condition>, you can use the details: keyword to set the value for this slot.

<socket-error> Class#

The superclass of all unrecoverable socket conditions.

Superclasses:

<socket-condition>

The class of socket conditions from which no recovery is possible.

<internal-socket-error> Class#

The class of unexpected socket errors.

Superclasses:

<socket-error>

Discussion:

The class of unexpected errors encountered while interacting with the underlying host system’s network library.

Inspect the contents of the socket-condition-details slot for more information.

<recoverable-socket-condition> Class#

The class of socket conditions for which an application may be able to take some remedial action.

Superclasses:

<socket-condition>

Discussion:

The general class of socket conditions for which an application may be able to take some remedial action.

For instance, a web browser receiving such conditions as <connection-refused> or <host-not-found> would normally expect to report those conditions to the user and continue with some other connection request from the user, while a server receiving a <connection-closed> condition from a connected <socket> would probably close the <socket> and continue to handle other requests for connections.

<network-not-responding> Class#

The network — probably a local network — is down. Try again later.

Superclasses:

<recoverable-socket-condition>

<invalid-address> Class#

A badly formed address string has been passed to a function trying to make an <internet-address>.

Superclasses:

<recoverable-socket-condition>

<host-not-found> Class#

The Domain Name Server (DNS) cannot resolve the named host or internet address. Try again with a different (correct) name or address.

Superclasses:

<recoverable-socket-condition>

<server-not-responding> Class#

The Domain Name Server (DNS) did not respond or returned an ambiguous result. Try again.

Superclasses:

<recoverable-socket-condition>

<host-unreachable> Class#

The remote host cannot be reached from this host at this time.

Superclasses:

<recoverable-socket-condition>

<socket-closed> Class#
Superclasses:

<recoverable-socket-condition>

Discussion:

The socket or server socket has been closed.

Most operations on closed instances of <tcp-socket> return instances of <stream-closed-error> (from the Streams library) rather than instances of <socket-closed>.

<connection-failed> Class#
Superclasses:

<recoverable-socket-condition>

Discussion:

The attempt to connect to the remote host was not successful. Connection failed for one of the following reasons: because the connect request timed out or because it was refused, or because the remote host could not be reached.

<connection-closed> Class#
Superclasses:

<recoverable-socket-condition>

Discussion:

The connection to the remote host has been broken. The socket should be closed. To try again, open a new socket.

<address-in-use> Class#
Superclasses:

<recoverable-socket-condition>

Discussion:

A process on the machine is already bound to the same fully qualified address. This condition probably occurred because you were trying to use a port with an active server already installed, or a process crashed without closing a socket.

<blocking-call-interrupted> Class#

A blocking socket call, like read, write or accept, was interrupted.

Superclasses:

<recoverable-socket-condition>

<out-of-resources> Class#
Superclasses:

<recoverable-socket-condition>

Discussion:

The implementation-dependent limit on the number of open sockets has been reached. You must close some sockets before you can open any more.

<socket-accessor-error> Class#
Superclasses:

<socket-error>

Discussion:

An implementation-specific error from the C-FFI interface to the native socket library. Usually instances of this class appear in the socket-condition-details slot of another <socket-condition>.

<win32-socket-error> Class#
Superclasses:

<socket-accessor-error>

Discussion:

A Win32-specific error from the Winsock2 library, a C-FFI interface to the native socket library Winsock2. A function in the FFI library has returned an error return code.

Slots:

WSA-numeric-error-code

Contains the numeric error code that was returned. An instance of <integer>.

WSA-symbolic-error-code

Contains an instance of <string> giving the symbolic (human-readable) form of the error code. For example, the string might be “wsanotsock”.

explanation

An explanation if any of the error. An instance of <string>.

calling-function

The name of Winsock2 FFI interface function which returned the error code. An instance of <string>.