// Wiki.java // ----------------------- // part of the AnomicHTTPD caching proxy // (C) by Michael Peter Christen; mc@yacy.net // first published on http://www.anomic.de // Frankfurt, Germany, 2004 // //$LastChangedDate$ //$LastChangedRevision$ //$LastChangedBy$ // // This program is free software; you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation; either version 2 of the License, or // (at your option) any later version. // // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // // You should have received a copy of the GNU General Public License // along with this program; if not, write to the Free Software // Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA // Contains contributions from Alexander Schier [AS] // and Marc Nause [MN] // You must compile this file with // javac -classpath .:../classes Wiki.java // if the shell's current path is HTROOT import java.io.IOException; import java.util.Date; import java.util.HashMap; import java.util.Iterator; import java.util.Map; import net.yacy.cora.date.GenericFormatter; import net.yacy.cora.document.encoding.UTF8; import net.yacy.cora.protocol.Domains; import net.yacy.cora.protocol.HeaderFramework; import net.yacy.cora.protocol.RequestHeader; import net.yacy.cora.util.ByteBuffer; import net.yacy.data.Diff; import net.yacy.data.wiki.WikiBoard; import net.yacy.peers.NewsPool; import net.yacy.peers.Seed; import net.yacy.search.Switchboard; import net.yacy.server.serverObjects; import net.yacy.server.serverSwitch; public class Wiki { private static final String ANONYMOUS = "anonymous"; //private static String ListLevel = ""; //private static String numListLevel = ""; public static String dateString(final Date date) { return GenericFormatter.SIMPLE_FORMATTER.format(date); } public static serverObjects respond(final RequestHeader header, serverObjects post, final serverSwitch env) throws IOException { final Switchboard sb = (Switchboard) env; final serverObjects prop = new serverObjects(); if (post == null) { post = new serverObjects(); post.put("page", "start"); } prop.put("topmenu", sb.getConfigBool("publicTopmenu", true) ? 1 : 0); String access = sb.getConfig("WikiAccess", "admin"); final String pagename = get(post, "page", "start"); final String ip = get(post, HeaderFramework.CONNECTION_PROP_CLIENTIP, Domains.LOCALHOST); String author = get(post, "author", ANONYMOUS); if (author.equals(ANONYMOUS)) { author = WikiBoard.guessAuthor(ip); if (author == null) { author = (sb.peers.mySeed() == null) ? ANONYMOUS : sb.peers.mySeed().get(Seed.NAME, ANONYMOUS); } } if (post != null && post.containsKey("access")) { // only the administrator may change the access right if (!sb.verifyAuthentication(header)) { // check access right for admin prop.authenticationRequired(); return prop; } access = post.get("access", "admin"); sb.setConfig("WikiAccess", access); } if (access.equals("admin")) { prop.put("mode_access", "0"); } else if (access.equals("all")) { prop.put("mode_access", "1"); } WikiBoard.Entry page = sb.wikiDB.read(pagename); if (post != null && post.containsKey("submit")) { if ((access.equals("admin") && (!sb.verifyAuthentication(header)))) { // check access right for admin prop.authenticationRequired(); return prop; } // store a new page byte[] content; content = UTF8.getBytes(post.get("content", "")); final WikiBoard.Entry newEntry = sb.wikiDB.newEntry(pagename, author, ip, post.get("reason", "edit"), content); sb.wikiDB.write(newEntry); // create a news message final Map<String, String> map = new HashMap<String, String>(); map.put("page", pagename); map.put("author", author.replace(',', ' ')); if (!sb.isRobinsonMode() && post.get("content", "").trim().length() > 0 && !ByteBuffer.equals(page.page(), content)) { sb.peers.newsPool.publishMyNews(sb.peers.mySeed(), NewsPool.CATEGORY_WIKI_UPDATE, map); } page = newEntry; prop.putHTML(serverObjects.ACTION_LOCATION, "Wiki.html?page=" + pagename); prop.put(serverObjects.ACTION_LOCATION, prop.get(serverObjects.ACTION_LOCATION)); } if (post != null && post.containsKey("edit")) { if ((access.equals("admin") && (!sb.verifyAuthentication(header)))) { // check access right for admin prop.authenticationRequired(); return prop; } prop.put("mode", "1"); //edit prop.putHTML("mode_author", author); prop.putHTML("mode_page-code", UTF8.String(page.page())); prop.putHTML("mode_pagename", pagename); } //contributed by [MN] else if (post != null && post.containsKey("preview")) { // preview the page prop.put("mode", "2");//preview prop.putHTML("mode_pagename", pagename); prop.putHTML("mode_author", author); prop.put("mode_date", dateString(new Date())); /* We do not fill hostport parameter : relative links should stay relative as it is more reliable * when the peer is behind any kind of reverse Proxy */ prop.putWiki("mode_page", post.get("content", "")); prop.putHTML("mode_page-code", post.get("content", "")); } //end contrib of [MN] else if (post != null && post.containsKey("index")) { // view an index prop.put("mode", "3"); //Index try { final Iterator<byte[]> i = sb.wikiDB.keys(true); int count=0; while (i.hasNext()) { final String subject = UTF8.String(i.next()); final WikiBoard.Entry entry = sb.wikiDB.read(subject); prop.putHTML("mode_pages_" + count + "_name",WikiBoard.webalize(subject)); prop.putHTML("mode_pages_" + count + "_subject", subject); prop.put("mode_pages_" + count + "_date", dateString(entry.date())); prop.putHTML("mode_pages_" + count + "_author", entry.author()); count++; } prop.put("mode_pages", count); } catch (final IOException e) { prop.put("mode_error", "1"); //IO Error reading Wiki prop.putHTML("mode_error_message", e.getMessage()); } prop.putHTML("mode_pagename", pagename); } else if (post != null && post.containsKey("diff")) { // Diff prop.put("mode", "4"); prop.putHTML("mode_page", pagename); prop.putHTML("mode_error_page", pagename); try { final Iterator<byte[]> it = sb.wikiDB.keysBkp(true); WikiBoard.Entry entry; WikiBoard.Entry oentry = null; WikiBoard.Entry nentry = null; int count = 0; boolean oldselected = false, newselected = false; while (it.hasNext()) { entry = sb.wikiDB.readBkp(UTF8.String(it.next())); prop.put("mode_error_versions_" + count + "_date", WikiBoard.dateString(entry.date())); prop.put("mode_error_versions_" + count + "_fdate", dateString(entry.date())); if (WikiBoard.dateString(entry.date()).equals(post.get("old", null))) { prop.put("mode_error_versions_" + count + "_oldselected", "1"); oentry = entry; oldselected = true; } else if (WikiBoard.dateString(entry.date()).equals(post.get("new", null))) { prop.put("mode_error_versions_" + count + "_newselected", "1"); nentry = entry; newselected = true; } count++; } count--; // don't show current version if (!oldselected) { // select latest old entry prop.put("mode_error_versions_" + (count - 1) + "_oldselected", "1"); } if (!newselected) { // select latest new entry (== current) prop.put("mode_error_curselected", "1"); } if (count == 0) { prop.put("mode_error", "2"); // no entries found } else { prop.put("mode_error_versions", count); } entry = sb.wikiDB.read(pagename); if (entry != null) { prop.put("mode_error_curdate", WikiBoard.dateString(entry.date())); prop.put("mode_error_curfdate", dateString(entry.date())); } if (nentry == null) { nentry = entry; } if (post.containsKey("compare") && oentry != null && nentry != null) { // TODO: split into paragraphs and compare them with the same diff-algo final Diff diff = new Diff( UTF8.String(oentry.page()), UTF8.String(nentry.page()), 3); prop.put("mode_versioning_diff", net.yacy.data.Diff.toHTML(new Diff[] { diff })); prop.put("mode_versioning", "1"); } else if (post.containsKey("viewold") && oentry != null) { prop.put("mode_versioning", "2"); prop.putHTML("mode_versioning_pagename", pagename); prop.putHTML("mode_versioning_author", oentry.author()); prop.put("mode_versioning_date", dateString(oentry.date())); /* We do not fill hostport parameter : relative links should stay relative as it is more reliable * when the peer is behind any kind of reverse Proxy */ prop.putWiki("mode_versioning_page", oentry.page()); prop.putHTML("mode_versioning_page-code", UTF8.String(oentry.page())); } } catch (final IOException e) { prop.put("mode_error", "1"); //IO Error reading Wiki prop.putHTML("mode_error_message", e.getMessage()); } } else { // show page prop.put("mode", "0"); //viewing prop.putHTML("mode_pagename", pagename); prop.putHTML("mode_author", page.author()); prop.put("mode_date", dateString(page.date())); /* We do not fill hostport parameter : relative links should stay relative as it is more reliable * when the peer is behind any kind of reverse Proxy */ prop.putWiki("mode_page", page.page()); prop.put("controls", "0"); prop.putHTML("controls_pagename", pagename); } // return rewrite properties return prop; } /** * get key from post, use dflt if (not present or post == null) * * @param post * @param string * @param string2 * @return */ private static String get(final serverObjects post, final String key, final String dflt) { return (post == null ? dflt : post.get(key, dflt)); } }