为 PHP 内置 WebServer 提供目录和文件索引

PHP 5.x 开始提供了一个简单的内置 WebServer,方便大家在没有安装 Apache 的情况下调试 PHP 代码。

要启动这个内置 WebServer 很简单,但它不能显示目录和文件索引,稍微有点不方便。本文提供了一个简单的实现。

使用方法:

php -S localhost:8080 webindex.php

然后打开浏览器访问 localhost:8080 即可以当前目录为根目录,显示目录列表和文件列表,并且可以在子目录中跳转。

实际运行效果:
参考来源:https://github.com/JBlond/php-built-in-webserver-router-script
朋友编写的版本:https://gist.github.com/dualface/70b62fc30b076026d0a8c87385eedebe

<?php
date_default_timezone_set('UTC');
function getfilesize($path) {
    $size = ceil(filesize($path)) . '';
    return str_repeat(' ', 8 - strlen($size)) . $size . ' KB';
}
function printlog($status = 200) {
    $time = date('D M j H:i:s Y');
    $addr = $_SERVER['REMOTE_ADDR'];
    $port = $_SERVER['REMOTE_PORT'];
    $uri = $_SERVER['REQUEST_URI'];
    $log = sprintf("[%s] %s:%s [%s]: %s\n", $time, $addr, $port, $status, $uri);
    file_put_contents('php://stdout', $log);
}
function echo_h($t) {
    echo htmlspecialchars($t);
}
function dumpfile($path) {
    $name = htmlspecialchars(pathinfo($path, PATHINFO_BASENAME));
    echo <<<EOT
<!DOCTYPE html>
<html>
<head>
<title>{$name}</title>
<meta charset="utf-8" />
<style>
pre {
    font-size: 14px;
    width: 80em;
    white-space: pre-wrap;
    white-space: -moz-pre-wrap;
    white-space: -pre-wrap;
    white-space: -o-pre-wrap;
    word-wrap: break-word;
}
</style>
</head>
<body>
<pre>
EOT;
echo_h(file_get_contents($path));
echo <<<EOT
</pre>
</body>
</html>
EOT;
}
// start
$uri = parse_url($_SERVER['REQUEST_URI'], PHP_URL_PATH);
$path = $_SERVER['DOCUMENT_ROOT'] . $uri;
if (!file_exists($path)) {
    printlog(404);
    http_response_code(404);
    return false;
}
$dump_exts = array(
    'txt', 'text', 'md', 'mdown', 'markdown', 'json',
    'c', 'cpp', 'h', 'hpp', 'm', 'mm', 'lua', 'py'
);
if (!is_dir($path)) {
    $ext = strtolower(pathinfo($path, PATHINFO_EXTENSION));
    if ($ext == '' || in_array($ext, $dump_exts)) {
        printlog();
        dumpfile($path);
        return true;
    } else {
        // let server handle file
        return false;
    }
}
$this_dir = substr($_SERVER['PHP_SELF'], 0, strrpos($_SERVER['PHP_SELF'], '/') + 1);
$dir = $_SERVER['DOCUMENT_ROOT'] . $this_dir;
if (!is_dir($dir)) {
    printlog(404);
    http_response_code(404);
    return false;
}
$folder = opendir($dir);
if (!readdir($folder)) {
    printlog(404);
    http_response_code(404);
    return false;
}
$files = array();
while ($file = readdir($folder)) {
    $base = $this_dir == '/' ? '' : $this_dir;
    $ext = pathinfo($file, PATHINFO_EXTENSION);
    $path = $dir . DIRECTORY_SEPARATOR . $file;
    $filesize = getfilesize($path);
    $time = date('d-M-Y H:i:s', filemtime($path));
    $is_dir = is_dir($path);
    if (substr($file, 0, 1) == '.') continue;
    $files[] = array(
        'name' => $is_dir ? $file . '/' : $file,
        'ext' => $ext,
        'size' => $filesize,
        'time' => $time,
        'is_dir' => $is_dir
    );
}
usort($files, function($a, $b) {
    if ($a['is_dir']) {
        if ($b['is_dir']) {
            return $a['name'] < $b['name'] ? -1 : 1;
        } else {
            return -1;
        }
    }
    if ($b['is_dir']) {
        return 1;
    }
    return $a['name'] < $b['name'] ? -1 : 1;
});
$name_len = 50;
?>
<!DOCTYPE html>
<html>
<head>
<title><?php echo_h($this_dir); ?></title>
<meta charset="utf-8" />
</head>
<body>
<h1>Index of <?php echo_h($this_dir); ?></h1>
<pre>
    Last modified               Size  Name
<hr /><?php if ($this_dir != '/'): ?>
[D] <a href="..">Parent Directory</a>
<?php
endif;
foreach ($files as $file):
    $dirflag = $file['is_dir'] ? '[D]' : '   ';
    $prefix = sprintf('%s %s %s', $dirflag, $file['time'], $file['size']);
    $name = $file['name'];
?>
<?php echo $prefix; ?>  <a href="<?php echo_h($name); ?>"><?php echo_h($name); ?></a>
<?php endforeach; ?>
</pre>
</body>
</html>

简化PHP命令行下接收参数

命令行下执行某个PHP脚本已是家常菜,大家也知道使用getopt接收参数(例子参考手册),但是它有一个缺点,必须先定义好需要接收哪些参数,并不能像$_GET/$_POST这么简单。
花了一些时间,写了一个类,让命令行参数像$_GET一样简单,随传随用。

class Options
{
    public static function get(string $name = '', $default = null)
    {
        static $argv = null;
        if (is_null($argv)) {
            [$shortopts, $longopts] = self::parse();
            $argv                   = getopt($shortopts, $longopts);
        }
        if ('' == $name) {
            return $argv;
        }
        return $argv[$name] ?? $default;
    }

    private static function parse(): array
    {
        if (empty($_SERVER['argv'])) {
            return [];
        }
        $opts = ['', []];
        foreach ($_SERVER['argv'] as $argv) {
            if (preg_match('/^\-\-([\w\-]+)/', $argv, $matches)) {
                $opts[1][] = $matches[1] . '::';
            } elseif (preg_match('/^\-([a-z])/', $argv, $matches)) {
                $opts[0] .= $matches[1] . '::';
            }
        }
        return $opts;
    }
}
php test.php --a=1 --b=2 -c3 -d4 -d5

全部接收

$options = Options::get();
var_dump($options);
/**
array(4) {
  ["a"]=>
  string(1) "1"
  ["b"]=>
  string(1) "2"
  ["c"]=>
  string(1) "3"
  ["d"]=>
  array(2) {
    [0]=>
    string(1) "4"
    [1]=>
    string(1) "5"
  }
}
*/

单个接收

echo Options::get('a');
// 1

使用默认值

echo Options::get('e', 1);
// 1

———————– 割 ———————–
在群里抛砖引玉,果然大牛们放出其他不错的方案。

【老王】环境变量的思路

AAAA=1111 php test.php
// 可得到 $_SERVER['AAAA'] = 1111

【supermoon】

php test.php "a=1&b=2"
// 结合parse_str()