* @package file * @version $Id: control.php 4129 2013-01-18 01:58:14Z wwccss $ * @link http://www.zentao.net */ class file extends control { /** * Build the upload form. * * @param int $fileCount * @param float $percent * @param string $filesName * @param string $labelsName * @access public * @return void */ public function buildForm($fileCount = 1, $percent = 0.9, $filesName = "files", $labelsName = "labels") { if(!file_exists($this->file->savePath)) { printf($this->lang->file->errorNotExists, $this->file->savePath); return false; } elseif(!is_writable($this->file->savePath)) { printf($this->lang->file->errorCanNotWrite, $this->file->savePath, $this->file->savePath); return false; } $this->view->filesName = $filesName; $this->view->labelsName = $labelsName; $this->display(); } /** * AJAX: get upload request from the web editor. * * @param $uid * @access public * @return void */ public function ajaxUpload($uid = '') { $file = $this->file->getUpload('imgFile'); if(!isset($file[0]) or !in_array($file[0]['extension'], $this->config->file->imageExtensions)) { return print(json_encode(array('result' => 'fail', 'message' => $this->lang->file->errorFileFormate))); } $file = $file[0]; if($file) { if($file['size'] == 0) { if(defined('RUN_MODE') && RUN_MODE == 'api') { return print(json_encode(array('status' => 'error', 'message' => $this->lang->file->errorFileUpload))); } else { return print(json_encode(array('error' => 1, 'message' => $this->lang->file->errorFileUpload))); } } if(@move_uploaded_file($file['tmpname'], $this->file->savePath . $this->file->getSaveName($file['pathname']))) { /* Compress image for jpg and bmp. */ $file = $this->file->compressImage($file); $file['addedBy'] = $this->app->user->account; $file['addedDate'] = helper::today(); unset($file['tmpname']); $this->dao->insert(TABLE_FILE)->data($file)->exec(); $fileID = $this->dao->lastInsertID(); $url = $this->createLink('file', 'read', "fileID=$fileID", $file['extension']); if($uid) $_SESSION['album'][$uid][] = $fileID; if(defined('RUN_MODE') && RUN_MODE == 'api') { if($uid) $_SESSION['album']['used'][$uid][$fileID] = $fileID; $_SERVER['SCRIPT_NAME'] = 'index.php'; return $this->send(array('status' => 'success', 'id' => $fileID, 'url' => $url)); } else { return print(json_encode(array('error' => 0, 'url' => $url))); } } else { $error = strip_tags(sprintf($this->lang->file->errorCanNotWrite, $this->file->savePath, $this->file->savePath)); if(defined('RUN_MODE') && RUN_MODE == 'api') { return $this->send(array('status' => 'error', 'message' => $error)); } else { return print(json_encode(array('error' => 1, 'message' => $error))); } } } return $this->send(array('status' => 'error', 'message' => $this->lang->file->uploadImagesExplain)); } /** * Down a file. * * @param int $fileID * @param string $mouse * @access public * @return void */ public function download($fileID, $mouse = '') { if(session_id() != $this->app->sessionID) helper::restartSession($this->app->sessionID); $file = $this->file->getById($fileID); if(empty($file)) { if(defined('RUN_MODE') && RUN_MODE == 'api') return $this->send(array('status' => 'fail', 'code' => 404, 'message' => $this->lang->file->fileNotFound)); return print("{$this->lang->file->fileNotFound}"); } if(!$this->file->checkPriv($file)) { echo(js::alert($this->lang->file->accessDenied)); if(isonlybody()) return print(js::reload('parent.parent')); return print(js::locate(helper::createLink('my', 'index'), 'parent.parent')); } /* Judge the mode, down or open. */ $mode = 'down'; $fileTypes = 'txt|jpg|jpeg|gif|png|bmp|xml|html'; if(stripos($fileTypes, $file->extension) !== false && $mouse == 'left') $mode = 'open'; if($file->extension == 'txt') { $extension = 'txt'; if(($postion = strrpos($file->title, '.')) !== false) $extension = substr($file->title, $postion + 1); if($extension != 'txt') $mode = 'down'; $file->extension = $extension; } if($this->file->fileExists($file)) { /* If the mode is open, locate directly. */ if($mode == 'open') { if(stripos('txt|jpg|jpeg|gif|png|bmp', $file->extension) !== false) { $this->view->file = $file; $this->view->charset = $this->get->charset ? $this->get->charset : $this->config->charset; $this->view->fileType = ($file->extension == 'txt') ? 'txt' : 'image'; $this->display(); } else { $this->locate($file->webPath); } } else { /* Down the file. */ $fileName = $file->title; if(!preg_match("/\.{$file->extension}$/", $fileName)) $fileName .= '.' . $file->extension; $this->sendDownHeader($fileName, $file->extension, $file->realPath, 'file'); } } else { if(defined('RUN_MODE') && RUN_MODE == 'api') return $this->send(array('status' => 'fail', 'code' => 404, 'message' => $this->lang->file->fileNotFound)); return print("{$this->lang->file->fileNotFound}"); } } /** * Export as csv format. * * @access public * @return void */ public function export2CSV() { $this->view->fields = $this->post->fields; $this->view->rows = $this->post->rows; $output = $this->parse('file', 'export2csv'); if($this->post->encode != "utf-8") $output = helper::convertEncoding($output, 'utf-8', $this->post->encode . '//TRANSLIT'); $this->sendDownHeader($this->post->fileName, 'csv', $output); } /** * export as xml format * * @access public * @return void */ public function export2XML() { $this->view->fields = $this->post->fields; $this->view->rows = $this->post->rows; $output = $this->parse('file', 'export2XML'); $this->sendDownHeader($this->post->fileName, 'xml', $output); } /** * export as html format * * @access public * @return void */ public function export2HTML() { $this->view->fields = $this->post->fields; $this->view->rows = $this->post->rows; $this->host = common::getSysURL(); $kind = $this->post->kind; foreach($this->view->rows as $row) { foreach($row as &$field) { if(empty($field)) continue; $field = preg_replace('/ src="{([0-9]+)(\.(\w+))?}" /', ' src="' . $this->host . helper::createLink('file', 'read', "fileID=$1", "$3") . '" ', $field); } if(in_array($kind, array('story', 'bug', 'testcase'))) $row->title = html::a($this->host . $this->createLink($kind, 'view', "{$kind}ID=$row->id"), $row->title); if($kind == 'task') $row->name = html::a($this->host . $this->createLink('task', 'view', "taskID=$row->id"), $row->name); } $this->view->fileName = $this->post->fileName; $output = $this->parse('file', 'export2Html'); $this->sendDownHeader($this->post->fileName, 'html', $output); } /** * Send the download header to the client. * * @param string $fileName * @param string $extension * @access public * @return void */ public function sendDownHeader($fileName, $fileType, $content, $type = 'content') { $this->file->sendDownHeader($fileName, $fileType, $content, $type); } /** * Delete a file. * * @param int $fileID * @param string $confirm yes|no * @access public * @return void */ public function delete($fileID, $confirm = 'no') { if($confirm == 'no') { return print(js::confirm($this->lang->file->confirmDelete, inlink('delete', "fileID=$fileID&confirm=yes"))); } else { $file = $this->file->getById($fileID); $this->dao->delete()->from(TABLE_FILE)->where('id')->eq($fileID)->exec(); $this->loadModel('action')->create($file->objectType, $file->objectID, 'deletedFile', '', $extra=$file->title); /* Fix Bug #1518. */ $fileRecord = $this->dao->select('id')->from(TABLE_FILE)->where('pathname')->eq($file->pathname)->fetch(); if(empty($fileRecord)) $this->file->unlinkFile($file); /* Update test case version for test case synchronization. */ if($file->objectType == 'testcase') $this->file->updateTestcaseVersion($file); return print(js::reload('parent')); } } /** * Print files. * * @param array $files * @param string $fieldset * @param object $object * @param string $method * @param bool $showDelete * @param bool $showEdit * @access public * @return void */ public function printFiles($files, $fieldset, $object = null, $method = 'view', $showDelete = true, $showEdit = true) { $this->view->files = $files; $this->view->fieldset = $fieldset; $this->view->object = $object; $this->view->method = $method; $this->view->showDelete = $showDelete; $this->view->showEdit = $showEdit; if(strpos('view,edit', $method) !== false and $this->app->clientDevice != 'mobile') return $this->display('file', 'viewfiles'); $this->display(); } /** * Edit file's name. * * @param int $fileID * @access public * @return void */ public function edit($fileID) { if($_POST) { $this->app->loadLang('action'); $file = $this->file->getByID($fileID); $data = fixer::input('post')->get(); if(validater::checkLength($data->fileName, 80, 1) == false) { $errTip = $this->lang->error->length; return print(js::alert(sprintf($errTip[1], $this->lang->file->title, 80, 1))); } $fileName = $data->fileName . '.' . $data->extension; $this->dao->update(TABLE_FILE)->set('title')->eq($fileName)->where('id')->eq($fileID)->exec(); $extension = "." . $file->extension; $actionID = $this->loadModel('action')->create($file->objectType, $file->objectID, 'editfile', '', $fileName); $changes[] = array('field' => 'fileName', 'old' => $file->title, 'new' => $fileName); $this->action->logHistory($actionID, $changes); /* Update test case version for test case synchronization. */ if($file->objectType == 'testcase' and $file->title != $fileName) $this->file->updateTestcaseVersion($file); $newFile = $this->file->getByID($fileID); if($this->app->clientDevice == 'mobile') return print(js::reload('parent.parent')); echo json_encode($newFile); } if($this->app->clientDevice == 'mobile') { $file = $this->file->getById($fileID); if(strrpos($file->title, '.') !== false) { /* Fix the file name exe.exe */ $title = explode('.', $file->title); $extension = end($title); if($file->extension == 'txt' && $extension != $file->extension) $file->extension = $extension; array_pop($title); $file->title = join('.', $title); } $this->view->file = $file; $this->display(); } } /** * Paste image in kindeditor at firefox and chrome. * * @access public * @return void */ public function ajaxPasteImg($uid = '') { if($_POST) return print($this->file->pasteImage($this->post->editor, $uid, $safe = true)); } /** * Upload Images. * * @param string $module * @param string $params * @access public * @return void */ public function uploadImages($module, $params, $uid = '', $locate = false) { if($locate) { $sessionName = $uid . 'ImagesFile'; $imageFiles = $this->session->$sessionName; $this->session->set($module . 'ImagesFile', $imageFiles); unset($_SESSION[$sessionName]); return print(js::locate($this->createLink($module, 'batchCreate', helper::safe64Decode($params)), 'parent')); } if($_FILES) { $file = $this->file->getUploadFile('file'); if(!$file) return print(json_encode(array('result' => 'fail', 'message' => $this->lang->error->noData))); if(empty($file['extension']) or !in_array($file['extension'], $this->config->file->imageExtensions)) { return print(json_encode(array('result' => 'fail', 'message' => $this->lang->file->errorFileFormate))); } $imageFile = $this->file->saveUploadFile($file, $uid); if($imageFile === false) { return print(json_encode(array('result' => 'fail', 'message' => $this->lang->file->errorFileMove))); } else { if(!empty($imageFile)) { $sessionName = $uid . 'ImagesFile'; $imageFiles = $this->session->$sessionName; $fileName = basename($imageFile['pathname']); $imageFiles[$fileName] = $imageFile; $this->session->set($sessionName, $imageFiles); } return print(json_encode(array('result' => 'success', 'file' => $file, 'message' => $this->lang->file->uploadSuccess))); } } $this->view->uid = empty($uid) ? uniqid() : $uid; $this->view->module = $module; $this->view->params = $params; $this->display(); } /** * Build export tpl. * * @param string $module * @param int $templateID * @access public * @return void */ public function buildExportTPL($module, $templateID = 0) { $templates = $this->file->getExportTemplate($module); $templatePairs[] = $this->lang->file->defaultTPL; foreach($templates as $template) $templatePairs[$template->id] = ($template->public ? "[{$this->lang->public}] " : '') . $template->title; $this->view->templates = $templates; $this->view->templatePairs = $templatePairs; $this->view->templateID = $templateID; $this->display(); } /** * Ajax save template. * * @param string $module * @access public * @return void */ public function ajaxSaveTemplate($module) { $templateID = $this->file->saveExportTemplate($module); if(dao::isError()) { echo js::error(dao::getError(), $full = false); $templateID = 0; } return print($this->fetch('file', 'buildExportTPL', "module=$module&templateID=$templateID")); } /** * Ajax delete template. * * @param int $templateID * @access public * @return void */ public function ajaxDeleteTemplate($templateID) { $this->dao->delete()->from(TABLE_USERTPL)->where('id')->eq($templateID)->andWhere('account')->eq($this->app->user->account)->exec(); } /** * Read file. * * @param int $fileID * @access public * @return void */ public function read($fileID) { $file = $this->file->getById($fileID); if(empty($file) or !$this->file->fileExists($file)) return false; $obLevel = ob_get_level(); for($i = 0; $i < $obLevel; $i++) ob_end_clean(); $mime = (isset($file->extension) and in_array($file->extension, $this->config->file->imageExtensions)) ? "image/{$file->extension}" : $this->config->file->mimes['default']; header("Content-type: $mime"); $cacheMaxAge = 10 * 365 * 24 * 3600; header("Cache-Control: private"); header("Pragma: cache"); header("Expires:" . gmdate("D, d M Y H:i:s", time() + $cacheMaxAge) . " GMT"); header("Cache-Control: max-age=$cacheMaxAge"); $handle = fopen($file->realPath, "r"); if($handle) { while(!feof($handle)) echo fgets($handle); fclose($handle); } } }