RGraph - Как правильно обновить линейный график и выполнять боковую прокрутку

Я пытаюсь получить динамически обновляемый линейный график боковой прокрутки для работы в rgraph. Он обновляется раз в секунду через ajax-вызов к php-скрипту, который выводит данные в следующем формате:

  {"cpu":28.43,"mem":75.01,"rx":0,"timestamp":1471535457240}

Прямо сейчас с помощью следующего кода я просто пытаюсь заставить его работать только с данными процессора. Вызовы ajax работают раз в секунду, но я не могу обновить график. Данные предварительно заполняются циклом for.

Что я делаю неправильно?

     var cpu = [], mem = [], rx = [], timestamp = [];
    var dataset;
    var totalPoints = 100;
    var updateInterval = 1000;
    var now = new Date().getTime();
    var canvas    = document.getElementById("cvs");
    var temp = [];

    // Make and draw the chart
    var line = new RGraph.Line({
        id: 'cvs',
        data: cpu,
        options: {
            ymax: 100,
            backgroundGridAutofitNumvlines: 10,
            backgroundGridVlines: true,
            backgroundGridBorder: true,
            numxticks: 10,
            tickmarks: true,
            noaxes: false,
            textSize: 12,
            gutterLeft: 50,
            textColor: '#aaa',
            scaleZerostart: true,
            textAccessible: true
        }
    }).draw();


    function initData() {
        now -= totalPoints * updateInterval;
        for (var i = 0; i < totalPoints; i++) {
           // temp = [now += updateInterval, 0];
            cpu[i] = 0;
        }
    }



    function GetData() {
        $.ajaxSetup({ cache: false });

        $.ajax({
            url: "server_load.php",
            dataType: 'json',
            success: function (_data) {
                drawGraph(_data);
            },
            error: function () {
                setTimeout(GetData, updateInterval);
            }
        });
    }




    function drawGraph(_data) {

        console.log(cpu);
        cpu.shift();
        var Now = _data['timestamp'];

        temp = [Now, _data.cpu];
        cpu.push(temp);

        console.log(temp[1]);

        // Reset the canvas
        line.reset(document.getElementById('cvs'));

        line.original_data[0] = temp[1];
        line.draw();


        setTimeout(GetData, updateInterval);
    }



    window.onload = function ()
    {
        initData();
        GetData();
    };

Вот мой php-скрипт, который работает на моей локальной машине с Ubuntu:

   class serverLoad {

const OS_LINUX = "Linux";
const OS_FREEBSD = "FreeBSD";
const OS_OSX = "Darwin";
const PATH_NETSTAT = "/usr/sbin/netstat";

public $cpuLoad;
public $memoryUsed;
public $timeStamp;
public $networkBandwidth;
public $interface = 'eth0';
protected $sampleSleep = 1000000;

public function __construct()
{
    $this->generateData();
}

/**
 *
 */

public function generateData(){

    $this->cpuLoad = $this->getServerLoad();
    $this->memoryUsed = $this->getSystemMemoryInfo();
    $this->networkBandwidth = $this->getNetworkBandwidth();
    $this->timeStamp = $this->msTimeStamp();


    $totalMemory = $this->memoryUsed['MemTotal'];
    $freeMemory = $this->memoryUsed['MemTotal'] - $this->memoryUsed['MemFree'] - $this->memoryUsed['Buffers'] - $this->memoryUsed['Cached'];
    $MemoryPercentage =  round(($freeMemory/$totalMemory) * 100, 2) ;

    $results = array('cpu' => $this->cpuLoad, 'mem' => $MemoryPercentage, 'rx' => $this->networkBandwidth['rx'], 'timestamp' => $this->timeStamp);

    echo json_encode($results);
}

/**
 *  Returns server load in percent (just number, without percent sign)
 */

protected function getServerLoad()
{
    $load = null;

    if (stristr(PHP_OS, "win"))
    {
        $cmd = "wmic cpu get loadpercentage /all";
        @exec($cmd, $output);

        if ($output)
        {
            foreach ($output as $line)
            {
                if ($line && preg_match("/^[0-9]+\$/", $line))
                {
                    $load = $line;
                    break;
                }
            }
        }
    }
    else
    {
        if (is_readable("/proc/stat"))
        {
            // Collect 2 samples - each with 1 second period
            // See: https://de.wikipedia.org/wiki/Load#Der_Load_Average_auf_Unix-Systemen
            $statData1 = $this->_getServerLoadLinuxData();
            usleep($this->sampleSleep);
            $statData2 = $this->_getServerLoadLinuxData();

            if
            (
                (!is_null($statData1)) &&
                (!is_null($statData2))
            )
            {
                // Get difference
                $statData2[0] -= $statData1[0];
                $statData2[1] -= $statData1[1];
                $statData2[2] -= $statData1[2];
                $statData2[3] -= $statData1[3];

                // Sum up the 4 values for User, Nice, System and Idle and calculate
                // the percentage of idle time (which is part of the 4 values!)
                $cpuTime = $statData2[0] + $statData2[1] + $statData2[2] + $statData2[3];

                // Invert percentage to get CPU time, not idle time
                $load = 100 - ($statData2[3] * 100 / $cpuTime);
            }
        }
    }

    return round($load, 2);
}

/**
 *  Used to get the server load on linux machines
 */

protected function _getServerLoadLinuxData()
{
    if (is_readable("/proc/stat"))
    {
        $stats = @file_get_contents("/proc/stat");

        if ($stats !== false)
        {
            // Remove double spaces to make it easier to extract values with explode()
            $stats = preg_replace("/[[:blank:]]+/", " ", $stats);

            // Separate lines
            $stats = str_replace(array("\r\n", "\n\r", "\r"), "\n", $stats);
            $stats = explode("\n", $stats);

            // Separate values and find line for main CPU load
            foreach ($stats as $statLine)
            {
                $statLineData = explode(" ", trim($statLine));

                // Found!
                if
                (
                    (count($statLineData) >= 5) &&
                    ($statLineData[0] == "cpu")
                )
                {
                    return array(
                        $statLineData[1],
                        $statLineData[2],
                        $statLineData[3],
                        $statLineData[4],
                    );
                }
            }
        }
    }

    return null;
}


protected function wmiWBemLocatorQuery( $query ) {
    if ( class_exists( '\\COM' ) ) {
        try {
            $WbemLocator = new \COM( "WbemScripting.SWbemLocator" );
            $WbemServices = $WbemLocator->ConnectServer( '127.0.0.1', 'root\CIMV2' );
            $WbemServices->Security_->ImpersonationLevel = 3;
            // use wbemtest tool to query all classes for namespace root\cimv2
            return $WbemServices->ExecQuery( $query );
        } catch ( \com_exception $e ) {
            echo $e->getMessage();
        }
    } elseif ( ! extension_loaded( 'com_dotnet' ) )
        trigger_error( 'It seems that the COM is not enabled in your php.ini', E_USER_WARNING );
    else {
        $err = error_get_last();
        trigger_error( $err['message'], E_USER_WARNING );
    }

    return false;
}

// _dir_in_allowed_path this is your function to detect if a file is withing the allowed path (see the open_basedir PHP directive)
protected function getSystemMemoryInfo( $output_key = '' ) {
    $keys = array( 'MemTotal', 'MemFree', 'MemAvailable', 'SwapTotal', 'SwapFree', 'Buffers', 'Cached' );
    $result = array();
    $isWin = false;

    try {
        // LINUX
        if ( !$isWin ) {
            $proc_dir = '/proc/';
            $data =  @file( $proc_dir . 'meminfo' );
            if ( is_array( $data ) )
                foreach ( $data as $d ) {
                    if ( 0 == strlen( trim( $d ) ) )
                        continue;
                    $d = preg_split( '/:/', $d );
                    $key = trim( $d[0] );
                    if ( ! in_array( $key, $keys ) )
                        continue;
                    $value = 1000 * floatval( trim( str_replace( ' kB', '', $d[1] ) ) );
                    $result[$key] = $value;
                }
        } else      // WINDOWS
        {
            $wmi_found = false;
            if ( $wmi_query = wmiWBemLocatorQuery(
                "SELECT FreePhysicalMemory,FreeVirtualMemory,TotalSwapSpaceSize,TotalVirtualMemorySize,TotalVisibleMemorySize FROM Win32_OperatingSystem" ) ) {
                foreach ( $wmi_query as $r ) { // TODO add Buffers and cached to this loop
                    $result['MemFree'] = $r->FreePhysicalMemory * 1024;
                    $result['MemAvailable'] = $r->FreeVirtualMemory * 1024;
                    $result['SwapFree'] = $r->TotalSwapSpaceSize * 1024;
                    $result['SwapTotal'] = $r->TotalVirtualMemorySize * 1024;
                    $result['MemTotal'] = $r->TotalVisibleMemorySize * 1024;
                    $wmi_found = true;
                }
            }
            // TODO a backup implementation using the $_SERVER array
        }
    } catch ( Exception $e ) {
        echo $e->getMessage();
    }
    return empty( $output_key ) || ! isset( $result[$output_key] ) ? $result : $result[$output_key];
}


 /**
  *  Returns network bandwidth usage
  */
protected function getNetworkBandwidth(){

    $rx[] = $this->getInterfaceReceivedBytes($this->interface);
    $tx[] = $this->getInterfaceSentBytes($this->interface);
    usleep($this->sampleSleep);
    $rx[] = $this->getInterfaceReceivedBytes($this->interface);
    $tx[] = $this->getInterfaceSentBytes($this->interface);

    $tbps = $tx[1] - $tx[0];
    $rbps = $rx[1] - $rx[0];
    //convert to kbps
    $bandwidth['rx'] = round(($rbps * 8) / 1000, 2);
    $bandwidth['tx'] = round(($tbps * 8) / 1000, 2);

    return $bandwidth;
}


protected function getInterfaceReceivedBytes($interface)
{
    if ($this->isLinux()) {
        $filepath = '/sys/class/net/%s/statistics/rx_bytes';
        $output = file_get_contents(sprintf($filepath, $interface));

        return $output;
    }

    if ($this->isFreeBSD() || $this->isOSX()) {
        $command = "%s -ibn| grep %s | grep Link | awk '{print $7}'";
        exec(sprintf($command, PATH_NETSTAT, $interface), $output);

        return array_shift($output);
    }

    throw new Exception('Unable to guess OS');
}

protected function getInterfaceSentBytes($interface)
{
    if ($this->isLinux()) {
        $filepath = '/sys/class/net/%s/statistics/tx_bytes';
        $output = file_get_contents(sprintf($filepath, $interface));

        return $output;
    }

    if ($this->isFreeBSD() || $this->isOSX()) {
        $command = "%s -ibn| grep %s | grep Link | awk '{print $10}'";
        exec(sprintf($command, PATH_NETSTAT, $interface), $output);

        return array_shift($output);
    }

    throw new Exception('Unable to guess OS');
}


protected function isOSX()
{
    $uname = explode(' ', php_uname());
    return $uname[0] === self::OS_OSX;
}

protected function isLinux()
{
    $uname = explode(' ', php_uname());
    return $uname[0] === self::OS_LINUX;
}

protected function isFreeBSD()
{
    $uname = explode(' ', php_uname());
    return $uname[0] === self::OS_FREEBSD;
}


/**
*  Returns current unix timestamp
*/
protected function msTimeStamp() {
    return round(microtime(1) * 1000);
}


 }

 $data = new serverLoad();

Я также настроил jsfiddle, но не уверен, как заставить его работать с вызовом ajax: http://fiddle.jshell.net/Vbgkk/18/

0 ответов

Другие вопросы по тегам