2 rcs_id('$Id: loadsave.php,v 1.8 2001-09-18 19:16:23 dairiki Exp $');
3 require_once("lib/ziplib.php");
4 require_once("lib/Template.php");
6 function StartLoadDump($title, $html = '')
8 // FIXME: This is a hack
9 echo ereg_replace('</body>.*', '',
10 GeneratePage('MESSAGE', $html, $title, 0));
13 function EndLoadDump()
15 // FIXME: This is a hack
17 echo Element('p', QElement('b', gettext("Complete.")));
18 echo Element('p', "Return to " . LinkExistingWikiWord($GLOBALS['pagename']));
19 echo "</body></html>\n";
23 ////////////////////////////////////////////////////////////////
25 // Functions for dumping.
27 ////////////////////////////////////////////////////////////////
29 function MailifyPage ($page, $nversions = 1)
33 $current = $page->getCurrentRevision();
34 $from = isset($SERVER_ADMIN) ? $SERVER_ADMIN : 'foo@bar';
36 $head = "From $from " . ctime(time()) . "\r\n";
37 $head .= "Subject: " . rawurlencode($page->getName()) . "\r\n";
38 $head .= "From: $from (PhpWiki)\r\n";
39 $head .= "Date: " . rfc1123date($current->get('mtime')) . "\r\n";
40 $head .= sprintf("Mime-Version: 1.0 (Produced by PhpWiki %s)\r\n", PHPWIKI_VERSION);
42 $iter = $page->getAllRevisions();
44 while ($revision = $iter->next()) {
45 $parts[] = MimeifyPageRevision($revision);
46 if ($nversions > 0 && count($parts) >= $nversions)
49 if (count($parts) > 1)
50 return $head . MimeMultipart($parts);
52 return $head . $parts[0];
56 * The main() function which generates a zip archive of a PhpWiki.
58 * If $include_archive is false, only the current version of each page
59 * is included in the zip file; otherwise all archived versions are
62 function MakeWikiZip ($dbi, $request)
64 if ($request->getArg('include') == 'all') {
65 $zipname = "wikidb.zip";
66 $include_archive = true;
69 $zipname = "wiki.zip";
70 $include_archive = false;
75 $zip = new ZipWriter("Created by PhpWiki", $zipname);
77 $pages = $dbi->getAllPages();
78 while ($page = $pages->next()) {
79 set_time_limit(30); // Reset watchdog.
81 $current = $page->getCurrentRevision();
82 if ($current->getVersion() == 0)
86 $attrib = array('mtime' => $current->get('mtime'),
88 if ($page->get('locked'))
89 $attrib['write_protected'] = 1;
92 $content = MailifyPage($page, 0);
94 $content = MailifyPage($page);
96 $zip->addRegularFile( rawurlencode($page->getName()),
102 function DumpToDir ($dbi, $request)
104 $directory = $request->getArg('directory');
105 if (empty($directory))
106 ExitWiki(gettext("You must specify a directory to dump to"));
108 // see if we can access the directory the user wants us to use
109 if (! file_exists($directory)) {
110 if (! mkdir($directory, 0755))
111 ExitWiki("Cannot create directory '$directory'<br>\n");
113 $html = "Created directory '$directory' for the page dump...<br>\n";
115 $html = "Using directory '$directory'<br>\n";
118 StartLoadDump("Dumping Pages", $html);
120 $pages = $dbi->getAllPages();
122 while ($page = $pages->next()) {
124 $enc_name = htmlspecialchars($page->getName());
125 $filename = rawurlencode($page->getName());
127 echo "<br>$enc_name ... ";
128 if($pagename != $filename)
129 echo "<small>saved as $filename</small> ... ";
131 $data = MailifyPage($page);
133 if ( !($fd = fopen("$directory/$filename", "w")) )
134 ExitWiki("<b>couldn't open file '$directory/$filename' for writing</b>\n");
136 $num = fwrite($fd, $data, strlen($data));
137 echo "<small>$num bytes written</small>\n";
140 assert($num == strlen($data));
147 ////////////////////////////////////////////////////////////////
149 // Functions for restoring.
151 ////////////////////////////////////////////////////////////////
153 function SavePage ($dbi, $pageinfo, $source, $filename)
155 $pagedata = $pageinfo['pagedata']; // Page level meta-data.
156 $versiondata = $pageinfo['versiondata']; // Revision level meta-data.
158 if (empty($pageinfo['pagename'])) {
159 echo Element('dd'). Element('dt', QElement('b', "Empty pagename!"));
163 if (empty($versiondata['author_id']))
164 $versiondata['author_id'] = $versiondata['author'];
166 $pagename = $pageinfo['pagename'];
167 $content = $pageinfo['content'];
169 $page = $dbi->getPage($pagename);
171 foreach ($pagedata as $key => $value) {
173 $page->set($key, $value);
179 $mesg[] = sprintf(gettext("from %s"), $source);
181 $current = $page->getCurrentRevision();
182 if ($current->getVersion() == 0) {
183 $mesg[] = gettext("new page");
187 if ($current->getPackedContent() == $content
188 && $current->get('author') == $versiondata['author']) {
189 $mesg[] = sprintf(gettext("is identical to current version %d"),
190 $current->getVersion());
191 $mesg[] = gettext("- skipped");
198 $new = $page->createRevision(WIKIDB_FORCE_CREATE, $content,
200 ExtractWikiPageLinks($content));
202 $mesg[] = gettext("- saved");
203 $mesg[] = sprintf(gettext("- saved as version %d"), $new->getVersion());
206 print( Element('dt', LinkExistingWikiWord($pagename))
207 . QElement('dd', join(" ", $mesg))
212 function ParseSerializedPage($text, $default_pagename)
214 if (!preg_match('/^a:\d+:{[si]:\d+/', $text))
217 $pagehash = unserialize($text);
219 // Split up pagehash into four parts:
222 // page-level meta-data
223 // revision-level meta-data
225 if (!defined('FLAG_PAGE_LOCKED'))
226 define('FLAG_PAGE_LOCKED', 1);
227 $pageinfo = array('pagedata' => array(),
228 'versiondata' => array());
230 $pagedata = &$pageinfo['pagedata'];
231 $versiondata = &$pageinfo['versiondata'];
234 if (empty($pagehash['pagename']))
235 $pagehash['pagename'] = $default_pagename;
236 if (empty($pagehash['author']))
237 $pagehash['author'] = $GLOBALS['user']->id();
240 foreach ($pagehash as $key => $value) {
244 $pageinfo[$key] = $value;
247 $pageinfo[$key] = join("\n", $value);
249 if (($value & FLAG_PAGE_LOCKED) != 0)
250 $pagedata['locked'] = 'yes';
253 $pagedata[$key] = $value;
256 $versiondata['mtime'] = $value;
259 $versiondata[$key] = $value;
266 function SortByPageVersion ($a, $b) {
267 return $a['version'] - $b['version'];
270 function LoadFile ($dbi, $filename, $text = false, $mtime = false)
272 if (!is_string($text)) {
274 $stat = stat($filename);
276 $text = implode("", file($filename));
279 set_time_limit(30); // Reset watchdog.
281 // FIXME: basename("filewithnoslashes") seems to return garbage sometimes.
282 $basename = basename("/dummy/" . $filename);
285 $mtime = time(); // Last resort.
287 $defaults = array('author' => $GLOBALS['user']->id(),
288 'pagename' => rawurldecode($basename));
290 $default_pagename = rawurldecode($basename);
292 if ( ($parts = ParseMimeifiedPages($text)) ) {
293 usort($parts, 'SortByPageVersion');
294 foreach ($parts as $pageinfo)
295 SavePage($dbi, $pageinfo, "MIME file $filename", $basename);
297 else if ( ($pageinfo = ParseSerializedPage($text, $default_pagename)) ) {
298 SavePage($dbi, $pageinfo, "Serialized file $filename", $basename);
301 // Assume plain text file.
302 $pageinfo = array('pagename' => $default_pagename,
303 'pagedata' => array(),
305 => array('author' => $GLOBALS['user']->id()),
307 => preg_replace('/[ \t\r]*\n/', "\n", chop($text))
309 SavePage($dbi, $pageinfo, "plain file $filename", $basename);
313 function LoadZip ($dbi, $zipfile, $files = false, $exclude = false)
315 $zip = new ZipReader($zipfile);
316 while (list ($fn, $data, $attrib) = $zip->readFile())
318 // FIXME: basename("filewithnoslashes") seems to return garbage sometimes.
319 $fn = basename("/dummy/" . $fn);
320 if ( ($files && !in_array($fn, $files))
321 || ($exclude && in_array($fn, $exclude)) )
323 print Element('dt', LinkExistingWikiWord($fn)) . QElement('dd', 'Skipping');
327 LoadFile($dbi, $fn, $data, $attrib['mtime']);
331 function LoadDir ($dbi, $dirname, $files = false, $exclude = false)
333 $handle = opendir($dir = $dirname);
334 while ($fn = readdir($handle))
336 if (filetype("$dir/$fn") != 'file')
339 if ( ($files && !in_array($fn, $files))
340 || ($exclude && in_array($fn, $exclude)) )
342 print Element('dt', LinkExistingWikiWord($fn)) . QElement('dd', 'Skipping');
346 LoadFile($dbi, "$dir/$fn");
351 function IsZipFile ($filename_or_fd)
353 // See if it looks like zip file
354 if (is_string($filename_or_fd))
356 $fd = fopen($filename_or_fd, "rb");
357 $magic = fread($fd, 4);
362 $fpos = ftell($filename_or_fd);
363 $magic = fread($filename_or_fd, 4);
364 fseek($filename_or_fd, $fpos);
367 return $magic == ZIP_LOCHEAD_MAGIC || $magic == ZIP_CENTHEAD_MAGIC;
371 function LoadAny ($dbi, $file_or_dir, $files = false, $exclude = false)
373 $type = filetype($file_or_dir);
377 LoadDir($dbi, $file_or_dir, $files, $exclude);
379 else if ($type != 'file' && !preg_match('/^(http|ftp):/', $file_or_dir))
381 ExitWiki("Bad file type: $type");
383 else if (IsZipFile($file_or_dir))
385 LoadZip($dbi, $file_or_dir, $files, $exclude);
387 else /* if (!$files || in_array(basename($file_or_dir), $files)) */
389 LoadFile($dbi, $file_or_dir);
393 function LoadFileOrDir ($dbi, $request)
395 $source = $request->getArg('source');
396 StartLoadDump("Loading '$source'");
398 LoadAny($dbi, $source/*, false, array(gettext('RecentChanges'))*/);
403 function SetupWiki ($dbi)
405 global $GenericPages, $LANG, $user;
407 //FIXME: This is a hack
408 $user->userid = 'The PhpWiki programming team';
410 StartLoadDump('Loading up virgin wiki');
413 LoadAny($dbi, FindLocalizedFile(WIKI_PGSRC)/*, false, $ignore*/);
415 LoadAny($dbi, FindFile(DEFAULT_WIKI_PGSRC), $GenericPages/*, $ignore*/);
421 function LoadPostFile ($dbi, $request)
423 $upload = $request->getUploadedFile('file');
426 ExitWiki('No uploade file to upload?');
428 // Dump http headers.
429 $fd = fopen($tmp_name, "rb");
430 while ( ($header = fgets($fd, 4096)) )
431 if (trim($header) == '')
434 StartLoadDump("Uploading " . $upload->getName());
438 LoadZip($dbi, $upload->open(), false, array(gettext('RecentChanges')));
440 Loadfile($dbi, $upload->getName(), $upload->getContents());
451 // c-hanging-comment-ender-p: nil
452 // indent-tabs-mode: nil