firehol/netdata

View on GitHub
src/go/plugin/go.d/modules/postgres/do_query_databases.go

Summary

Maintainability
A
35 mins
Test Coverage
// SPDX-License-Identifier: GPL-3.0-or-later

package postgres

import (
    "fmt"
)

func (p *Postgres) doQueryDatabasesMetrics() error {
    if err := p.doQueryDatabaseStats(); err != nil {
        return fmt.Errorf("querying database stats error: %v", err)
    }
    if err := p.doQueryDatabaseSize(); err != nil {
        return fmt.Errorf("querying database size error: %v", err)
    }
    if p.isPGInRecovery() {
        if err := p.doQueryDatabaseConflicts(); err != nil {
            return fmt.Errorf("querying database conflicts error: %v", err)
        }
    }
    if err := p.doQueryDatabaseLocks(); err != nil {
        return fmt.Errorf("querying database locks error: %v", err)
    }
    return nil
}

func (p *Postgres) doQueryDatabaseStats() error {
    q := queryDatabaseStats()

    var db string
    return p.doQuery(q, func(column, value string, _ bool) {
        switch column {
        case "datname":
            db = value
            p.getDBMetrics(db).updated = true
        case "numbackends":
            p.getDBMetrics(db).numBackends = parseInt(value)
        case "datconnlimit":
            p.getDBMetrics(db).datConnLimit = parseInt(value)
        case "xact_commit":
            p.getDBMetrics(db).xactCommit = parseInt(value)
        case "xact_rollback":
            p.getDBMetrics(db).xactRollback = parseInt(value)
        case "blks_read_bytes":
            p.getDBMetrics(db).blksRead.last = parseInt(value)
        case "blks_hit_bytes":
            p.getDBMetrics(db).blksHit.last = parseInt(value)
        case "tup_returned":
            p.getDBMetrics(db).tupReturned.last = parseInt(value)
        case "tup_fetched":
            p.getDBMetrics(db).tupFetched.last = parseInt(value)
        case "tup_inserted":
            p.getDBMetrics(db).tupInserted = parseInt(value)
        case "tup_updated":
            p.getDBMetrics(db).tupUpdated = parseInt(value)
        case "tup_deleted":
            p.getDBMetrics(db).tupDeleted = parseInt(value)
        case "conflicts":
            p.getDBMetrics(db).conflicts = parseInt(value)
        case "temp_files":
            p.getDBMetrics(db).tempFiles = parseInt(value)
        case "temp_bytes":
            p.getDBMetrics(db).tempBytes = parseInt(value)
        case "deadlocks":
            p.getDBMetrics(db).deadlocks = parseInt(value)
        }
    })
}

func (p *Postgres) doQueryDatabaseSize() error {
    q := queryDatabaseSize(p.pgVersion)

    var db string
    return p.doQuery(q, func(column, value string, _ bool) {
        switch column {
        case "datname":
            db = value
        case "size":
            p.getDBMetrics(db).size = newInt(parseInt(value))
        }
    })
}

func (p *Postgres) doQueryDatabaseConflicts() error {
    q := queryDatabaseConflicts()

    var db string
    return p.doQuery(q, func(column, value string, _ bool) {
        switch column {
        case "datname":
            db = value
            p.getDBMetrics(db).updated = true
        case "confl_tablespace":
            p.getDBMetrics(db).conflTablespace = parseInt(value)
        case "confl_lock":
            p.getDBMetrics(db).conflLock = parseInt(value)
        case "confl_snapshot":
            p.getDBMetrics(db).conflSnapshot = parseInt(value)
        case "confl_bufferpin":
            p.getDBMetrics(db).conflBufferpin = parseInt(value)
        case "confl_deadlock":
            p.getDBMetrics(db).conflDeadlock = parseInt(value)
        }
    })
}

func (p *Postgres) doQueryDatabaseLocks() error {
    q := queryDatabaseLocks()

    var db, mode string
    var granted bool
    return p.doQuery(q, func(column, value string, _ bool) {
        switch column {
        case "datname":
            db = value
            p.getDBMetrics(db).updated = true
        case "mode":
            mode = value
        case "granted":
            granted = value == "true" || value == "t"
        case "locks_count":
            // https://github.com/postgres/postgres/blob/7c34555f8c39eeefcc45b3c3f027d7a063d738fc/src/include/storage/lockdefs.h#L36-L45
            // https://www.postgresql.org/docs/7.2/locking-tables.html
            switch {
            case mode == "AccessShareLock" && granted:
                p.getDBMetrics(db).accessShareLockHeld = parseInt(value)
            case mode == "AccessShareLock":
                p.getDBMetrics(db).accessShareLockAwaited = parseInt(value)
            case mode == "RowShareLock" && granted:
                p.getDBMetrics(db).rowShareLockHeld = parseInt(value)
            case mode == "RowShareLock":
                p.getDBMetrics(db).rowShareLockAwaited = parseInt(value)
            case mode == "RowExclusiveLock" && granted:
                p.getDBMetrics(db).rowExclusiveLockHeld = parseInt(value)
            case mode == "RowExclusiveLock":
                p.getDBMetrics(db).rowExclusiveLockAwaited = parseInt(value)
            case mode == "ShareUpdateExclusiveLock" && granted:
                p.getDBMetrics(db).shareUpdateExclusiveLockHeld = parseInt(value)
            case mode == "ShareUpdateExclusiveLock":
                p.getDBMetrics(db).shareUpdateExclusiveLockAwaited = parseInt(value)
            case mode == "ShareLock" && granted:
                p.getDBMetrics(db).shareLockHeld = parseInt(value)
            case mode == "ShareLock":
                p.getDBMetrics(db).shareLockAwaited = parseInt(value)
            case mode == "ShareRowExclusiveLock" && granted:
                p.getDBMetrics(db).shareRowExclusiveLockHeld = parseInt(value)
            case mode == "ShareRowExclusiveLock":
                p.getDBMetrics(db).shareRowExclusiveLockAwaited = parseInt(value)
            case mode == "ExclusiveLock" && granted:
                p.getDBMetrics(db).exclusiveLockHeld = parseInt(value)
            case mode == "ExclusiveLock":
                p.getDBMetrics(db).exclusiveLockAwaited = parseInt(value)
            case mode == "AccessExclusiveLock" && granted:
                p.getDBMetrics(db).accessExclusiveLockHeld = parseInt(value)
            case mode == "AccessExclusiveLock":
                p.getDBMetrics(db).accessExclusiveLockAwaited = parseInt(value)
            }
        }
    })
}