如果想使用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, );
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 ^(.*[^/])$ $/ break; }

Windows下MOVE文件夹不以/结尾

if (-d $request_filename) {
rewrite ^(.*[^/])$ $/;
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;
}

打完收工。

修复Nginx的WebDAV功能的更多相关文章

  1. Nginx range filter模块数字错误漏洞修复 (Nginx平滑升级) 【转】

    对线上生产环境服务器进行漏洞扫描, 发现有两台前置机器存在Nginx range filter模块数字错误漏洞, 当使用nginx标准模块时,攻击者可以通过发送包含恶意构造range域的header ...

  2. Nginx 状态信息功能配置

    Nginx 状态信息功能介绍 Nginx 有一个 ngx_http_stub_status_module 模块,主要功能是记录 Nginx 的基本访问状态信息,让使用者了解 Nginx 的工作状态 要 ...

  3. Nginx启动SSL功能

    Nginx启动SSL功能,并进行功能优化,你看这个就足够了 一:开始Nginx的SSL模块 1.1 Nginx如果未开启SSL模块,配置Https时提示错误 nginx: [emerg] the &q ...

  4. 启动Nginx目录浏览功能及 让用户通过用户名密码认证访问web站点

    一.启动Nginx目录浏览功能  [root@abcdocker extra]# cat w.conf server { listen 80; server_name IP地址; location / ...

  5. Nginx代理缓存功能

    Nginx代理缓存功能      Nginx缓存主要是用于减轻后端服务器的负载,提高网站并发量,提升用户体验度. 注意:Nginx反向代理的缓存功能是由ngx_http_proxy_module提供, ...

  6. Nginx 反向代理功能-动静分离

    Nginx 反向代理功能-动静分离 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任.

  7. Nginx 反向代理功能-实现Nginx tcp负载均衡

    Nginx 反向代理功能-实现Nginx tcp负载均衡 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任.

  8. Nginx 反向代理功能-实现http反向代理

    Nginx 反向代理功能-实现http反向代理 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任.

  9. Nginx Rewrite相关功能-防盗链

    Nginx Rewrite相关功能-防盗链 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任.

随机推荐

  1. nmon监控指标

    一.NMON中的各项参数指标: SYS_SUMM:显示当前服务器的总体性能情况 Total System I/OStatistics: Avg tps during an interval:显示采集间 ...

  2. android愤怒小鸟游戏、自定义View、掌上餐厅App、OpenGL自定义气泡、抖音电影滤镜效果等源码

    Android精选源码 精练的范围选择器,范围和单位可以自定义 自定义View做的小鸟游戏 android popwindow选择商品规格颜色尺寸效果源码 实现Android带有锯齿背景的优惠样式源码 ...

  3. maven-assembly-plugin 打包包含多余依赖问题一则

    有同事反馈自己maven-assembly-plugin打的包里面多了很多mvn dependency:tree中没有的jar. 我当时只是试着把他的maven-assembly-plugin更新到了 ...

  4. [LC] 77. Combinations

    Given two integers n and k, return all possible combinations of k numbers out of 1 ... n. Example: I ...

  5. C#阶段小结

    一.数据类型: (一)内建类型: 整型(int ,short, long ,byte ,uint ,ushort, ulong ,sbyte): 浮点型(double float decimal): ...

  6. EMCCD

    EMCCD 即电子倍增CCD,是探测领域内灵敏度极高的一种高端光电探测产品. 在光子探测领域的应用发展对探测器灵敏度的要求不断提高,EMCCD (Electron-Multiplying CCD)技术 ...

  7. python数据类型:元组

    python数据类型:元组 python的元组与列表类似,但是元组的元素不能修改 元组使用小括号,列表使用大括号 元组创建简单,只需要在括号中添加元素,使用逗号隔开 创建元组: tup1 = (50, ...

  8. 关于va_list实例

    printf函数: #include <stdio.h> #include <stdarg.h> int myself_printf(char *format, ...) { ...

  9. [SDOI2006] 线性方程组

    洛谷 P2455 传送门 刚开始写了个消成上三角的,结果狂wa. 后来经过研究发现,消成上三角那种不能直接判断无解或无穷多解,需要其它的操作. 所以干脆学了个消成对角线的,写了一发A了. 其实两种消元 ...

  10. java连接sqlserver数据简单操作

    首先我们下载微软的JDBC驱动包 https://www.microsoft.com/en-us/download/details.aspx?id=11774 下载好后我们打开如下图 点击Unzip ...