修复Nginx的WebDAV功能
如果想使用WebDAV来实现文件共享,尤其是想使用操作系统内置功能来挂载文件系统的话,省心的话还是用Apache吧。
下文介绍如何用Nginx来实现这个目标。Windows内置的客户端是Microsoft-WebDAV-MiniRedir,macOS是WebDAVFS Darwin,Linux是gvfs。
首先需要nginx-dav-ext-module,不然任何WebDAV客户端都无法工作,因为不支持PROPFIND指令无法列目录。Windows/macOS写文件需要LOCK指令,Linux不需要。
按照文档配置好了之后,挂载网络盘,OK,然后发现无法新建文件夹和重命名文件夹……没有一个操作系统能幸免……
下面介绍两种方法:
修改源代码自己编译
只要删代码就行了,不需要写任何新代码,这就导致你都不好意思向官方提BUG——不能用不是我的错,是Windows/macOS/Linux的错,它们不按规矩出牌。
ngx_http_dav_module.c
504行,判断MKCOL指令的uri必须用“/”结尾,最后传给操作系统层的时候去掉了“/”,实际上操作系统的mkdir函数能兼容“a”和“a/”两种格式。
Windows/macOS/Linux挂载后不能创建文件夹的原因。
// if (r->uri.data[r->uri.len - 1] != '/') { // ngx_log_error(NGX_LOG_ERR, r->connection->log, 0, // "MKCOL can create a collection only"); // return NGX_HTTP_CONFLICT; // } p = ngx_http_map_uri_to_path(r, &path, &root, 0); if (p == NULL) { return NGX_HTTP_INTERNAL_SERVER_ERROR; } // *(p - 1) = '\0'; // r->uri.len--;
636行,判断MOVE指令的uri和Destination的结尾的“/”必须匹配,实际上操作系统的rename函数同样能兼容“a”和“a/”,并不能把文件改成文件夹。
macOS不能重命名文件夹的原因之一——uri以/结尾,Destination没有以/结尾。
// if ((r->uri.data[r->uri.len - 1] == '/' && *(last - 1) != '/') // || (r->uri.data[r->uri.len - 1] != '/' && *(last - 1) == '/')) // { // ngx_log_error(NGX_LOG_ERR, r->connection->log, 0, // "both URI \"%V\" and \"Destination\" URI \"%V\" " // "should be either collections or non-collections", // &r->uri, &dest->value); // return NGX_HTTP_CONFLICT; // }
764行,如果MOVE的类型是文件夹的话,uri必须以“/”结尾,同上,操作系统的rename函数能够兼容。
Windows/Linux不能重命名文件夹的原因,macOS不能重命名文件夹的原因之二。
// if (r->uri.data[r->uri.len - 1] != '/') { // ngx_log_error(NGX_LOG_ERR, r->connection->log, 0, // "\"%V\" is collection", &r->uri); // return NGX_HTTP_BAD_REQUEST; // }
改完之后重新编译,Windows/macOS/Linux挂载网络盘后就基本可用了。
但是Windows上,可以新建文件,上传文件依然不可用。Windows复制文件后固执的要把新文件的时间改成和原文件一样的,因为缺少PROPPATCH指令,无法完成这个操作就罢工了,macOS/Linux就不做这个操作。实际上Windows这个习性对make之类的靠文件时间来判断是否更新的工具很不友好,经常误判。用Apache做WebDAV的时候,有时我就发现上传新文件的时候无法触发更新脚本,就是因为上传的文件的时间落后于服务器时间。
有人提供了一个fake PROPPATCH实现,就是直接当成PROPFIND指令处理。在我看来比正确的实现还要好,可以避免Windows的固执行为带来的让人烦躁的错误。
配置文件
如果不想自己编译,用配置文件也是可以解决的。
MKCOL不以/结尾
if ($request_method = MKCOL) { rewrite ^(.*[^/])$ $1/ break; }
Windows下MOVE文件夹不以/结尾
if (-d $request_filename) { rewrite ^(.*[^/])$ $1/; set $md /; }
重命名文件夹Destination不以/结尾,需要headers-more-nginx-module
set $x $http_destination$request_method; if ($x ~ [^/]MOVE) { more_set_input_headers -r "Destination: ${http_destination}${md}"; }
没有PROPPATCH指令,用PROPFIND处理。
proxy_method PROPFIND; include proxy_params; if ($request_method = PROPPATCH) { proxy_pass http://127.0.0.1;#or https://127.0.0.1 }
打完收工。