# # collectd - bindings/buildperl/Collectd/Unixsock.pm # Copyright (C) 2007,2008 Florian octo Forster # # Permission is hereby granted, free of charge, to any person obtaining a # copy of this software and associated documentation files (the "Software"), # to deal in the Software without restriction, including without limitation # the rights to use, copy, modify, merge, publish, distribute, sublicense, # and/or sell copies of the Software, and to permit persons to whom the # Software is furnished to do so, subject to the following conditions: # # The above copyright notice and this permission notice shall be included in # all copies or substantial portions of the Software. # # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR # IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, # FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE # AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER # LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING # FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER # DEALINGS IN THE SOFTWARE. # # Authors: # Florian Forster # package Collectd::Unixsock; =head1 NAME Collectd::Unixsock - Abstraction layer for accessing the functionality by collectd's unixsock plugin. =head1 SYNOPSIS use Collectd::Unixsock; my $sock = Collectd::Unixsock->new ($path); my $value = $sock->getval (%identifier); $sock->putval (%identifier, time => time (), values => [123, 234, 345]); $sock->destroy (); =head1 DESCRIPTION collectd's unixsock plugin allows external programs to access the values it has collected or received and to submit own values. This Perl-module is simply a little abstraction layer over this interface to make it even easier for programmers to interact with the daemon. =cut use strict; use warnings; use Carp qw(cluck confess carp croak); use POSIX; use IO::Socket::UNIX; use Scalar::Util qw( looks_like_number ); our $Debug = 0; sub _debug { print @_ if $Debug; } sub _create_socket { my $path = shift; my $sock = IO::Socket::UNIX->new (Type => SOCK_STREAM, Peer => $path); if (!$sock) { cluck ("Cannot open UNIX-socket $path: $!"); return; } return ($sock); } # _create_socket =head1 VALUE IDENTIFIERS The values in the collectd are identified using a five-tuple (host, plugin, plugin-instance, type, type-instance) where only plugin instance and type instance may be undef. Many functions expect an I<%identifier> hash that has at least the members B, B, and B, possibly completed by B and B. Usually you can pass this hash as follows: $self->method (host => $host, plugin => $plugin, type => $type, %other_args); =cut sub _create_identifier { my $args = shift; my ($host, $plugin, $type); if (!$args->{host} || !$args->{plugin} || !$args->{type}) { cluck ("Need `host', `plugin' and `type'"); return; } $host = $args->{host}; $plugin = $args->{plugin}; $plugin .= '-' . $args->{plugin_instance} if defined $args->{plugin_instance}; $type = $args->{type}; $type .= '-' . $args->{type_instance} if defined $args->{type_instance}; return "$host/$plugin/$type"; } # _create_identifier sub _parse_identifier { my $string = shift; my ($plugin_instance, $type_instance); my ($host, $plugin, $type) = split /\//, $string; ($plugin, $plugin_instance) = split /-/, $plugin, 2; ($type, $type_instance) = split /-/, $type, 2; my $ident = { host => $host, plugin => $plugin, type => $type }; $ident->{plugin_instance} = $plugin_instance if defined $plugin_instance; $ident->{type_instance} = $type_instance if defined $type_instance; return $ident; } # _parse_identifier sub _escape_argument { my $arg = shift; return $arg if $arg =~ /^\w+$/; $arg =~ s#\\#\\\\#g; $arg =~ s#"#\\"#g; return "\"$arg\""; } # Handle socket errors. sub _socket_error { my ($self, $where) = @_; # If the peer has reset the connection, try to reconnect, # otherwise fail. if ($! == EPIPE) { _debug "^^ error on $where: $!; reconnecting\n"; $self->destroy; $self->{sock} = _create_socket ($self->{path}) or return 1; return; } else { carp ("error on $where: $!; aborting action\n"); $self->{error} = $!; return 1; } } # Send a command on a socket, including any required argument escaping. # Return a single line of result. sub _socket_command { my ($self, $command, $args) = @_; my $fh = $self->{sock} or confess ('object has no filehandle'); if($args) { my $identifier = _create_identifier ($args) or return; $command .= ' ' . _escape_argument ($identifier) . "\n"; } else { $command .= "\n"; } _debug "-> $command"; while (not $fh->print($command)) { return if $self->_socket_error ('print'); $fh = $self->{sock}; } my $response; while (not defined ($response = $fh->getline)) { return if $self->_socket_error ('getline'); $fh = $self->{sock}; } chomp $response; _debug "<- $response\n"; return $response; } # Read any remaining results from a socket and pass them to # a callback for caller-defined mangling. sub _socket_chat { my ($self, $msg, $callback, $cbdata) = @_; my ($nresults, $ret); my $fh = $self->{sock} or confess ('object has no filehandle'); ($nresults, $msg) = split / /, $msg, 2; if ($nresults <= 0) { $self->{error} = $msg; return; } for (1 .. $nresults) { my $entry; while (not defined($entry = $fh->getline)) { return if $self->_socket_error ('getline'); $fh = $self->{sock}; } chomp $entry; _debug "<- $entry\n"; $callback->($entry, $cbdata); } return $cbdata; } # Send a raw message on a socket. # Returns true upon success and false otherwise. sub _send_message { my ($self, $msg) = @_; my $fh = $self->{'sock'} or confess ('object has no filehandle'); $msg .= "\n" unless $msg =~/\n$/; #1024 is default buffer size at unixsock.c us_handle_client() warn "Collectd::Unixsock->_send_message(\$msg): message is too long!" if length($msg) > 1024; _debug "-> $msg"; while (not $fh->print($msg)) { return if $self->_socket_error ('print'); $fh = $self->{sock}; } while (not defined ($msg = <$fh>)) { return if $self->_socket_error ('readline'); $fh = $self->{sock}; } chomp ($msg); _debug "<- $msg\n"; my ($status, $error) = split / /, $msg, 2; return 1 if $status == 0; $self->{error} = $error; return; } =head1 PUBLIC METHODS =over 4 =item I<$self> = Collectd::Unixsock->B ([I<$path>]); Creates a new connection to the daemon. The optional I<$path> argument gives the path to the UNIX socket of the C and defaults to F. Returns the newly created object on success and false on error. =cut sub new { my $class = shift; my $path = shift || '/var/run/collectd-unixsock'; my $sock = _create_socket ($path) or return; return bless { path => $path, sock => $sock, error => 'No error' }, $class; } # new =item I<$res> = I<$self>-EB (I<%identifier>); Requests a value-list from the daemon. On success a hash-ref is returned with the name of each data-source as the key and the according value as, well, the value. On error false is returned. =cut sub getval # {{{ { my $self = shift; my %args = @_; my $ret = {}; my $msg = $self->_socket_command('GETVAL', \%args) or return; $self->_socket_chat($msg, sub { local $_ = shift; my $ret = shift; /^(\w+)=NaN$/ and $ret->{$1} = undef, return; /^(\w+)=(.*)$/ and looks_like_number($2) and $ret->{$1} = 0 + $2, return; }, $ret ); return $ret; } # }}} sub getval =item I<$res> = I<$self>-EB (I<%identifier>); Requests a threshold from the daemon. On success a hash-ref is returned with the threshold data. On error false is returned. =cut sub getthreshold # {{{ { my $self = shift; my %args = @_; my $ret = {}; my $msg = $self->_socket_command('GETTHRESHOLD', \%args) or return; $self->_socket_chat($msg, sub { local $_ = shift; my $ret = shift; my ( $key, $val ); ( $key, $val ) = /^\s*([^:]+):\s*(.*)/ and do { $key =~ s/\s*$//; $ret->{$key} = $val; }; }, $ret ); return $ret; } # }}} sub getthreshold =item I<$self>-EB (I<%identifier>, B