plugin = Action('webdavPlugin'); $this->checkUser(); $this->initPath($DAV_PRE); } public function run(){ $method = 'http'.HttpHeader::method(); if(!method_exists($this,$method)){ return HttpAuth::error(); } $result = $this->$method(); if(!$result) return;//文件下载; self::response($result); } /** * 用户登录校验;权限判断; * 性能优化: 通过cookie处理为已登录; (避免ad域用户或用户集成每次进行登录验证;) * */ public function checkUser(){ $userInfo = Session::get("kodUser"); if(!$userInfo || !is_array($userInfo)){ $user = HttpAuth::get(); // 兼容webdav挂载不支持中文用户名; 中文名用户名编解码处理; if(substr($user['user'],0,2) == '$$'){ $user['user'] = rawurldecode(substr($user['user'],2)); } $find = ActionCall('user.index.userInfo', $user['user'],$user['pass']); if ( !is_array($find) || !isset($find['userID']) ){ $this->plugin->log(array($user,$find,$_SERVER['HTTP_AUTHORIZATION'])); return HttpAuth::error(); } ActionCall('user.index.loginSuccess',$find); // 登录日志; if(HttpHeader::method() == 'OPTIONS'){ Model('User')->userEdit($find['userID'],array("lastLogin"=>time())); ActionCall('admin.log.loginLog'); } } if(!$this->plugin->authCheck()){ self::response(array('code'=>404,'body'=>"您没有此权限!"));exit; } } public function parsePath($path){ $options = $this->plugin->getConfig(); $rootPath = '{block:files}/'; if($options['pathAllow'] == 'self'){ //个人网盘 $rootPath = MY_HOME; } if(!$path || $path == '/') return $rootPath; $pathArr = explode('/',KodIO::clear(trim($path,'/'))); $rootList = Action('explorer.list')->path($rootPath); return $this->pathInfoDeep($rootList,$pathArr); } /** * 向下回溯路径; */ private function pathInfoDeep($parent,$pathArr){ $list = $this->pathListMerge($parent); $itemArr = array_to_keyvalue($list,'name'); $item = $itemArr[$pathArr[0]]; if(!$item) return false; if(count($pathArr) == 1) return $item['path']; $pathAppend = implode('/',array_slice($pathArr,1)); $newPath = KodIO::clear($item['path'].'/'.$pathAppend); $info = IO::infoFull($newPath); // 已存在回收站中处理; if($info && $info['isDelete'] == '1'){ $resetName = $info['name'] .date('(H:i:s)'); if($info['type'] == 'file'){ $ext = '.'.get_path_ext($info['name']); $theName = substr($info['name'],0,strlen($info['name']) - strlen($ext)); $resetName = $theName.date('(H:i:s)').$ext; } IO::rename($info['path'],$resetName); $info = IO::infoFull($newPath); } // pr($newPath,$item,$pathArr,$info,count($parent['folderList'])); if($info) return $info['path']; $parent = Action('explorer.list')->path($item['path']); $result = $this->pathInfoDeep($parent,array_slice($pathArr,1)); if(!$result){ $result = $newPath; //虚拟目录追; 没找到字内容;则认为不存在; if(Action('explorer.auth')->pathOnlyShow($item['path']) ){ $result = false; } } return $result; } public function can($path,$action){ $result = Action('explorer.auth')->fileCan($path,$action); // 编辑;则检测当前存储空间使用情况; if($result && $action == 'edit'){ $result = Action('explorer.auth')->spaceAllow($path); } return $result; } public function pathExists($path){ $info = IO::infoFull($path); if(!$info) return false; if($info['isDelete'] == '1') return false; return true; } /** * 文档属性及列表; * 不存在:404;存在207; 文件--该文件属性item; 文件夹--该文件属性item + 多个子内容属性 */ public function pathList($path){ if(!$path) return false; $info = IO::infoFull($path); if(!$info && !Action('explorer.auth')->pathOnlyShow($path) ){ return false; } if(!$this->can($path,'show')) return false; if($info && $info['isDelete'] == '1') return false;//回收站中; if($info && $info['type'] == 'file'){ //单个文件; return array('fileList'=>array($info),'current'=>$info); } $pathParse = KodIO::parse($path); // 分页大小处理--不分页; 搜索结果除外; if($pathParse['type'] != KodIO::KOD_SEARCH){ $GLOBALS['in']['pageNum'] = -1; } // write_log([$path,$pathParse,$GLOBALS['in']],'test'); return Action('explorer.list')->path($path); } public function pathMkdir($path){ $path = $this->pathCreateParent($path); if(!$this->can($path,'edit')) return false; return IO::mkdir($path); } public function pathOut($path){ if(!$this->pathExists($path) || !$this->can($path,'view')){ self::response(array('code' => 404));exit; } if(IO::size($path)<=0) return;//空文件处理; //部分webdav客户端不支持301跳转; if($this->notSupportHeader()){ IO::fileOutServer($path); }else{ IO::fileOut($path); } } // GET 下载文件;是否支持301跳转;对象存储下载走直连; private function notSupportHeader(){ $software = array( 'ReaddleDAV Documents', //ios Documents 不支持; ); $ua = $_SERVER['HTTP_USER_AGENT']; foreach ($software as $type){ if(stristr($ua,$type)) return true; } return false; } // 收藏夹下文件夹处理;(新建,上传) private function pathCreateParent($path){ if($path) return $path; $inPath = $this->pathGet(); return rtrim($this->parsePath(IO::pathFather($inPath)),'/').'/'.IO::pathThis($inPath); } public function pathPut($path,$localFile=''){ $pathBefore = $path; $path = $this->pathCreateParent($path); $info = IO::infoFull($path); if($info){ // 文件已存在; 则使用文件父目录追加文件名; $name = IO::pathThis($this->pathGet()); $uploadPath = rtrim(IO::pathFather($info['path']),'/').'/'.$name; //构建上层目录追加文件名; }else{// 首次请求创建,文件不存在; 则使用{source:xx}/newfile.txt; $uploadPath = $path; } if(!$this->can($path,'edit')) return false; // 传入了文件; wscp等直接一次上传处理的情况; windows/mac等会调用锁定,解锁,判断是否存在等之后再上传; // 文件夹下已存在,或在回收站中处理; // 删除临时文件; mac系统生成两次 ._file.txt; $size = 0; if($localFile){ $size = filesize($localFile); $result = IO::upload($uploadPath,$localFile,true,REPEAT_REPLACE); $this->pathPutRemoveTemp($uploadPath); }else{ if(!$info){ // 不存在,创建; $result = IO::mkfile($uploadPath,'',REPEAT_REPLACE); } $result = true; } $this->plugin->log("upload=$uploadPath;path=$path,$pathBefore;res=$result;local=$localFile;size=".$size); return $result; } private function pathPutRemoveTemp($path){ $pathArr = explode('/',$path); $pathArr[count($pathArr) - 1] = '._'.$pathArr[count($pathArr) - 1]; $tempPath = implode('/',$pathArr); $tempInfo = IO::infoFull($tempPath); if($tempInfo && $tempInfo['type'] == 'file'){ IO::remove($tempInfo['path'],false); } } public function pathRemove($path){ if(!$this->can($path,'remove')) return false; $tempInfo = IO::infoFull($path); if(!$tempInfo) return true; $toRecycle = Model('UserOption')->get('recycleOpen'); return IO::remove($tempInfo['path'], $toRecycle); } public function pathMove($path,$dest){ $pathUrl = $this->pathGet(); $destURL = $this->pathGet(true); $path = $this->parsePath($pathUrl); $dest = $this->parsePath(IO::pathFather($destURL)); //多出一层-来源文件(夹)名 $this->plugin->log("from=$path;to=$dest;$pathUrl;$destURL"); // 目录不变,重命名,(编辑文件) $io = IO::init('/'); if($io->pathFather($pathUrl) == $io->pathFather($destURL)){ if(!$this->can($path,'edit')) return false; $destFile = rtrim($dest,'/').'/'.$io->pathThis($destURL); $this->plugin->log("edit=$destFile;exists=".intval($this->pathExists($destFile))); $fromExt = get_path_ext($pathUrl); $toExt = get_path_ext($destURL); $officeExt = array('doc','docx','xls','xlsx','ppt','pptx'); /** * office 编辑保存最后落地时处理(导致历史记录丢失); * 0. 上传~tmp1601041332501525796.TMP //锁定,上传,解锁; * 1. 移动 test.docx => test~388C66.tmp // 改造,识别到之后不进行移动重命名; * 2. 移动 ~tmp1601041332501525796.TMP => test.docx; // 改造;目标文件已存在则更新文件;删除原文件; * 3. 删除 test~388C66.tmp */ if( $this->isWindows() && $toExt == 'tmp' && in_array($fromExt,$officeExt) ){ $result = IO::mkfile($destFile); $this->plugin->log("move mkfile=$path;$pathUrl;$destURL;result=".$result); return $result; } // 都存在则覆盖; if( $this->pathExists($path) && $this->pathExists($destFile) ){ $destFileInfo = IO::infoFull($destFile); // $content = IO::getContent($path); // IO::setContent($destFileInfo['path'],$content); // IO::remove($path);$result = $destFileInfo['path']; $result = IO::saveFile($path,$destFileInfo['path']);//覆盖保存; $this->plugin->log("move saveFile; to=$path;toFile=".$destFileInfo['path'].';result='.$result); return $result; } return IO::rename($path,$io->pathThis($destURL)); } if(!$this->can($path,'remove')) return false; if(!$this->can($dest,'edit')) return false; // 名称不同先重命名; if( $io->pathThis($destURL) != $io->pathThis($pathUrl) ){ $path = IO::rename($path,$io->pathThis($destURL)); } return IO::move($path,$dest); } public function pathCopy($path,$dest){ $pathUrl = $this->pathGet(); $destURL = $this->pathGet(true); $path = $this->parsePath($pathUrl); $dest = $this->parsePath(IO::pathFather($destURL)); //多出一层-来源文件(夹)名 $this->plugin->log("from=$path;to=$dest;$pathUrl;$destURL"); if(!$this->can($path,'download')) return false; if(!$this->can($dest,'edit')) return false; return IO::copy($path,$dest); } private function isWindows(){ return stristr($_SERVER['HTTP_USER_AGENT'],'Microsoft-WebDAV-MiniRedir'); } }