Channels

Channels — Using channels for network-based operation of Libdexter

Synopsis

#include <dexter.h>

#define             DEXTER_CHANNEL_ERROR
enum                DexterChannelError;

                    DexterChannel;
enum                DexterChannelState;
enum                DexterChannelEvent;

void                (*DexterChannelEventCallback)       (DexterChannel *channel,
                                                         gint event);

struct              DexterChannelTLSSettings;
void                (*DexterChannelTLSStartCallback)    (DexterChannel *channel,
                                                         const struct sockaddr *server_sa,
                                                         socklen_t server_salen,
                                                         DexterChannelTLSSettings *tls_settings);

enum                DexterChannelSettingFlags;
struct              DexterChannelSettings;

DexterChannel *     dexter_channel_new                  (const DexterChannelSettings *settings);
gboolean            dexter_channel_open                 (DexterChannel *channel,
                                                         GError **error);
gboolean            dexter_channel_close                (DexterChannel *channel,
                                                         GError **error);
gboolean            dexter_channel_greet                (DexterChannel *channel,
                                                         GError **error);
gboolean            dexter_channel_noop                 (DexterChannel *channel,
                                                         GError **error);
gboolean            dexter_channel_starttls             (DexterChannel *channel,
                                                         GError **error);
gboolean            dexter_channel_part                 (DexterChannel *channel,
                                                         GError **error);
void                dexter_channel_free                 (DexterChannel *channel);
gint                dexter_channel_get_state            (DexterChannel *channel);

Description

Channels are Libdexter objects used to interact with hosts on the network running Libdexter servers. To use a channel, you must first create it with dexter_channel_new(). Next, open the channel with dexter_channel_open() to establish a network connection with the server. You must then greet the server with dexter_channel_greet() to perform the protocol handshake that readies the channel for actual use. You may secure your channel communications anytime after the greeting is exchanged with dexter_channel_starttls(). Note that the server may require that you issue dexter_channel_starttls() in order to perform any meaningful data operation. The server may also require that the channel present valid X.509 certificate credentials.

Once the greeting is exchanged, you may then create remote Libdexter objects on the server by specifying the channel when creating your objects, e.g. with dexter_sampler_new(). When you are done with the channel and have destroyed all objects you created on it, issue dexter_channel_part() to tell the server that the channel will no longer be used. At that point, the server will disconnect, you can then issue dexter_channel_close() and, finally, dexter_channel_free() to destroy the channel.

When using channels to create objects on other hosts, two objects are actually created: one on the remote host and one local to your application. The local object acts as a transparent proxy to the the remote object, issuing network messages to the remote object as you manipulate it. There are no api differences, however, from your application's point of view, i.e. working with a local object is identical to working with a remote object.

Each channel is implemented with a single tcp/ip connection to the specified host. You may create many Libdexter objects, e.g. samplers or timed samplers, over a single channel. There is no need to create a separate channel for each object (although you can). The Libdexter configuration file is used to set default channel settings, such as socket options, etc.

Details

DEXTER_CHANNEL_ERROR

#define DEXTER_CHANNEL_ERROR dexter_channel_error_quark()

The error domain of the libdexter channel subsystem.


enum DexterChannelError

typedef enum {
  DEXTER_CHANNEL_ERROR_EADDRINUSE,
  DEXTER_CHANNEL_ERROR_EAGAIN,
  DEXTER_CHANNEL_ERROR_ECONNREFUSED,
  DEXTER_CHANNEL_ERROR_EINTR,
  DEXTER_CHANNEL_ERROR_ENETUNREACH,
  DEXTER_CHANNEL_ERROR_STATE,
  DEXTER_CHANNEL_ERROR_TIMEOUT,
  DEXTER_CHANNEL_ERROR_TLS_NOT_ENABLED,
  DEXTER_CHANNEL_ERROR_FAILED
} DexterChannelError;

Error codes for DexterChannel operations.

DEXTER_CHANNEL_ERROR_EADDRINUSE

local address in use.

DEXTER_CHANNEL_ERROR_EAGAIN

no more free ports or insufficient routing cache entries.

DEXTER_CHANNEL_ERROR_ECONNREFUSED

no one listening on the remote address.

DEXTER_CHANNEL_ERROR_EINTR

system call interrupted by signal that was caught.

DEXTER_CHANNEL_ERROR_ENETUNREACH

network is unreachable.

DEXTER_CHANNEL_ERROR_STATE

operation not allowed in current channel state.

DEXTER_CHANNEL_ERROR_TIMEOUT

operation timed out.

DEXTER_CHANNEL_ERROR_TLS_NOT_ENABLED

TLS not enabled on this installation.

DEXTER_CHANNEL_ERROR_FAILED

other failure.

DexterChannel

typedef struct _DexterChannel DexterChannel;

Opaque type with no public members.


enum DexterChannelState

typedef enum {
  DEXTER_CHANNEL_STATE_OPEN,
  DEXTER_CHANNEL_STATE_ABORT,
  DEXTER_CHANNEL_STATE_NOCONN,
  DEXTER_CHANNEL_STATE_CLOSED,
} DexterChannelState;

All possible channel states.

DEXTER_CHANNEL_STATE_OPEN

channel is open.

DEXTER_CHANNEL_STATE_ABORT

channel aborted connection.

DEXTER_CHANNEL_STATE_NOCONN

server disconnected.

DEXTER_CHANNEL_STATE_CLOSED

channel is closed.

enum DexterChannelEvent

typedef enum {
  DEXTER_CHANNEL_EVENT_OPEN,
  DEXTER_CHANNEL_EVENT_ABORT,
  DEXTER_CHANNEL_EVENT_NOCONN,
  DEXTER_CHANNEL_EVENT_CLOSED
} DexterChannelEvent;

All possible channel events.

DEXTER_CHANNEL_EVENT_OPEN

channel opened successfully.

DEXTER_CHANNEL_EVENT_ABORT

channel aborted connection.

DEXTER_CHANNEL_EVENT_NOCONN

server disconnected.

DEXTER_CHANNEL_EVENT_CLOSED

channel closed successfully.

DexterChannelEventCallback ()

void                (*DexterChannelEventCallback)       (DexterChannel *channel,
                                                         gint event);

Signature of callback function used to receive channel event notifications.

channel :

DexterChannel firing the event.

event :

DexterChannelEvent fired.

struct DexterChannelTLSSettings

struct DexterChannelTLSSettings {
  gchar    *cert_file;        /* ChannelTLSCertFile */ 
  gchar    *key_file;         /* ChannelTLSKeyFile */
  gchar    *trust_file;       /* ChannelTLSTrustFile */
  gchar    *crl_file;         /* ChannelTLSCrlFile  */
  gboolean  verify_hostname;  /* ChannelTLSVerifyHostname */
};

TLS settings that the channel will use when starting TLS. They are copied from the DexterChannelSettings structure provided to dexter_channel_new().

When this structure is seen within the context of a DexterChannelTLSStartCallback, you may change the elements as needed. For any non-NULL gchar elements that you want to change, you must first g_free() the element and then allocate a replacement string using a GLib string allocation api, e.g. g_strdup_printf(). The Library will free everything for you when the callback exits.

gchar *cert_file;

full path to X.509 certificate file in PEM format.

gchar *key_file;

full path to X.509 certificate private key file in PEM format.

gchar *trust_file;

full path to X.509 trust file in PEM format.

gchar *crl_file;

full path to X.509 certificate revocation list file in PEM format.

gboolean verify_hostname;

verify that the hostname of server matches its X.509 certificate.

Since 0.2.0


DexterChannelTLSStartCallback ()

void                (*DexterChannelTLSStartCallback)    (DexterChannel *channel,
                                                         const struct sockaddr *server_sa,
                                                         socklen_t server_salen,
                                                         DexterChannelTLSSettings *tls_settings);

Signature of callback function used to adjust TLS settings, as needed for this server address, as TLS starts.

If specificed in dexter_channel_starttls(), this callback allows you to inspect the server socket address and adjust your channel TLS parameters as needed. In this manner you can decide which certificate, trust and revocation files to use based on the server you are connected to.

channel :

DexterChannel on which TLS is starting.

server_sa :

server socket address.

server_salen :

length of server socket address.

tls_settings :

DexterChannelTLSSettings of TLS settings.

Since 0.2.0


enum DexterChannelSettingFlags

typedef enum {
  DEXTER_CHANNEL_SETTING_HOST                 = 1 << 0,
  DEXTER_CHANNEL_SETTING_SERVICE              = 1 << 1,
  DEXTER_CHANNEL_SETTING_SOCKET_OPTIONS       = 1 << 2,
  DEXTER_CHANNEL_SETTING_RECV_BUFFER_SIZE     = 1 << 3,
  DEXTER_CHANNEL_SETTING_TIMEOUT_MSECS        = 1 << 4,
  DEXTER_CHANNEL_SETTING_EVENT_CALLBACK       = 1 << 5,
  DEXTER_CHANNEL_SETTING_TLS_START_CALLBACK   = 1 << 6,
  DEXTER_CHANNEL_SETTING_TLS_CERT_FILE        = 1 << 7,
  DEXTER_CHANNEL_SETTING_TLS_KEY_FILE         = 1 << 8,
  DEXTER_CHANNEL_SETTING_TLS_TRUST_FILE       = 1 << 9,
  DEXTER_CHANNEL_SETTING_TLS_CRL_FILE         = 1 << 10,
  DEXTER_CHANNEL_SETTING_TLS_VERIFY_HOSTNAME  = 1 << 11
} DexterChannelSettingFlags;

Flags to indicate which elements of DexterChannelSettings have data on them.

Since 0.2.0

DEXTER_CHANNEL_SETTING_HOST

flag for host element of DexterServerSettings.

DEXTER_CHANNEL_SETTING_SERVICE

flag for service element of DexterServerSettings.

DEXTER_CHANNEL_SETTING_SOCKET_OPTIONS

flag for socket_options element of DexterServerSettings.

DEXTER_CHANNEL_SETTING_RECV_BUFFER_SIZE

flag for recv_buffer_size element of DexterServerSettings.

DEXTER_CHANNEL_SETTING_TIMEOUT_MSECS

flag for timeout_msecs element of DexterServerSettings.

DEXTER_CHANNEL_SETTING_EVENT_CALLBACK

flag for event_callback element of DexterServerSettings.

DEXTER_CHANNEL_SETTING_TLS_START_CALLBACK

flag for tls_start_callback element of DexterServerSettings.

DEXTER_CHANNEL_SETTING_TLS_CERT_FILE

flag for tls_cert_file element of DexterServerSettings.

DEXTER_CHANNEL_SETTING_TLS_KEY_FILE

flag for tls_key_file element of DexterServerSettings.

DEXTER_CHANNEL_SETTING_TLS_TRUST_FILE

flag for tls_trust_file element of DexterServerSettings.

DEXTER_CHANNEL_SETTING_TLS_CRL_FILE

flag for tls_cert_file element of DexterServerSettings.

DEXTER_CHANNEL_SETTING_TLS_VERIFY_HOSTNAME

flag for tls_verify_hostname element of DexterServerSettings.

struct DexterChannelSettings

struct DexterChannelSettings {
  DexterChannelSettingFlags     flags;
  gchar   *host;                /* ChannelHost */
  gchar   *service;             /* ChannelService */
  gchar   *socket_options;      /* ChannelSocketOptions */
  gint     recv_buffer_size;    /* ChannelRecvBufferSize */
  gint     timeout_msecs;       /* ChannelTimeoutMsecs */
  DexterChannelEventCallback    event_callback;         /* -- */
  DexterChannelTLSStartCallback tls_start_callback;     /* -- */
  gchar   *tls_cert_file;       /* ChannelTLSCertFile */
  gchar   *tls_key_file;        /* ChannelTLSKeyFile */
  gchar   *tls_trust_file;      /* ChannelTLSTrustFile */
  gchar   *tls_crl_file;        /* ChannelTLSCrlFile */
  gboolean tls_verify_hostname; /* ChannelTLSVerifyHostname */
};

Channel settings.

The flags element indicates which elements have user-supplied data.

Elements for which no flag is set, i.e. have no user-supplied data, will be set to a default: either from the configuration file, if available, or an internal default will be used.

DexterChannelSettingFlags flags;

Bitwise OR of flags from DexterChannelSettingFlags.

gchar *host;

host name or host ip.

gchar *service;

service name or port number.

gchar *socket_options;

tab or space-delimited list of socket options.

gint recv_buffer_size;

size of channel socket receive buffer.

gint timeout_msecs;

milliseconds to wait for server response before timing out.

DexterChannelEventCallback event_callback;

callback to receive channel event notifications.

DexterChannelTLSStartCallback tls_start_callback;

callback to override TLS settings as TLS starts.

gchar *tls_cert_file;

full path to X.509 certificate file in PEM format.

gchar *tls_key_file;

full path to X.509 certificate private key file in PEM format.

gchar *tls_trust_file;

full path to X.509 trust file in PEM format.

gchar *tls_crl_file;

full path to X.509 certificate revocation list file in PEM format.

gboolean tls_verify_hostname;

whether or not client requires that the server's DNS hostname match its X.509 certificate.

Since 0.2.0


dexter_channel_new ()

DexterChannel *     dexter_channel_new                  (const DexterChannelSettings *settings);

Creates a channel using the supplied settings.

NULL may be indicated for settings, in which case the configuration file is used for all channel settings. Internal defaults are used for items not found in the configuration file.

No network calls are performed until the channel is opened with dexter_channel_open().

The channel must be freed when done with dexter_channel_free().

settings :

address of DexterServerSettings or NULL.

Returns :

newly allocated DexterChannel if successful. NULL otherwise.

dexter_channel_open ()

gboolean            dexter_channel_open                 (DexterChannel *channel,
                                                         GError **error);

Opens the channel created with dexter_channel_new(). A new tcp/ip connection is established to the server for the channel. It remains open until dexter_channel_close() is called, the server disconnects, or a network error occurs.

In the event an open channel is broken due to a network or server problem, the channel will notify all objects running on channel and then the channel will transition to DEXTER_CHANNEL_STATE_NOCONN.

channel :

DexterChannel to open.

error :

return location for optional GError.

Returns :

gboolean indicating success or failure.

dexter_channel_close ()

gboolean            dexter_channel_close                (DexterChannel *channel,
                                                         GError **error);

Closes a DexterChannel previously opened with dexter_channel_open(). If the channel is already closed, no error occurs and TRUE is returned.

channel :

DexterChannel to close.

error :

return location for optional GError.

Returns :

gboolean indicating success or failure.

dexter_channel_greet ()

gboolean            dexter_channel_greet                (DexterChannel *channel,
                                                         GError **error);

Sends the protocol greeting on a DexterChannel previously opened with dexter_channel_open(). The greeting must be sent before using the channel, e.g. to create samplers.

channel :

DexterChannel to send greeting.

error :

return location for optional GError.

Returns :

gboolean indicating success or failure.

dexter_channel_noop ()

gboolean            dexter_channel_noop                 (DexterChannel *channel,
                                                         GError **error);

Sends a "no operation" command over the DexterChannel to the connected server.

channel :

DexterChannel to send noop.

error :

return location for optional GError.

Returns :

gboolean indicating success or failure.

Since 0.2.0


dexter_channel_starttls ()

gboolean            dexter_channel_starttls             (DexterChannel *channel,
                                                         GError **error);

Initializes TLS over the currently opened channel.

channel :

DexterChannel to send start tls request.

error :

return location for optional GError.

Returns :

gboolean indicating success or failure.

Since 0.2.0


dexter_channel_part ()

gboolean            dexter_channel_part                 (DexterChannel *channel,
                                                         GError **error);

Sends protocol parting message on a DexterChannel over which dexter_channel_greet() has already been called. Should be called prior to dexter_channel_close().

channel :

DexterChannel to send parting message.

error :

return location for optional GError.

Returns :

gboolean indicating success or failure.

dexter_channel_free ()

void                dexter_channel_free                 (DexterChannel *channel);

Frees a DexterChannel created with dexter_channel_new(). If the channel is not closed, it will be closed for you, although it is better form to use dexter_channel_close() before freeing the channel.

channel :

DexterChannel to free.

dexter_channel_get_state ()

gint                dexter_channel_get_state            (DexterChannel *channel);

Returns the current state of the channel.

channel :

a DexterChannel.

Returns :

gint from DexterChannelState indicating current state of the channel.