Linux查找所有非UTF-8编码的文件或转换成UTF-8

一行Linux命令查找所有非UTF-8编码的文件

find . ! -iregex '.*\.svn.*' -type f -name '*.php' -exec bash -c "enca -L zh_CN {}|grep GB2312 > /dev/null && echo {}" \;

一条命令将他们都转换成UTF8编码

find . ! -iregex '.*\.svn.*' -type f -name '*.php' -exec bash -c "enca -L zh_CN {}  | grep GB2312 >/dev/null && enconv -L zh_CN -x UTF-8 {}" \;

通过javascript判断浏览器是否支持webp

function check_webp_feature(feature, callback) {
    var kTestImages = {
        lossy: "UklGRiIAAABXRUJQVlA4IBYAAAAwAQCdASoBAAEADsD+JaQAA3AAAAAA",
        lossless: "UklGRhoAAABXRUJQVlA4TA0AAAAvAAAAEAcQERGIiP4HAA==",
        alpha: "UklGRkoAAABXRUJQVlA4WAoAAAAQAAAAAAAAAAAAQUxQSAwAAAARBxAR/Q9ERP8DAABWUDggGAAAABQBAJ0BKgEAAQAAAP4AAA3AAP7mtQAAAA==",
        animation: "UklGRlIAAABXRUJQVlA4WAoAAAASAAAAAAAAAAAAQU5JTQYAAAD/////AABBTk1GJgAAAAAAAAAAAAAAAAAAAGQAAABWUDhMDQAAAC8AAAAQBxAREYiI/gcA"
    };
    var img = new Image();
    img.onload = function () {
        var result = (img.width > 0) && (img.height > 0);
        callback(feature, result);
    };
    img.onerror = function () {
        callback(feature, false);
    };
    img.src = "data:image/webp;base64," + kTestImages[feature];
}
check_webp_feature('lossy', function(f, r){ console.log(f, r);});

原文发址:https://developers.google.com/speed/webp/faq#how_can_i_detect_browser_support_using_javascript

sublime text ConvertToUTF8乱码

大家都知道,sublime 对不支持,需要依赖插件才能支持,比较推荐的是 ConvertToUTF8,安装后GBK能正常显示。但是有时候会遇到某个文件打开显示乱码之后,怎么刷新、重新打开都还是一样,估计插件有缓存机制。
打开插件配置,果不其然!

"max_cache_size" : 100,

将这个值改为0即可解决问题了。

为 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()

nginx中配置跨域支持功能

在nginx.conf中配置

http {
  ......
  add_header Access-Control-Allow-Origin *;
  add_header Access-Control-Allow-Headers X-Requested-With;
  add_header Access-Control-Allow-Methods GET,POST,OPTIONS;
  ......
}

这样就可以实现GET,POST,OPTIONS的跨域请求的支持
也可以

add_header Access-Control-Allow-Origin http://test.51testing.com;

指定允许的url。
或者动态指定

if ($http_origin ~* ".*\.example\.com") {
    add_header Access-Control-Allow-Origin $http_origin;
    add_header Access-Control-Allow-Credentials true;
}

配置项详细解释:w3c-Cross-Origin Resource Sharing

php7.1新特性

Allow specifying keys in list()
Square bracket syntax for array destructuring assignment

下面这个支持效果,还是不错的

list($a, $b) = array($b, $a);
[$a, $b] = [$b, $a];

foreach ($points as ["x" => $x, "y" => $y]) {
    var_dump($x, $y);
}

这个在遍历的时候会很方便,也很强大
感觉这两个特性会使数组使用便捷一些

Nullable Types
这个就是一个简写的标量或null,用处有,但感觉不大

Generalize support of negative string offsets
这个特性好,可以简化一些简单的字符串处理

Support Class Constant Visibility
暂时想不到太大用处,只是将类常量完善了

Void Return Type
这个限制函数返回值有用
同时希望miexd,number什么的也尽快出来

Catching Multiple Exception Types
多cacth异常感觉没什么用,可能还是不太会用异常吧