4 Copyright 2003,2004,2007 $ThePhpWikiProgrammingTeam
5 Copyright 2008-2009 Marc-Etienne Vargenau, Alcatel-Lucent
7 This file is part of PhpWiki.
9 PhpWiki is free software; you can redistribute it and/or modify
10 it under the terms of the GNU General Public License as published by
11 the Free Software Foundation; either version 2 of the License, or
12 (at your option) any later version.
14 PhpWiki is distributed in the hope that it will be useful,
15 but WITHOUT ANY WARRANTY; without even the implied warranty of
16 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 GNU General Public License for more details.
19 You should have received a copy of the GNU General Public License
20 along with PhpWiki; if not, write to the Free Software
21 Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
26 * UpLoad: Allow Administrator to upload files to a special directory,
27 * which should preferably be added to the InterWikiMap
28 * Usage: <?plugin UpLoad ?>
29 * Author: NathanGass <gass@iogram.ch>
30 * Changes: ReiniUrban <rurban@x-ray.at>,
31 * qubit <rtryon@dartmouth.edu>
32 * Marc-Etienne Vargenau, Alcatel-Lucent
33 * Note: See also Jochen Kalmbach's plugin/UserFileManagement.php
36 class WikiPlugin_UpLoad
39 var $disallowed_extensions;
40 // TODO: use PagePerms instead
41 var $only_authenticated = true; // allow only authenticated users may upload.
47 function getDescription () {
48 return _("Upload files to the local InterWiki Upload:<filename>");
51 function getVersion() {
52 return preg_replace("/[Revision: $]/", '',
56 function getDefaultArguments() {
57 return array('logfile' => 'phpwiki-upload.log',
58 // add a link of the fresh file automatically to the
59 // end of the page (or current page)
61 'page' => '[pagename]',
63 'mode' => 'actionpage', // or edit
67 function run($dbi, $argstr, &$request, $basepage) {
68 $this->allowed_extensions = explode("\n",
106 $this->disallowed_extensions = explode("\n",
144 //removed "\{[[:xdigit:]]{8}(?:-[[:xdigit:]]{4}){3}-[[:xdigit:]]{12}\}"
146 $args = $this->getArgs($argstr, $request);
149 $file_dir = getUploadFilePath();
151 $form = HTML::form(array('action' => $request->getPostURL(),
152 'enctype' => 'multipart/form-data',
153 'method' => 'post'));
154 $contents = HTML::div(array('class' => 'wikiaction'));
155 $contents->pushContent(HTML::input(array('type' => 'hidden',
156 'name' => 'MAX_FILE_SIZE',
157 'value'=> MAX_UPLOAD_SIZE)));
158 $contents->pushContent(HTML::input(array('name' => 'userfile',
161 if ($mode == 'edit') {
162 $contents->pushContent(HTML::input(array('name' => 'action',
165 $contents->pushContent(HTML::raw(" "));
166 $contents->pushContent(HTML::input(array('value' => _("Upload"),
167 'name' => 'edit[upload]',
168 'type' => 'submit')));
170 $contents->pushContent(HTML::raw(" "));
171 $contents->pushContent(HTML::input(array('value' => _("Upload"),
172 'type' => 'submit')));
174 $form->pushContent($contents);
177 if ($request->isPost() and $this->only_authenticated) {
178 // Make sure that the user is logged in.
179 $user = $request->getUser();
180 if (!$user->isAuthenticated()) {
182 if (isa($WikiTheme, 'WikiTheme_gforge')) {
183 $message->pushContent(HTML::div(array('class' => 'error'),
184 HTML::p(_("You cannot upload files.")),
186 HTML::li(_("Check you are logged in.")),
187 HTML::li(_("Check you are in the right project.")),
188 HTML::li(_("Check you are a member of the current project."))
192 $message->pushContent(HTML::div(array('class' => 'error'),
193 HTML::p(_("ACCESS DENIED: You must log in to upload files."))));
196 $result->pushContent($form);
197 $result->pushContent($message);
202 $userfile = $request->getUploadedFile('userfile');
204 $userfile_name = $userfile->getName();
205 $userfile_name = trim(basename($userfile_name));
206 if (UPLOAD_USERDIR) {
207 $file_dir .= $request->_user->_userid;
208 if (!file_exists($file_dir))
209 mkdir($file_dir, 0775);
211 $u_userfile = $request->_user->_userid . "/" . $userfile_name;
213 $u_userfile = $userfile_name;
215 $u_userfile = preg_replace("/ /", "%20", $u_userfile);
216 $userfile_tmpname = $userfile->getTmpName();
217 $err_header = HTML::div(array('class' => 'error'),
218 HTML::p(fmt("ERROR uploading '%s'", $userfile_name)));
219 if (preg_match("/(\." . join("|\.", $this->disallowed_extensions) . ")(\.|\$)/i",
222 $message->pushContent($err_header);
223 $message->pushContent(HTML::p(fmt("Files with extension %s are not allowed.",
224 join(", ", $this->disallowed_extensions))));
226 elseif (! DISABLE_UPLOAD_ONLY_ALLOWED_EXTENSIONS and
227 ! preg_match("/(\." . join("|\.", $this->allowed_extensions) . ")\$/i",
230 $message->pushContent($err_header);
231 $message->pushContent(HTML::p(fmt("Only files with the extension %s are allowed.",
232 join(", ", $this->allowed_extensions))));
234 elseif (preg_match("/[^._a-zA-Z0-9- ]/", strip_accents($userfile_name)))
236 $message->pushContent($err_header);
237 $message->pushContent(HTML::p(_("Invalid filename. File names may only contain alphanumeric characters and dot, underscore, space or dash.")));
239 elseif (file_exists($file_dir . $userfile_name)) {
240 $message->pushContent($err_header);
241 $message->pushContent(HTML::p(fmt("There is already a file with name %s uploaded.",
244 elseif ($userfile->getSize() > (MAX_UPLOAD_SIZE)) {
245 $message->pushContent($err_header);
246 $message->pushContent(HTML::p(_("Sorry but this file is too big.")));
248 elseif (move_uploaded_file($userfile_tmpname, $file_dir . $userfile_name) or
249 (IsWindows() and rename($userfile_tmpname, $file_dir . $userfile_name))
252 $interwiki = new PageType_interwikimap();
253 $link = $interwiki->link("Upload:$u_userfile");
254 $message->pushContent(HTML::div(array('class' => 'feedback'),
255 HTML::p(_("File successfully uploaded.")),
258 // the upload was a success and we need to mark this event in the "upload log"
260 $upload_log = $file_dir . basename($logfile);
261 $this->log($userfile, $upload_log, $message);
264 require_once("lib/loadsave.php");
265 $pagehandle = $dbi->getPage($page);
266 if ($pagehandle->exists()) {// don't replace default contents
267 $current = $pagehandle->getCurrentRevision();
268 $version = $current->getVersion();
269 $text = $current->getPackedContent();
270 $newtext = $text . "\n* Upload:$u_userfile"; // don't inline images
271 $meta = $current->_data;
272 $meta['summary'] = sprintf(_("uploaded %s"),$u_userfile);
273 $pagehandle->save($newtext, $version + 1, $meta);
277 $message->pushContent($err_header);
278 $message->pushContent(HTML::br(),_("Uploading failed."),HTML::br());
282 $message->pushContent(HTML::br(),_("No file selected. Please select one."),HTML::br());
285 //$result = HTML::div( array( 'class' => 'wikiaction' ) );
287 $result->pushContent($form);
288 $result->pushContent($message);
292 function log ($userfile, $upload_log, &$message) {
294 $user = $GLOBALS['request']->_user;
295 if (file_exists($upload_log) and (!is_writable($upload_log))) {
296 trigger_error(_("The upload logfile exists but is not writable."), E_USER_WARNING);
298 elseif (!$log_handle = fopen ($upload_log, "a")) {
299 trigger_error(_("Can't open the upload logfile."), E_USER_WARNING);
301 else { // file size in KB; precision of 0.1
302 $file_size = round(($userfile->getSize())/1024, 1);
303 if ($file_size <= 0) {
304 $file_size = "< 0.1";
306 $userfile_name = $userfile->getName();
309 . "<tr><td><a href=\"$userfile_name\">$userfile_name</a></td>"
310 . "<td align=\"right\">$file_size kB</td>"
311 . "<td> " . $WikiTheme->formatDate(time()) . "</td>"
312 . "<td> <em>" . $user->getId() . "</em></td></tr>");
320 // (c-file-style: "gnu")
325 // c-hanging-comment-ender-p: nil
326 // indent-tabs-mode: nil