gpt4 book ai didi

linux - 在 html/php 输出中显示/var/log/messages?

转载 作者:太空宇宙 更新时间:2023-11-04 09:25:21 25 4
gpt4 key购买 nike

我试图在我通过同一主机上的网络服务器访问的网页中显示/var/log/messages 或类似(例如../secure)的输出。

我应该使用 bash 来 tail -f >> 消息文件到一个新的输出文件并在 html 页面中显示该文本文件,还是有更好的方法来做到这一点?

谢谢!idiglivemusic

最佳答案

如果您正在寻找一种无需重新加载页面即可在线显示实际文件内容的方法,那么您应该设置一个 WebSocket服务器。

您可以使用 phpDaemon 等框架构建 WebSocket 服务器, ReactPHP , Ratchet , icicle ,或借助包装异步库的 PHP 扩展实现您自己的服务器:event , ev ,或类似的。

我从上面的列表中随机选择了一个框架:Ratchet。 Ratchet 基于 ReactPHP。 ReactPHP 从以下列表中为事件循环选择后端:- libevent 扩展,- libev 扩展,- 事件扩展,- 或基于内置 stream_select() 函数的内部类。

作为 event 扩展的维护者,我选择了 event

我写了一个“快速”示例,只是为了让您了解如何实现它。您很可能必须制定自己的版本,也许使用不同的工具。但是代码应该会给你一个冲动。

src/MyApp/Server.php

<?php
namespace MyApp;
use Ratchet\MessageComponentInterface;
use Ratchet\ConnectionInterface;

class Server implements MessageComponentInterface {
protected $clients;

public function __construct() {
$this->clients = new \SplObjectStorage;
}

public function onOpen(ConnectionInterface $conn) {
$this->clients->attach($conn);
echo "New connection! ({$conn->resourceId})\n";
}

public function onMessage(ConnectionInterface $from, $msg) {
$numRecv = count($this->clients) - 1;
printf("Connection %d sending '%s' to %d other connection%s\n",
$from->resourceId, $msg, $numRecv, $numRecv == 1 ? '' : 's');

foreach ($this->clients as $client) {
if ($from !== $client) {
$client->send($msg);
}
}
}

public function onClose(ConnectionInterface $conn) {
$this->clients->detach($conn);
echo "Connection {$conn->resourceId} has disconnected\n";
}

public function onError(ConnectionInterface $conn, \Exception $e) {
echo "An error has occurred: {$e->getMessage()}\n";
$conn->close();
}

public function broadcast($msg) {
foreach ($this->clients as $client) {
$client->send($msg);
}
}
}

server.php

<?php
use Ratchet\Server\IoServer;
use Ratchet\Http\HttpServer;
use Ratchet\WebSocket\WsServer;
use MyApp\Server;

require __DIR__ . '/vendor/autoload.php';

$server = IoServer::factory(
new HttpServer(
new WsServer(
$my_app_server = new Server()
)
),
9989
);

$loop = $server->loop;
$filename = '/var/log/messages';

$loop->addPeriodicTimer(5, function ()
use ($filename, $my_app_server)
{
static $stat_info;

if ($stat_info == null) {
clearstatcache(true, $filename);
$stat_info = stat($filename);
}

clearstatcache(true, $filename);
$st = stat($filename);

$size_diff = $st['size'] - $stat_info['size'];
echo "Diff = $size_diff bytes\n";

if ($size_diff > 0) {
$offset = $stat_info['size'];
$bytes = $size_diff;
} elseif ($size_diff < 0) {
// The file is likely truncated by `logrotate` or similar utility
$offset = 0;
$bytes = $st['size'];
} else {
$bytes = 0;
}

$stat_info = $st;

if ($bytes) {
if (! $fp = fopen($filename, 'r')) {
fprintf(STDERR, "Failed to open file $filename\n");
return;
}
if ($offset > 0) {
fseek($fp, $offset);
}
if ($msg = fread($fp, $bytes)) {
$my_app_server->broadcast($msg);
}
fclose($fp);
}
}
);

$server->run();

test.html

<html>
<head>
<meta http-equiv="content-type" content="text/html; charset=utf-8">
<title>Test</title>
</head>
<body>
<script>
var conn = new WebSocket('ws://localhost:9989');
conn.onopen = function(e) {
console.log("Connection established!");
};
conn.onmessage = function(e) {
console.log("Msg from server", e.data);
};
</script>
</body>
</html>

我将跳过使用 Composer 设置基本测试环境所需的步骤.假设您已成功为上述文件配置测试环境,您将能够使用以下命令运行服务器:

php server.php

检查用户是否有权读取/var/log/messages。在我的系统上,只有 root 可以读取该文件。因此,您可能需要使用 sudo(root 权限)运行上述命令。

现在您可以在浏览器中打开 test.html 并查看控制台输出。然后触发一些通常记录到消息文件中的事件。例如,您可以使用错误的密码调用 sudo。服务器应在 5 秒的间隔内检测到变化,然后将其发送给 WebSocket 客户端。

关于linux - 在 html/php 输出中显示/var/log/messages?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/37107706/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com