*) added comments

*) more beautyful and easier to understand code (IMO)
*) added display= parameter to a lot of links in Wiki.html

git-svn-id: https://svn.berlios.de/svnroot/repos/yacy/trunk@7226 6c8d7289-2bf4-0310-a012-ef5d649a1542
pull/1/head
low012 15 years ago
parent 70576e88d2
commit b9f405d1e8

@ -17,20 +17,19 @@
#(mode)# #(mode)#
<!-- 0: viewing --> <!-- 0: viewing -->
<div class="Post"> <div class="Post">
<h2 class="PostSubject">YaCyWiki page: #[pagename]#</h2> <h2 class="PostSubject">YaCyWiki page: #[pagename]#</h2>
<div class="PostBody"> <div class="PostBody">
#[page]# #[page]#
</div> </div>
<p class="PostInfo"> <p class="PostInfo">
last edited by #[author]# | change date #[date]# | <a href="Wiki.html?page=#[pagename]#&amp;edit=Edit">Edit</a> #(access)#(only granted to admin)::#(/access)# last edited by #[author]# | change date #[date]# | <a href="Wiki.html?page=#[pagename]#&amp;display=#[display]#&amp;edit=Edit">Edit</a> #(access)#(only granted to admin)::#(/access)#
</p> </p>
</div> </div>
<p class="Navigation"> <p class="Navigation">
<a href="Wiki.html">Start Page</a> - <a href="Wiki.html?display=#[display]#">Start Page</a> -
<a href="Wiki.html?index=Index">Index</a> - <a href="Wiki.html?index=Index&amp;display=#[display]#">Index</a> -
<a href="Wiki.html?page=#[pagename]#&amp;diff=">Versions</a> - <a href="Wiki.html?page=#[pagename]#&amp;diff=&amp;display=#[display]#">Versions</a> -
</p> </p>
<form method="post" action="Wiki.html"><div> <form method="post" action="Wiki.html"><div>
<label for="access">Grant Write Access to</label> <label for="access">Grant Write Access to</label>
@ -38,6 +37,7 @@
</div></form> </div></form>
:: ::
<!-- 1: edit --> <!-- 1: edit -->
<h2>Edit</h2>
<form action="Wiki.html" method="post" enctype="multipart/form-data" accept-charset="UTF-8"> <form action="Wiki.html" method="post" enctype="multipart/form-data" accept-charset="UTF-8">
<fieldset> <fieldset>
<dl> <dl>
@ -54,6 +54,7 @@
You can use <a href="WikiHelp.html" onclick="window.open('WikiHelp.html','WikiHelp','toolbar=0,scrollbars=1,location=0,statusbar=0,menubar=0,resizable=1,width=640,height=480');return false;">Wiki Code</a> here. You can use <a href="WikiHelp.html" onclick="window.open('WikiHelp.html','WikiHelp','toolbar=0,scrollbars=1,location=0,statusbar=0,menubar=0,resizable=1,width=640,height=480');return false;">Wiki Code</a> here.
</p> </p>
<input type="hidden" name="page" value="#[pagename]#" /> <input type="hidden" name="page" value="#[pagename]#" />
<input type="hidden" name="display" value="#[display]#" />
<input type="hidden" name="reason" value="edit" /> <input type="hidden" name="reason" value="edit" />
<input type="submit" name="submit" value="Submit" /> <input type="submit" name="submit" value="Submit" />
<input type="submit" name="preview" value="Preview" /> <input type="submit" name="preview" value="Preview" />
@ -96,6 +97,7 @@
You can use <a href="WikiHelp.html" onclick="window.open('WikiHelp.html','WikiHelp','toolbar=0,scrollbars=1,location=0,statusbar=0,menubar=0,resizable=1,width=640,height=480');return false;">Wiki Code</a> here. You can use <a href="WikiHelp.html" onclick="window.open('WikiHelp.html','WikiHelp','toolbar=0,scrollbars=1,location=0,statusbar=0,menubar=0,resizable=1,width=640,height=480');return false;">Wiki Code</a> here.
</p> </p>
<input type="hidden" name="page" value="#[pagename]#" /> <input type="hidden" name="page" value="#[pagename]#" />
<input type="hidden" name="display" value="#[display]#" />
<input type="hidden" name="reason" value="edit" /> <input type="hidden" name="reason" value="edit" />
<input type="submit" name="submit" value="Submit" /> <input type="submit" name="submit" value="Submit" />
<input type="submit" name="preview" value="Preview" /> <input type="submit" name="preview" value="Preview" />
@ -104,6 +106,7 @@
</form> </form>
:: ::
<!-- 3: Index --> <!-- 3: Index -->
<h2>Index</h2>
<table border="0" cellpadding="5" cellspacing="0"> <table border="0" cellpadding="5" cellspacing="0">
<tr class="TableHeader"> <tr class="TableHeader">
<td>Subject</td> <td>Subject</td>
@ -112,7 +115,7 @@
</tr> </tr>
#{pages}# #{pages}#
<tr class="TableCellLight"> <tr class="TableCellLight">
<td><a href="Wiki.html?page=#[name]#">#[subject]#</a></td> <td><a href="Wiki.html?page=#[name]#&amp;display=#[display]#">#[subject]#</a></td>
<td>#[date]#</td> <td>#[date]#</td>
<td>#[author]#</td> <td>#[author]#</td>
</tr> </tr>
@ -120,14 +123,16 @@
</table> </table>
#(error)#::<p class="error">IO Error reading wiki database: #[message]#</p>#(/error)# #(error)#::<p class="error">IO Error reading wiki database: #[message]#</p>#(/error)#
<p class="Navigation"> <p class="Navigation">
<a href="Wiki.html">Start Page</a> <a href="Wiki.html?display=#[display]#">Start Page</a>
</p> </p>
:: ::
<!-- 4: Diff --> <!-- 4: Diff -->
<h2>Versions</h2>
#(error)#<!-- no error --> #(error)#<!-- no error -->
<form method="get" action="Wiki.html"> <form method="get" action="Wiki.html">
<fieldset><legend>Select versions of page '#[page]#</legend> <fieldset><legend>Select versions of page '#[page]#'</legend>
<input type="hidden" name="page" value="#[page]#" /> <input type="hidden" name="page" value="#[page]#" />
<input type="hidden" name="display" value="#[display]#" />
<input type="hidden" name="diff" value="" /> <input type="hidden" name="diff" value="" />
<dl> <dl>
<dt><label for="old">Compare version from</label>:</dt> <dt><label for="old">Compare version from</label>:</dt>
@ -149,6 +154,7 @@
</fieldset> </fieldset>
</form> </form>
::<!-- IO error --> ::<!-- IO error -->
<h2>Error</h2>
<p class="error">IO Error reading wiki database: #[message]#</p> <p class="error">IO Error reading wiki database: #[message]#</p>
::<!-- no entries available for #[name]# --> ::<!-- no entries available for #[name]# -->
#(/error)# #(/error)#
@ -189,6 +195,7 @@
<a href="WikiHelp.html" onclick="window.open('WikiHelp.html','WikiHelp','toolbar=0,scrollbars=1,location=0,statusbar=0,menubar=0,resizable=1,width=640,height=480');return false;">Wiki Code</a> here. <a href="WikiHelp.html" onclick="window.open('WikiHelp.html','WikiHelp','toolbar=0,scrollbars=1,location=0,statusbar=0,menubar=0,resizable=1,width=640,height=480');return false;">Wiki Code</a> here.
</p> </p>
<input type="hidden" name="page" value="#[pagename]#" /> <input type="hidden" name="page" value="#[pagename]#" />
<input type="hidden" name="display" value="#[display]#" />
<input type="hidden" name="reason" value="edit" /> <input type="hidden" name="reason" value="edit" />
<input type="submit" name="submit" value="Submit" /> <input type="submit" name="submit" value="Submit" />
<input type="submit" name="preview" value="Preview" /> <input type="submit" name="preview" value="Preview" />
@ -197,8 +204,8 @@
</form> </form>
#(/versioning)# #(/versioning)#
<p class="Navigation"> <p class="Navigation">
<a href="Wiki.html">Start Page</a> - <a href="Wiki.html?display=#[display]#">Start Page</a> -
<a href="Wiki.html?page=#[page]#">Return to #[page]#</a> <a href="Wiki.html?page=#[page]#&amp;display=#[display]#">Return to #[page]#</a>
</p> </p>
#(/mode)# #(/mode)#

@ -47,8 +47,10 @@ import de.anomic.search.Switchboard;
import de.anomic.server.serverObjects; import de.anomic.server.serverObjects;
import de.anomic.server.serverSwitch; import de.anomic.server.serverSwitch;
import de.anomic.yacy.yacyNewsPool; import de.anomic.yacy.yacyNewsPool;
import java.util.Map;
public class Wiki { public class Wiki {
private static final String ANONYMOUS = "anonymous";
//private static String ListLevel = ""; //private static String ListLevel = "";
//private static String numListLevel = ""; //private static String numListLevel = "";
@ -74,12 +76,11 @@ public class Wiki {
String access = sb.getConfig("WikiAccess", "admin"); String access = sb.getConfig("WikiAccess", "admin");
final String pagename = get(post, "page", "start"); final String pagename = get(post, "page", "start");
final String ip = get(post, HeaderFramework.CONNECTION_PROP_CLIENTIP, "127.0.0.1"); final String ip = get(post, HeaderFramework.CONNECTION_PROP_CLIENTIP, "127.0.0.1");
String author = get(post, "author", "anonymous"); String author = get(post, "author", ANONYMOUS);
if (author.equals("anonymous")) { if (author.equals(ANONYMOUS)) {
author = wikiBoard.guessAuthor(ip); author = wikiBoard.guessAuthor(ip);
if (author == null) { if (author == null) {
if (sb.peers.mySeed() == null) author = "anonymous"; author = (sb.peers.mySeed() == null) ? ANONYMOUS : sb.peers.mySeed().get("Name", ANONYMOUS);
else author = sb.peers.mySeed().get("Name", "anonymous");
} }
} }
@ -94,10 +95,13 @@ public class Wiki {
access = post.get("access", "admin"); access = post.get("access", "admin");
sb.setConfig("WikiAccess", access); sb.setConfig("WikiAccess", access);
} }
if (access.equals("admin")) prop.put("mode_access", "0"); if (access.equals("admin")) {
if (access.equals("all")) prop.put("mode_access", "1"); prop.put("mode_access", "0");
} else if (access.equals("all")) {
prop.put("mode_access", "1");
}
wikiBoard.entry page = sb.wikiDB.read(pagename); wikiBoard.Entry page = sb.wikiDB.read(pagename);
if (post != null && post.containsKey("submit")) { if (post != null && post.containsKey("submit")) {
@ -114,18 +118,22 @@ public class Wiki {
} catch (final UnsupportedEncodingException e) { } catch (final UnsupportedEncodingException e) {
content = post.get("content", "").getBytes(); content = post.get("content", "").getBytes();
} }
final wikiBoard.entry newEntry = sb.wikiDB.newEntry(pagename, author, ip, post.get("reason", "edit"), content); final wikiBoard.Entry newEntry = sb.wikiDB.newEntry(pagename, author, ip, post.get("reason", "edit"), content);
sb.wikiDB.write(newEntry); sb.wikiDB.write(newEntry);
// create a news message // create a news message
final HashMap<String, String> map = new HashMap<String, String>(); final Map<String, String> map = new HashMap<String, String>();
map.put("page", pagename); map.put("page", pagename);
map.put("author", author.replace(',', ' ')); map.put("author", author.replace(',', ' '));
if (post.get("content", "").trim().length() > 0 && !page.page().equals(content)) if (post.get("content", "").trim().length() > 0 && !new String(page.page()).equals(new String(content))) {
sb.peers.newsPool.publishMyNews(sb.peers.mySeed(), yacyNewsPool.CATEGORY_WIKI_UPDATE, map); sb.peers.newsPool.publishMyNews(sb.peers.mySeed(), yacyNewsPool.CATEGORY_WIKI_UPDATE, map);
}
page = newEntry; page = newEntry;
prop.putHTML("LOCATION", "/Wiki.html?page=" + pagename); prop.putHTML("LOCATION", "/Wiki.html?page=" + pagename);
prop.put("LOCATION", prop.get("LOCATION") + "&display=" + display);
} }
prop.put("mode_display", display);
if (post != null && post.containsKey("edit")) { if (post != null && post.containsKey("edit")) {
if ((access.equals("admin") && (!sb.verifyAuthentication(header, true)))) { if ((access.equals("admin") && (!sb.verifyAuthentication(header, true)))) {
// check access right for admin // check access right for admin
@ -139,6 +147,7 @@ public class Wiki {
prop.putHTML("mode_author", author); prop.putHTML("mode_author", author);
prop.putHTML("mode_page-code", new String(page.page(), "UTF-8")); prop.putHTML("mode_page-code", new String(page.page(), "UTF-8"));
prop.putHTML("mode_pagename", pagename); prop.putHTML("mode_pagename", pagename);
prop.put("mode_display", display);
} catch (final UnsupportedEncodingException e) {} } catch (final UnsupportedEncodingException e) {}
} }
@ -147,6 +156,7 @@ public class Wiki {
// preview the page // preview the page
prop.put("mode", "2");//preview prop.put("mode", "2");//preview
prop.putHTML("mode_pagename", pagename); prop.putHTML("mode_pagename", pagename);
prop.put("mode_display", display);
prop.putHTML("mode_author", author); prop.putHTML("mode_author", author);
prop.put("mode_date", dateString(new Date())); prop.put("mode_date", dateString(new Date()));
prop.putWiki("mode_page", post.get("content", "")); prop.putWiki("mode_page", post.get("content", ""));
@ -157,14 +167,12 @@ public class Wiki {
else if (post != null && post.containsKey("index")) { else if (post != null && post.containsKey("index")) {
// view an index // view an index
prop.put("mode", "3"); //Index prop.put("mode", "3"); //Index
String subject;
try { try {
final Iterator<byte[]> i = sb.wikiDB.keys(true); final Iterator<byte[]> i = sb.wikiDB.keys(true);
wikiBoard.entry entry;
int count=0; int count=0;
while (i.hasNext()) { while (i.hasNext()) {
subject = new String(i.next()); final String subject = new String(i.next());
entry = sb.wikiDB.read(subject); final wikiBoard.Entry entry = sb.wikiDB.read(subject);
prop.putHTML("mode_pages_" + count + "_name",wikiBoard.webalize(subject)); prop.putHTML("mode_pages_" + count + "_name",wikiBoard.webalize(subject));
prop.putHTML("mode_pages_" + count + "_subject", subject); prop.putHTML("mode_pages_" + count + "_subject", subject);
prop.put("mode_pages_" + count + "_date", dateString(entry.date())); prop.put("mode_pages_" + count + "_date", dateString(entry.date()));
@ -177,6 +185,7 @@ public class Wiki {
prop.putHTML("mode_error_message", e.getMessage()); prop.putHTML("mode_error_message", e.getMessage());
} }
prop.putHTML("mode_pagename", pagename); prop.putHTML("mode_pagename", pagename);
prop.put("mode_display", display);
} }
else if (post != null && post.containsKey("diff")) { else if (post != null && post.containsKey("diff")) {
@ -184,12 +193,13 @@ public class Wiki {
prop.put("mode", "4"); prop.put("mode", "4");
prop.putHTML("mode_page", pagename); prop.putHTML("mode_page", pagename);
prop.putHTML("mode_error_page", pagename); prop.putHTML("mode_error_page", pagename);
prop.put("mode_error_display", display);
try { try {
final Iterator<byte[]> it = sb.wikiDB.keysBkp(true); final Iterator<byte[]> it = sb.wikiDB.keysBkp(true);
wikiBoard.entry entry; wikiBoard.Entry entry;
wikiBoard.entry oentry = null; wikiBoard.Entry oentry = null;
wikiBoard.entry nentry = null; wikiBoard.Entry nentry = null;
int count = 0; int count = 0;
boolean oldselected = false, newselected = false; boolean oldselected = false, newselected = false;
while (it.hasNext()) { while (it.hasNext()) {
@ -209,10 +219,12 @@ public class Wiki {
} }
count--; // don't show current version count--; // don't show current version
if (!oldselected) // select latest old entry if (!oldselected) { // select latest old entry
prop.put("mode_error_versions_" + (count - 1) + "_oldselected", "1"); prop.put("mode_error_versions_" + (count - 1) + "_oldselected", "1");
if (!newselected) // select latest new entry (== current) }
if (!newselected) { // select latest new entry (== current)
prop.put("mode_error_curselected", "1"); prop.put("mode_error_curselected", "1");
}
if (count == 0) { if (count == 0) {
prop.put("mode_error", "2"); // no entries found prop.put("mode_error", "2"); // no entries found
@ -226,7 +238,9 @@ public class Wiki {
prop.put("mode_error_curfdate", dateString(entry.date())); prop.put("mode_error_curfdate", dateString(entry.date()));
} }
if (nentry == null) nentry = entry; if (nentry == null) {
nentry = entry;
}
if (post.containsKey("compare") && oentry != null && nentry != null) { if (post.containsKey("compare") && oentry != null && nentry != null) {
// TODO: split into paragraphs and compare them with the same diff-algo // TODO: split into paragraphs and compare them with the same diff-algo
final diff diff = new diff( final diff diff = new diff(
@ -237,6 +251,7 @@ public class Wiki {
} else if (post.containsKey("viewold") && oentry != null) { } else if (post.containsKey("viewold") && oentry != null) {
prop.put("mode_versioning", "2"); prop.put("mode_versioning", "2");
prop.putHTML("mode_versioning_pagename", pagename); prop.putHTML("mode_versioning_pagename", pagename);
prop.put("mode_versioning_display", display);
prop.putHTML("mode_versioning_author", oentry.author()); prop.putHTML("mode_versioning_author", oentry.author());
prop.put("mode_versioning_date", dateString(oentry.date())); prop.put("mode_versioning_date", dateString(oentry.date()));
prop.putWiki("mode_versioning_page", oentry.page()); prop.putWiki("mode_versioning_page", oentry.page());
@ -252,6 +267,7 @@ public class Wiki {
// show page // show page
prop.put("mode", "0"); //viewing prop.put("mode", "0"); //viewing
prop.putHTML("mode_pagename", pagename); prop.putHTML("mode_pagename", pagename);
prop.put("mode_display", display);
prop.putHTML("mode_author", page.author()); prop.putHTML("mode_author", page.author());
prop.put("mode_date", dateString(page.date())); prop.put("mode_date", dateString(page.date()));
prop.putWiki("mode_page", page.page()); prop.putWiki("mode_page", page.page());

@ -41,7 +41,9 @@ import net.yacy.kelondro.logging.Log;
import net.yacy.kelondro.order.Base64Order; import net.yacy.kelondro.order.Base64Order;
import net.yacy.kelondro.order.NaturalOrder; import net.yacy.kelondro.order.NaturalOrder;
/**
*
*/
public class wikiBoard { public class wikiBoard {
public static final int keyLength = 64; public static final int keyLength = 64;
@ -54,10 +56,16 @@ public class wikiBoard {
SimpleFormatter.setTimeZone(TimeZone.getTimeZone("GMT")); SimpleFormatter.setTimeZone(TimeZone.getTimeZone("GMT"));
} }
MapHeap datbase = null; private MapHeap datbase = null;
MapHeap bkpbase = null; private MapHeap bkpbase = null;
static HashMap<String, String> authors = new HashMap<String, String>(); private static final Map<String, String> AUTHORS = new HashMap<String, String>();
/**
* Constructor.
* @param actpath path of database which contains current wiki data.
* @param bkppath path of backup database.
* @throws IOException if error occurs during HDD access.
*/
public wikiBoard(final File actpath, final File bkppath) throws IOException { public wikiBoard(final File actpath, final File bkppath) throws IOException {
new File(actpath.getParent()).mkdirs(); new File(actpath.getParent()).mkdirs();
if (datbase == null) { if (datbase == null) {
@ -71,224 +79,329 @@ public class wikiBoard {
} }
} }
/**
* Gets total number of entries of wiki DB and DB which contains backup entries.
* @return number of entries in wiki plus number of old entries.
*/
public int sizeOfTwo() { public int sizeOfTwo() {
return datbase.size() + bkpbase.size(); return datbase.size() + bkpbase.size();
} }
/**
* Gets number of entries of wiki DB.
* @return number of entries in wiki.
*/
public int size() { public int size() {
return datbase.size(); return datbase.size();
} }
/**
* Closes database files.
*/
public void close() { public void close() {
datbase.close(); datbase.close();
bkpbase.close(); bkpbase.close();
} }
/**
* Gets current date.
* @return current date.
*/
static String dateString() { static String dateString() {
return dateString(new Date()); return dateString(new Date());
} }
/**
* Gets String representation of a Date.
* @param date the Date.
* @return String representation of Date.
*/
public static String dateString(final Date date) { public static String dateString(final Date date) {
synchronized (SimpleFormatter) { synchronized (SimpleFormatter) {
return SimpleFormatter.format(date); return SimpleFormatter.format(date);
} }
} }
/**
* Gets normalized version of a key.
* @param key the key.
* @return normalized version of key.
*/
private static String normalize(final String key) { private static String normalize(final String key) {
return (key != null) ? key.trim().toLowerCase() : "null"; return (key != null) ? key.trim().toLowerCase() : "null";
} }
/**
* Normalizes key and replaces spaces by escape code.
* @param key the key.
* @return normalized and webalized version.
*/
public static String webalize(final String key) { public static String webalize(final String key) {
return (key != null) ? normalize(key).replaceAll(" ", "%20") : "null"; return (key != null) ? normalize(key).replaceAll(" ", "%20") : "null";
} }
/**
* Tries to guess the name of the author by a given IP address.
* @param ip the IP address.
* @return
*/
public static String guessAuthor(final String ip) { public static String guessAuthor(final String ip) {
final String author = authors.get(ip); final String author = AUTHORS.get(ip);
//yacyCore.log.logDebug("DEBUG: guessing author for ip = " + ip + " is '" + author + "', authors = " + authors.toString()); //yacyCore.log.logDebug("DEBUG: guessing author for ip = " + ip + " is '" + author + "', authors = " + authors.toString());
return author; return author;
} }
/**
* Adds an author name and a corresponding IP to internal Map.
* @param ip IP address of the author.
* @param author name of author.
*/
public static void setAuthor(final String ip, final String author) { public static void setAuthor(final String ip, final String author) {
authors.put(ip,author); AUTHORS.put(ip,author);
} }
public entry newEntry(final String subject, final String author, final String ip, final String reason, final byte[] page) throws IOException { /**
return new entry(normalize(subject), author, ip, reason, page); * Creates new Entry.
* @param subject subject of entry.
* @param author author of entry.
* @param ip IP address of author.
* @param reason reason for new Entry (for example "edit").
* @param page content of Entry.
* @return new Entry.
* @throws IOException
*/
public Entry newEntry(final String subject, final String author, final String ip, final String reason, final byte[] page) throws IOException {
return new Entry(normalize(subject), author, ip, reason, page);
} }
public class entry { /**
* Contains information of wiki page.
*/
public class Entry {
private static final String ANONYMOUS = "anonymous"; private static final String ANONYMOUS = "anonymous";
String key; private String key;
Map<String, String> record; private final Map<String, String> record;
public entry(final String subject, String author, String ip, String reason, final byte[] page) throws IOException { /**
record = new HashMap<String, String>(); * Constructor which creates new Entry using given information.
key = subject; * @param subject subject of Entry.
if (key.length() > keyLength) { * @param author author of Entry.
key = key.substring(0, keyLength); * @param ip IP address of author.
} * @param reason reason for new Entry (for example "edit").
record.put("date", dateString()); * @param page content of Entry.
if ((author == null) || (author.length() == 0)) { * @throws IOException
author = ANONYMOUS; */
} public Entry(final String subject, final String author, final String ip, final String reason, final byte[] page) throws IOException {
record.put("author", Base64Order.enhancedCoder.encode(author.getBytes("UTF-8"))); this.record = new HashMap<String, String>();
if ((ip == null) || (ip.length() == 0)) { this.key = subject.substring(0, Math.min((key != null) ? key.length() : 0, keyLength));
ip = ""; this.record.put("date", dateString());
} this.record.put("author", Base64Order.enhancedCoder.encodeString((author != null && author.length() > 0) ? author : ANONYMOUS));
record.put("ip", ip); this.record.put("ip", (ip != null && ip.length() > 0) ? ip : "");
if ((reason == null) || (reason.length() == 0)) { this.record.put("reason", Base64Order.enhancedCoder.encodeString((reason != null && reason.length() > 0) ? reason : ""));
reason = ""; this.record.put("page", (page != null) ? Base64Order.enhancedCoder.encode(page) : "");
} AUTHORS.put(ip, author);
record.put("reason", Base64Order.enhancedCoder.encode(reason.getBytes("UTF-8"))); }
if (page == null) {
record.put("page", ""); /**
} else { * Constructor which creates Entry using key and record.
record.put("page", Base64Order.enhancedCoder.encode(page)); * @param key key of Entry.
} * @param record record which contains data.
authors.put(ip, author); */
//System.out.println("DEBUG: setting author " + author + " for ip = " + ip + ", authors = " + authors.toString()); Entry(final String key, final Map<String, String> record) {
}
entry(final String key, final Map<String, String> record) {
this.key = key; this.key = key;
this.record = record; this.record = record;
} }
/**
* Gets subject of Entry.
* @return subject of entry.
*/
public String subject() { public String subject() {
return key; return key;
} }
/**
* Gets date of Entry.
* @return date of Entry.
*/
public Date date() { public Date date() {
Date ret;
try { try {
final String c = record.get("date"); final String c = record.get("date");
if (c == null) { if (c == null) {
System.out.println("DEBUG - ERROR: date field missing in wikiBoard"); System.out.println("DEBUG - ERROR: date field missing in wikiBoard");
return new Date(); ret = new Date();
} } else {
synchronized (SimpleFormatter) { synchronized (SimpleFormatter) {
return SimpleFormatter.parse(c); ret = SimpleFormatter.parse(c);
}
} }
} catch (final ParseException e) { } catch (final ParseException e) {
return new Date(); ret = new Date();
} }
return ret;
} }
/**
* Gets author of Entry.
* @return author of Entry.
*/
public String author() { public String author() {
final String a = record.get("author"); final String a = record.get("author");
final byte[] b; final byte[] b;
return (a != null && (b = Base64Order.enhancedCoder.decode(a)) != null) ? new String(b) : ANONYMOUS; return (a != null && (b = Base64Order.enhancedCoder.decode(a)) != null) ? new String(b) : ANONYMOUS;
} }
/**
* Gets reason for Entry.
* @return reason for Entry.
*/
public String reason() { public String reason() {
final String ret;
final String r = record.get("reason"); final String r = record.get("reason");
if (r == null) { if (r != null) {
return ""; final byte[] b;
} ret = ((b = Base64Order.enhancedCoder.decode(r)) != null) ? new String(b) : "unknown";
final byte[] b = Base64Order.enhancedCoder.decode(r); } else {
if (b == null) { ret = "";
return "unknown";
} }
return new String(b); return ret;
} }
/**
* Gets actual content of Entry.
* @return content of Entry.
*/
public byte[] page() { public byte[] page() {
final String m = record.get("page"); final String m = record.get("page");
final byte[] b; final byte[] b;
return (m != null && (b = Base64Order.enhancedCoder.decode(m)) != null) ? b : new byte[0]; return (m != null && (b = Base64Order.enhancedCoder.decode(m)) != null) ? b : new byte[0];
} }
/**
* Sets date of previous version of Entry.
* @param date date of previous version of Entry.
*/
void setAncestorDate(final Date date) { void setAncestorDate(final Date date) {
record.put("bkp", dateString(date)); record.put("bkp", dateString(date));
} }
/**
* Gets date of previous version of Entry.
* @return date of previous version of Entry.
*/
private Date getAncestorDate() { private Date getAncestorDate() {
Date ret = null;
try { try {
final String c = record.get("date"); final String c = record.get("date");
if (c == null) return null; if (c != null) {
synchronized (SimpleFormatter) { synchronized (SimpleFormatter) {
return SimpleFormatter.parse(c); ret = SimpleFormatter.parse(c);
}
} }
} catch (final ParseException e) { } catch (final ParseException e) {
return null; ret = null;
} }
return ret;
} }
/* /**
public boolean hasAncestor() { * Gets previous version of Entry.
Date ancDate = getAncestorDate(); * @return previous version of Entry.
if (ancDate == null) return false;
try {
return bkpbase.has(key + dateString(ancDate));
} catch (IOException e) {
return false;
}
}
*/ */
public Entry getAncestor() {
public entry getAncestor() {
final Date ancDate = getAncestorDate(); final Date ancDate = getAncestorDate();
return (ancDate == null) ? null : read(key + dateString(ancDate), bkpbase); return (ancDate == null) ? null : read(key + dateString(ancDate), bkpbase);
} }
/**
* Adds child of current Entry.
* @param subject subject of child of current Entry.
*/
void setChild(final String subject) { void setChild(final String subject) {
record.put("child", Base64Order.enhancedCoder.encode(subject.getBytes())); record.put("child", Base64Order.enhancedCoder.encode(subject.getBytes()));
} }
/**
* Gets name (= subject) of child of this Entry.
* @return name of child of this Entry.
*/
private String getChildName() { private String getChildName() {
final String c = record.get("child"); final String c = record.get("child");
final byte[] subject; final byte[] subject;
return (c != null && (subject = Base64Order.enhancedCoder.decode(c)) != null) ? new String(subject) : null; return (c != null && (subject = Base64Order.enhancedCoder.decode(c)) != null) ? new String(subject) : null;
} }
/**
* Tells if Entry has child.
* @return true if has child, else false.
*/
public boolean hasChild() { public boolean hasChild() {
final String c = record.get("child"); final String c = record.get("child");
return (c != null && Base64Order.enhancedCoder.decode(c) != null) ? true : false; return (c != null && Base64Order.enhancedCoder.decode(c) != null) ? true : false;
} }
public entry getChild() { /**
* Gets child of this Entry.
* @return child of this Entry.
*/
public Entry getChild() {
final String childName = getChildName(); final String childName = getChildName();
return (childName == null) ? null : read(childName, datbase); return (childName == null) ? null : read(childName, datbase);
} }
} }
public String write(final entry page) { /**
* Writes content of Entry to database and returns key.
* @param entry Entry to be written.
* @return key of Entry.
*/
public String write(final Entry entry) {
// writes a new page and returns key // writes a new page and returns key
String ret = null; String key = null;
try { try {
// first load the old page // first load the old page
final entry oldEntry = read(page.key); final Entry oldEntry = read(entry.key);
// set the bkp date of the new page to the date of the old page // set the bkp date of the new page to the date of the old page
final Date oldDate = oldEntry.date(); final Date oldDate = oldEntry.date();
page.setAncestorDate(oldDate); entry.setAncestorDate(oldDate);
oldEntry.setChild(page.subject()); oldEntry.setChild(entry.subject());
// write the backup // write the backup
//System.out.println("key = " + page.key); bkpbase.insert((entry.key + dateString(oldDate)).getBytes(), oldEntry.record);
//System.out.println("oldDate = " + oldDate);
//System.out.println("record = " + oldEntry.record.toString());
bkpbase.insert((page.key + dateString(oldDate)).getBytes(), oldEntry.record);
// write the new page // write the new page
datbase.insert(page.key.getBytes(), page.record); datbase.insert(entry.key.getBytes(), entry.record);
ret = page.key; key = entry.key;
} catch (final Exception e) { } catch (final Exception e) {
Log.logException(e); Log.logException(e);
} }
return ret; return key;
} }
public entry read(final String key) { /**
* Reads content of Entry from database.
* @param key key of Entry.
* @return Entry which contains data.
*/
public Entry read(final String key) {
return read(key, datbase); return read(key, datbase);
} }
entry read(String key, final MapHeap base) { /**
entry ret = null; * Reads content of Entry from database.
* @param key key of Entry.
* @param base database containing data.
* @return Entry which contains data.
*/
Entry read(final String key, final MapHeap base) {
Entry ret = null;
try { try {
key = normalize(key); String copyOfKey = normalize(key);
if (key.length() > keyLength) { if (copyOfKey.length() > keyLength) {
key = key.substring(0, keyLength); copyOfKey = copyOfKey.substring(0, keyLength);
} }
final Map<String, String> record = base.get(key.getBytes()); final Map<String, String> record = base.get(copyOfKey.getBytes());
ret = (record == null) ? newEntry(key, ANONYMOUS, "127.0.0.1", "New Page", "".getBytes()) : new entry(key, record); ret = (record == null) ? newEntry(copyOfKey, ANONYMOUS, "127.0.0.1", "New Page", "".getBytes()) : new Entry(copyOfKey, record);
} catch (final IOException e) { } catch (final IOException e) {
Log.logException(e); Log.logException(e);
} catch (RowSpaceExceededException e) { } catch (RowSpaceExceededException e) {
@ -297,24 +410,31 @@ public class wikiBoard {
return ret; return ret;
} }
public entry readBkp(final String key) { /**
* Reads old Entry from backup database.
* @param key key of Entry.
* @return the Entry.
*/
public Entry readBkp(final String key) {
return read(key, bkpbase); return read(key, bkpbase);
} }
/* /**
public boolean has(String key) { * Gets Iterator of keys in database.
try { * @param up
return datbase.has(normalize(key)); * @return keys of Entries in database.
} catch (IOException e) { * @throws IOException
return false;
}
}
*/ */
public Iterator<byte[]> keys(final boolean up) throws IOException { public Iterator<byte[]> keys(final boolean up) throws IOException {
return datbase.keys(up, false); return datbase.keys(up, false);
} }
/**
* Gets Iterator of keys in backup database.
* @param up
* @return keys of Entries in backup database.
* @throws IOException
*/
public Iterator<byte[]> keysBkp(final boolean up) throws IOException { public Iterator<byte[]> keysBkp(final boolean up) throws IOException {
return bkpbase.keys(up, false); return bkpbase.keys(up, false);
} }

Loading…
Cancel
Save