我试图在我通过同一主机上的网络服务器访问的网页中显示/ var / log / messages或类似的信息(例如../secure)。
我应该使用bash尾巴-f>>消息文件到新的输出文件并在html页面中显示该文本文件,或者有更好的方法来执行此操作吗?
谢谢! idiglivemusic
答案 0 :(得分:2)
如果您正在寻找一种在线显示实际文件内容而无需重新加载页面的方法,那么您应该设置一个WebSocket服务器。
您可以使用phpDaemon,ReactPHP,Ratchet,icicle之类的框架构建WebSocket服务器,或者在PHP扩展包装的帮助下实现您自己的服务器异步库:event,ev或类似的。
我从上面的列表中选择了一个随机框架:Ratchet。 Ratchet基于ReactPHP。 ReactPHP从以下列表中为事件循环选择后端:
- libevent
分机,
- libev
分机,
- event
分机,
- 或基于内置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客户端。
答案 1 :(得分:1)
如果你使用tail -f,这意味着当命令运行时,你会不断地从文件中获取数据。 你可以用cat或tail -n。当然,您也可以通过创建符号或硬链接来直接访问文件(ln source-file link-file,ln -s source-file link-file) - 但要确保您的Web服务器足够访问它们的权利。
答案 2 :(得分:0)
确保您的html服务器有权访问该页面并阅读该页面(例如cat
,tail
,grep
)。
在<html>
中,将输出放在<pre>
和</pre>
之间。
答案 3 :(得分:0)
方法1
在您的一个基目录中,创建一个符号链接
ln -s /var/log/messages messages
如果目录属于test.web
,请访问日志
unmesh
http://test.web/messages
方法2
如果您正在寻找php
脚本,请首先按方法1中的说明创建链接。然后,在{{1}的基目录中创建一个新文件,例如readlog.php
以下内容:
test.web
访问<?php
readfile(“$DOCUMENT_ROOT/messages”);
?>
,如:
readlog.php
<强>要求:强>
应为http://test.web/readlog.php
的所有用户启用 Read
访问权限。
注意:
为全世界/var/log/messages
设置read
选项不是一个好主意。
答案 4 :(得分:-1)
<!DOCTYPE html>
<html>
<head>
<title>toyLogs</title>
</head>
<body>
<div><p><?php include('/var/www/html/accesslog.txt'); ?></p></div>
</body>
</html>