Repository
Munin (contrib)
Last change
2018-08-02
Graph Categories
Family
contrib
Capabilities
Keywords
Language
Bash
License
GPL-2.0-only

zfs_arcstats

Example graph: 1 Example graph: 2

Name

zfs_arcstats - Munin multi-graph plugin to monitor ZFS ARC statistics

These functions are implemented:
  size     : to monitor ARC size
  activity : to monitor ARC activities
  actlist  : to monitor ARC activities by cache list (MFU/MRU)
  actdata  : to monitor ARC activities by data type (Demand/Prefetch)
  hitratio : to monitor ARC hit ratio

Tested with Solaris 10 and 11, OpenIndiana Hipster, FreeBSD 11, CentOS 7
This plugin is inspired by arcstat.pl [https://github.com/mharsch/arcstat]

Configuration

Make symlink:
  cd /path/to/munin/etc/plugins
  ln -s /path/to/munin/lib/plugins/zfs_arcstats .

For FreeBSD, it should be necessary to change shebang /bin/bash -> /usr/local/bin/bash

Environment Variables

None

Author

K.Cima https://github.com/shakemid

License

GPLv2

Magic Markers

#%# family=contrib
#%# capabilities=autoconf
#!/bin/bash

: << =cut

=head1 NAME

  zfs_arcstats - Munin multi-graph plugin to monitor ZFS ARC statistics

  These functions are implemented:
    size     : to monitor ARC size
    activity : to monitor ARC activities
    actlist  : to monitor ARC activities by cache list (MFU/MRU)
    actdata  : to monitor ARC activities by data type (Demand/Prefetch)
    hitratio : to monitor ARC hit ratio

  Tested with Solaris 10 and 11, OpenIndiana Hipster, FreeBSD 11, CentOS 7
  This plugin is inspired by arcstat.pl [https://github.com/mharsch/arcstat]

=head1 CONFIGURATION

  Make symlink:
    cd /path/to/munin/etc/plugins
    ln -s /path/to/munin/lib/plugins/zfs_arcstats .

  For FreeBSD, it should be necessary to change shebang /bin/bash -> /usr/local/bin/bash

=head1 ENVIRONMENT VARIABLES

  None

=head1 AUTHOR

  K.Cima https://github.com/shakemid

=head1 LICENSE

  GPLv2

=head1 Magic markers

  #%# family=contrib
  #%# capabilities=autoconf

=cut

# Include plugin.sh
. "${MUNIN_LIBDIR:-}/plugins/plugin.sh"
is_multigraph "$@"

# Shell options
set -o nounset

# Set global variables
plugin_name=zfs_arcstats
functions='size activity actlist actdata hitratio'

# Functions

get_osname() {
    local osname osver

    osname=$( uname -s )
    osver=$( uname -v )

    case $osname in
    SunOS)
        case $osver in
        illumos*)
            osname=illumos
        ;;
        esac
    ;;
    esac

    echo "$osname"
}

preconfig() {
    local func=$1

    # data_attr format: field type draw label
    #   label can contain white-spaces.

    case $func in
    size)
        global_attr="
            graph_title ZFS ARC - Size
            graph_category fs
            graph_args --base 1024 --lower-limit 0
            graph_vlabel Bytes
            graph_info ZFS ARC - Size
        "
        case $osname in
        SunOS)
            # For Solaris 10,11
            data_attr="
                data_size          GAUGE AREASTACK Data size
                prefetch_meta_size GAUGE AREASTACK Prefetch meta size
                buf_size           GAUGE AREASTACK Buf size
                other_size         GAUGE AREASTACK Other size
            "
            ;;
        *)
            # For illumos, FreeBSD, Linux (OpenZFS)
            data_attr="
                data_size     GAUGE AREASTACK Data size
                metadata_size GAUGE AREASTACK Metadata size
                hdr_size      GAUGE AREASTACK Hdr size
                other_size    GAUGE AREASTACK Other size
                mru_size      GAUGE LINE      MRU size
                mfu_size      GAUGE LINE      MFU size
            "
            ;;
        esac
        data_attr="
            $data_attr
            size GAUGE LINE ARC size
            c    GAUGE LINE Target size
            p    GAUGE LINE Target MRU size
        "
        ;;
    activity)
        global_attr="
            graph_title ZFS ARC - Activities
            graph_category fs
            graph_args --base 1000 --lower-limit 0
            graph_vlabel misses (-) / hits (+) per second
            graph_info ZFS ARC - Activities

            hits.negative misses
            l2_hits.negative l2_misses
        "
        data_attr="
            misses    DERIVE LINE dummy
            hits      DERIVE LINE ARC
            l2_misses DERIVE LINE dummy
            l2_hits   DERIVE LINE L2ARC
        "
        ;;
    actlist)
        global_attr="
            graph_title ZFS ARC - Activities by cache list
            graph_category fs
            graph_args --base 1000 --lower-limit 0
            graph_vlabel ghost hits (-) / hits (+) per second
            graph_info ZFS ARC - Activities by cache list

            mfu_hits.negative mfu_ghost_hits
            mru_hits.negative mru_ghost_hits
        "
        data_attr="
            mfu_ghost_hits DERIVE LINE dummy
            mfu_hits       DERIVE LINE MFU
            mru_ghost_hits DERIVE LINE dummy
            mru_hits       DERIVE LINE MRU
        "
        ;;
    actdata)
        global_attr="
            graph_title ZFS ARC - Activities by data type
            graph_category fs
            graph_args --base 1000 --lower-limit 0
            graph_vlabel misses (-) / hits (+) per second
            graph_info ZFS ARC - Activities by data type

            demand_data_hits.negative demand_data_misses
            demand_metadata_hits.negative demand_metadata_misses
            prefetch_data_hits.negative prefetch_data_misses
            prefetch_metadata_hits.negative prefetch_metadata_misses
        "
        data_attr="
            demand_data_misses       DERIVE LINE dummy
            demand_data_hits         DERIVE LINE D data
            demand_metadata_misses   DERIVE LINE dummy
            demand_metadata_hits     DERIVE LINE D meta
            prefetch_data_misses     DERIVE LINE dummy
            prefetch_data_hits       DERIVE LINE P data
            prefetch_metadata_misses DERIVE LINE dummy
            prefetch_metadata_hits   DERIVE LINE P meta
        "
        ;;
    hitratio)
        global_attr="
            graph_title ZFS ARC - Hit ratio
            graph_category fs
            graph_args --base 1000 --lower-limit 0 --upper-limit 100 --rigid
            graph_vlabel % hits
            graph_info ZFS ARC - Hit ratio - The graph shows cache hit ratio between munin-update intervals (usually 5 minutes).

            hitratio.cdef                   hits,DUP,misses,+,/,100,*
            l2_hitratio.cdef                l2_hits,DUP,l2_misses,+,/,100,*
            demand_data_hitratio.cdef       demand_data_hits,DUP,demand_data_misses,+,/,100,*
            demand_metadata_hitratio.cdef   demand_metadata_hits,DUP,demand_metadata_misses,+,/,100,*
            prefetch_data_hitratio.cdef     prefetch_data_hits,DUP,prefetch_data_misses,+,/,100,*
            prefetch_metadata_hitratio.cdef prefetch_metadata_hits,DUP,prefetch_metadata_misses,+,/,100,*
        "
        data_attr="
            hits                       DERIVE LINE  dummy
            misses                     DERIVE LINE  dummy
            l2_hits                    DERIVE LINE  dummy
            l2_misses                  DERIVE LINE  dummy
            demand_data_hits           DERIVE LINE  dummy
            demand_data_misses         DERIVE LINE  dummy
            demand_metadata_hits       DERIVE LINE  dummy
            demand_metadata_misses     DERIVE LINE  dummy
            prefetch_data_hits         DERIVE LINE  dummy
            prefetch_data_misses       DERIVE LINE  dummy
            prefetch_metadata_hits     DERIVE LINE  dummy
            prefetch_metadata_misses   DERIVE LINE  dummy
            hitratio                   GAUGE  LINE2 ARC hits
            l2_hitratio                GAUGE  LINE  L2ARC hits
            demand_data_hitratio       GAUGE  LINE  Demand data hits
            demand_metadata_hitratio   GAUGE  LINE  Demand metadata hits
            prefetch_data_hitratio     GAUGE  LINE  Prefetch data hits
            prefetch_metadata_hitratio GAUGE  LINE  Prefetch metadata hits
        "
        ;;
    *)
        echo "Unknown function: $func"
        exit 1
        ;;
    esac
}

do_config() {
    local func=$1
    local label_max_length=45
    local field type draw label

    preconfig "$func"
    echo "multigraph ${plugin_name}_${func}"

    # print global attributes
    echo "$global_attr" | sed -e 's/^  *//' -e '/^$/d'

    # print data source attributes
    echo "$data_attr" | while read -r field type draw label
    do
        [ -z "$field" ] && continue

        echo "${field}.type ${type}"
        echo "${field}.draw ${draw}"
        echo "${field}.label ${label:0:${label_max_length}}"
        if [ "$type" = 'DERIVE' ]; then
            echo "${field}.min 0"
        fi
        if [ "$label" = 'dummy' ]; then
            echo "${field}.graph no"
        fi
    done

    echo
}

get_stats() {
    local arcstats stat value

    case $osname in
    SunOS|illumos)
        arcstats=$( kstat -p 'zfs:0:arcstats' | sed -e 's/:/ /g' | awk '{ print $4,$5 }' )
        # kstat output example:
        #   $ kstat -p zfs:0:arcstats
        #   zfs:0:arcstats:c        4135233544
        #   ...
        ;;
    *BSD)
        arcstats=$( /sbin/sysctl -a | sed -n -e 's/^kstat\.zfs\.misc\.arcstats\.//p' | awk -F: '{ print $1,$2 }' )
        # sysctl output example:
        #   $ sysctl -a
        #   ...
        #   kstat.zfs.misc.arcstats.c: 632540160
        #   ...
        ;;
    Linux)
        arcstats=$( sed '1,2d' /proc/spl/kstat/zfs/arcstats | awk '{ print $1,$3 }' )
        # proc file output example:
        #   $ cat /proc/spl/kstat/zfs/arcstats
        #   ...
        #   name        type  data
        #   hits        4     62
        #   ...
        ;;
    *)
        echo "Unsupported OS: $osname"
        exit 1
    esac

    while read -r stat value
    do
        printf -v "arcstats_${stat}" "%s" "$value"
        # printf -v means indirect variable assignment (similar to eval)
    done <<< "$arcstats"
}

do_fetch() {
    local func=$1
    local field type draw label value ref

    preconfig "$func"
    echo "multigraph ${plugin_name}_${func}"

    echo "$data_attr" | while read -r field type draw label
    do
        [ -z "$field" ] && continue

        ref="arcstats_${field}"
        value=${!ref:-0}
        # ${!varname} means indirect evaluation (similar to eval)

        echo "${field}.value ${value}"
    done

    echo
}

autoconf() {
    if [ -x /sbin/zfs ]; then
        echo yes
    else
        echo "no (ZFS looks unavailable)"
    fi
}

config() {
    local func

    for func in $functions
    do
        do_config "$func"
    done
}

fetch() {
    local func

    get_stats

    for func in $functions
    do
        do_fetch "$func"
    done
}

# Main

osname=$( get_osname )

case ${1:-} in
autoconf)
    autoconf
    ;;
config)
    config
    if [ "${MUNIN_CAP_DIRTYCONFIG:-0}" = "1" ]; then fetch; fi
    ;;
*)
    fetch
    ;;
esac

exit 0