513 lines
18 KiB
PHP
Executable File
513 lines
18 KiB
PHP
Executable File
<?php
|
|
/**
|
|
* The control file of file module of ZenTaoPMS.
|
|
*
|
|
* @copyright Copyright 2009-2015 禅道软件(青岛)有限公司(ZenTao Software (Qingdao) Co., Ltd. www.cnezsoft.com)
|
|
* @license ZPL(http://zpl.pub/page/zplv12.html) or AGPL(https://www.gnu.org/licenses/agpl-3.0.en.html)
|
|
* @author Chunsheng Wang <chunsheng@cnezsoft.com>
|
|
* @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("<html><head><meta charset='utf-8'></head><body>{$this->lang->file->fileNotFound}</body></html>");
|
|
}
|
|
|
|
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("<html><head><meta charset='utf-8'></head><body>{$this->lang->file->fileNotFound}</body></html>");
|
|
}
|
|
}
|
|
|
|
/**
|
|
* 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);
|
|
}
|
|
}
|
|
}
|