diff --git a/apps/syndie/java/src/net/i2p/syndie/Archive.java b/apps/syndie/java/src/net/i2p/syndie/Archive.java index 7d080ab86df9dce9aaa85f540219ace444fb8979..77fbdcbe2888817c9b4885399e67dec0da9978fc 100644 --- a/apps/syndie/java/src/net/i2p/syndie/Archive.java +++ b/apps/syndie/java/src/net/i2p/syndie/Archive.java @@ -40,6 +40,10 @@ public class Archive { public boolean accept(File dir, String name) { return name.endsWith(".snd"); } }; + static { + TimeZone.setDefault(TimeZone.getTimeZone("GMT")); + } + public Archive(I2PAppContext ctx, String rootDir, String cacheDir) { _context = ctx; _rootDir = new File(rootDir); @@ -319,7 +323,7 @@ public class Archive { public static String getEntryFilename(long entryId) { return entryId + ".snd"; } - private static SimpleDateFormat _dateFmt = new SimpleDateFormat("yyyyMMdd"); + private static SimpleDateFormat _dateFmt = new SimpleDateFormat("yyyyMMdd", Locale.UK); public static String getIndexName(long entryId, int numBytes) { try { synchronized (_dateFmt) { diff --git a/apps/syndie/java/src/net/i2p/syndie/ArchiveIndexer.java b/apps/syndie/java/src/net/i2p/syndie/ArchiveIndexer.java index a1b78f5837a087fdf201c9bd8bc83061aba840cd..35411350c76ddd5d87ac67654a16601241d40742 100644 --- a/apps/syndie/java/src/net/i2p/syndie/ArchiveIndexer.java +++ b/apps/syndie/java/src/net/i2p/syndie/ArchiveIndexer.java @@ -6,6 +6,7 @@ import java.util.*; import net.i2p.I2PAppContext; import net.i2p.data.*; import net.i2p.syndie.data.*; +import net.i2p.syndie.sml.*; /** * Dig through the archive to build an index @@ -53,6 +54,8 @@ class ArchiveIndexer { long totalSize = 0; int newBlogs = 0; + SMLParser parser = new SMLParser(); + for (int i = 0; i < blogs.size(); i++) { BlogInfo cur = (BlogInfo)blogs.get(i); Hash key = cur.getKey().calculateHash(); @@ -86,6 +89,16 @@ class ArchiveIndexer { newEntries++; newSize += entry.getCompleteSize(); } + HeaderReceiver rec = new HeaderReceiver(); + parser.parse(entry.getEntry().getText(), rec); + String reply = rec.getHeader(HTMLRenderer.HEADER_IN_REPLY_TO); + if (reply != null) { + BlogURI parent = new BlogURI(reply.trim()); + if ( (parent.getKeyHash() != null) && (parent.getEntryId() >= 0) ) + rv.addReply(parent, entry.getURI()); + else + System.err.println("Parent of " + entry.getURI() + " is not valid: [" + reply.trim() + "]"); + } } long lowestEntryId = -1; @@ -134,4 +147,43 @@ class ArchiveIndexer { return rv; } + + private static class HeaderReceiver implements SMLParser.EventReceiver { + private Properties _headers; + public HeaderReceiver() { _headers = null; } + public String getHeader(String name) { return (_headers != null ? _headers.getProperty(name) : null); } + public void receiveHeader(String header, String value) { + if (_headers == null) _headers = new Properties(); + _headers.setProperty(header, value); + } + + public void receiveAddress(String name, String schema, String location, String anchorText) {} + public void receiveArchive(String name, String description, String locationSchema, String location, String postingKey, String anchorText) {} + public void receiveAttachment(int id, String anchorText) {} + public void receiveBegin() {} + public void receiveBlog(String name, String blogKeyHash, String blogPath, long blogEntryId, List blogArchiveLocations, String anchorText) {} + public void receiveBold(String text) {} + public void receiveCode(String text, String codeLocationSchema, String codeLocation) {} + public void receiveCut(String summaryText) {} + public void receiveEnd() {} + public void receiveGT() {} + public void receiveH1(String text) {} + public void receiveH2(String text) {} + public void receiveH3(String text) {} + public void receiveH4(String text) {} + public void receiveH5(String text) {} + public void receiveHR() {} + public void receiveHeaderEnd() {} + public void receiveImage(String alternateText, int attachmentId) {} + public void receiveItalic(String text) {} + public void receiveLT() {} + public void receiveLeftBracket() {} + public void receiveLink(String schema, String location, String text) {} + public void receiveNewline() {} + public void receivePlain(String text) {} + public void receivePre(String text) {} + public void receiveQuote(String text, String whoQuoted, String quoteLocationSchema, String quoteLocation) {} + public void receiveRightBracket() {} + public void receiveUnderline(String text) {} + } } diff --git a/apps/syndie/java/src/net/i2p/syndie/data/ArchiveIndex.java b/apps/syndie/java/src/net/i2p/syndie/data/ArchiveIndex.java index 819fe3e8666f31d054fb57179c042186b088c527..66668fcd676d377429bb68c9cd14bbf0c39aa0ae 100644 --- a/apps/syndie/java/src/net/i2p/syndie/data/ArchiveIndex.java +++ b/apps/syndie/java/src/net/i2p/syndie/data/ArchiveIndex.java @@ -26,6 +26,8 @@ public class ArchiveIndex { protected List _newestBlogs; /** list of BlogURI objects */ protected List _newestEntries; + /** parent message to a set of replies, ordered with the oldest first */ + protected Map _replies; protected Properties _headers; public ArchiveIndex() { @@ -36,6 +38,7 @@ public class ArchiveIndex { _newestBlogs = new ArrayList(); _newestEntries = new ArrayList(); _headers = new Properties(); + _replies = Collections.synchronizedMap(new HashMap()); _generatedOn = -1; if (shouldLoad) setIsLocal("true"); @@ -138,7 +141,13 @@ public class ArchiveIndex { rv.add(getBlog(i)); return rv; } - + public List getReplies(BlogURI uri) { + Set replies = (Set)_replies.get(uri); + if (replies == null) return Collections.EMPTY_LIST; + synchronized (replies) { + return new ArrayList(replies); + } + } public void setLocation(String location) { try { File l = new File(location); @@ -255,14 +264,16 @@ public class ArchiveIndex { } if (tag != null) { if (!tag.equals(summary.tag)) { - System.out.println("Tag [" + summary.tag + "] does not match the requested [" + tag + "]"); + System.out.println("Tag [" + summary.tag + "] does not match the requested [" + tag + "] in " + summary.blog.toBase64()); continue; } } for (int j = 0; j < summary.entries.size(); j++) { EntrySummary entry = (EntrySummary)summary.entries.get(j); - ordered.put(new Long(0-entry.entry.getEntryId()), entry.entry); + String k = (Long.MAX_VALUE-entry.entry.getEntryId()) + "-" + entry.entry.getKeyHash().toBase64(); + ordered.put(k, entry.entry); + System.err.println("Including match: " + k); } } for (Iterator iter = ordered.values().iterator(); iter.hasNext(); ) { @@ -319,7 +330,7 @@ public class ArchiveIndex { _blogs.add(summary); } - private SimpleDateFormat _dateFmt = new SimpleDateFormat("yyyyMMdd"); + private SimpleDateFormat _dateFmt = new SimpleDateFormat("yyyyMMdd", Locale.UK); private long getIndexDate(String yyyymmdd) { synchronized (_dateFmt) { try { diff --git a/apps/syndie/java/src/net/i2p/syndie/data/BlogURI.java b/apps/syndie/java/src/net/i2p/syndie/data/BlogURI.java index 6efc154703a29873d63243f58d97df9ee08346ee..99d8b17078a9c63e9cb4483bd5b31c69418edabd 100644 --- a/apps/syndie/java/src/net/i2p/syndie/data/BlogURI.java +++ b/apps/syndie/java/src/net/i2p/syndie/data/BlogURI.java @@ -31,6 +31,19 @@ public class BlogURI { _entryId = -1; } } + } else if (uri.startsWith("entry://")) { + int off = "entry://".length(); + _blogHash = new Hash(Base64.decode(uri.substring(off, off+44))); // 44 chars == base64(32 bytes) + int entryStart = uri.indexOf('/', off+1); + if (entryStart < 0) { + _entryId = -1; + } else { + try { + _entryId = Long.parseLong(uri.substring(entryStart+1).trim()); + } catch (NumberFormatException nfe) { + _entryId = -1; + } + } } else { _blogHash = null; _entryId = -1; @@ -61,7 +74,10 @@ public class BlogURI { DataHelper.eq(_blogHash, ((BlogURI)obj)._blogHash); } public int hashCode() { - return (int)_entryId; + int rv = (int)_entryId; + if (_blogHash != null) + rv += _blogHash.hashCode(); + return rv; } public static void main(String args[]) { @@ -69,6 +85,8 @@ public class BlogURI { test("blog://Vq~AlW-r7OM763okVUFIDvVFzxOjpNNsAx0rFb2yaE8="); test("blog://Vq~AlW-r7OM763okVUFIDvVFzxOjpNNsAx0rFb2yaE8=/"); test("blog://Vq~AlW-r7OM763okVUFIDvVFzxOjpNNsAx0rFb2yaE8=/123456789"); + test("entry://Vq~AlW-r7OM763okVUFIDvVFzxOjpNNsAx0rFb2yaE8=/"); + test("entry://Vq~AlW-r7OM763okVUFIDvVFzxOjpNNsAx0rFb2yaE8=/123456789"); } private static void test(String uri) { BlogURI u = new BlogURI(uri); diff --git a/apps/syndie/java/src/net/i2p/syndie/data/LocalArchiveIndex.java b/apps/syndie/java/src/net/i2p/syndie/data/LocalArchiveIndex.java index 69df155437e80ae37d6bc9146f8b65975a75eee3..bc5bbeca4c7aecc0159f4163f0cb3ae67cb599e2 100644 --- a/apps/syndie/java/src/net/i2p/syndie/data/LocalArchiveIndex.java +++ b/apps/syndie/java/src/net/i2p/syndie/data/LocalArchiveIndex.java @@ -67,4 +67,36 @@ public class LocalArchiveIndex extends ArchiveIndex { if (!_newestEntries.contains(entry)) _newestEntries.add(entry); } + + public void addReply(BlogURI parent, BlogURI reply) { + Set replies = (Set)_replies.get(parent); + if (replies == null) { + replies = Collections.synchronizedSet(new TreeSet(BlogURIComparator.HIGHEST_ID_FIRST)); + _replies.put(parent, replies); + } + replies.add(reply); + System.err.println("Adding reply to " + parent + " from child " + reply + " (# replies: " + replies.size() + ")"); + } + + private static class BlogURIComparator implements Comparator { + public static final BlogURIComparator HIGHEST_ID_FIRST = new BlogURIComparator(true); + public static final BlogURIComparator HIGHEST_ID_LAST = new BlogURIComparator(false); + private boolean _highestFirst; + public BlogURIComparator(boolean highestFirst) { + _highestFirst = highestFirst; + } + + public int compare(Object lhs, Object rhs) { + if ( (lhs == null) || !(lhs instanceof BlogURI) ) return 1; + if ( (rhs == null) || !(rhs instanceof BlogURI) ) return -1; + BlogURI l = (BlogURI)lhs; + BlogURI r = (BlogURI)rhs; + if (l.getEntryId() > r.getEntryId()) + return (_highestFirst ? 1 : -1); + else if (l.getEntryId() < r.getEntryId()) + return (_highestFirst ? -1 : 1); + else + return DataHelper.compareTo(l.getKeyHash().getData(), r.getKeyHash().getData()); + } + } } diff --git a/apps/syndie/java/src/net/i2p/syndie/data/TransparentArchiveIndex.java b/apps/syndie/java/src/net/i2p/syndie/data/TransparentArchiveIndex.java new file mode 100644 index 0000000000000000000000000000000000000000..8660366a53794bc45db04c2d428fb8757d0b8e18 --- /dev/null +++ b/apps/syndie/java/src/net/i2p/syndie/data/TransparentArchiveIndex.java @@ -0,0 +1,81 @@ +package net.i2p.syndie.data; + +import java.io.*; +import java.text.*; +import java.util.*; +import net.i2p.I2PAppContext; +import net.i2p.data.*; +import net.i2p.syndie.Archive; +import net.i2p.syndie.BlogManager; + +/** + * Simple read-only summary of an archive, proxied to the BlogManager's instance + */ +public class TransparentArchiveIndex extends ArchiveIndex { + public TransparentArchiveIndex() { super(false); } + + private static ArchiveIndex index() { return BlogManager.instance().getArchive().getIndex(); } + + public String getVersion() { return index().getVersion(); } + public Properties getHeaders() { return index().getHeaders(); } + public int getAllBlogs() { return index().getAllBlogs(); } + public int getNewBlogs() { return index().getNewBlogs(); } + public int getAllEntries() { return index().getAllEntries(); } + public int getNewEntries() { return index().getNewEntries(); } + public long getTotalSize() { return index().getTotalSize(); } + public long getNewSize() { return index().getNewSize(); } + public long getGeneratedOn() { return index().getGeneratedOn(); } + + public String getNewSizeStr() { return index().getNewSizeStr(); } + public String getTotalSizeStr() { return index().getTotalSizeStr(); } + + /** how many blogs/tags are indexed */ + public int getIndexBlogs() { return index().getIndexBlogs(); } + /** get the blog used for the given blog/tag pair */ + public Hash getBlog(int index) { return index().getBlog(index); } + /** get the tag used for the given blog/tag pair */ + public String getBlogTag(int index) { return index().getBlogTag(index); } + /** get the highest entry ID for the given blog/tag pair */ + public long getBlogLastUpdated(int index) { return index().getBlogLastUpdated(index); } + /** get the entry count for the given blog/tag pair */ + public int getBlogEntryCount(int index) { return index().getBlogEntryCount(index); } + /** get the entry from the given blog/tag pair */ + public BlogURI getBlogEntry(int index, int entryIndex) { return index().getBlogEntry(index, entryIndex); } + /** get the raw entry size (including attachments) from the given blog/tag pair */ + public long getBlogEntrySizeKB(int index, int entryIndex) { return index().getBlogEntrySizeKB(index, entryIndex); } + public boolean getEntryIsKnown(BlogURI uri) { return index().getEntryIsKnown(uri); } + public long getBlogEntrySizeKB(BlogURI uri) { return index().getBlogEntrySizeKB(uri); } + public Set getBlogEntryTags(BlogURI uri) { return index().getBlogEntryTags(uri); } + /** how many 'new' blogs are listed */ + public int getNewestBlogCount() { return index().getNewestBlogCount(); } + public Hash getNewestBlog(int index) { return index().getNewestBlog(index); } + /** how many 'new' entries are listed */ + public int getNewestBlogEntryCount() { return index().getNewestBlogEntryCount(); } + public BlogURI getNewestBlogEntry(int index) { return index().getNewestBlogEntry(index); } + + /** list of locally known tags (String) under the given blog */ + public List getBlogTags(Hash blog) { return index().getBlogTags(blog); } + /** list of unique blogs locally known (set of Hash) */ + public Set getUniqueBlogs() { return index().getUniqueBlogs(); } + public void setLocation(String location) { return; } + public void setIsLocal(String val) { return; } + public void load(File location) throws IOException { return; } + /** load up the index from an archive.txt */ + public void load(InputStream index) throws IOException { return; } + + /** + * Dig through the index for BlogURIs matching the given criteria, ordering the results by + * their own entryIds. + * + * @param out where to store the matches + * @param blog if set, what blog key must the entries be under + * @param tag if set, what tag must the entry be in + * + */ + public void selectMatchesOrderByEntryId(List out, Hash blog, String tag) { + index().selectMatchesOrderByEntryId(out, blog, tag); + } + + /** export the index into an archive.txt */ + public String toString() { return index().toString(); } +} diff --git a/apps/syndie/java/src/net/i2p/syndie/sml/HTMLRenderer.java b/apps/syndie/java/src/net/i2p/syndie/sml/HTMLRenderer.java index 622b2a4e60bf81419af0b7f8a0c9a41201951f08..883e620130e8b02e5cf09370922af267139fbfb1 100644 --- a/apps/syndie/java/src/net/i2p/syndie/sml/HTMLRenderer.java +++ b/apps/syndie/java/src/net/i2p/syndie/sml/HTMLRenderer.java @@ -412,6 +412,20 @@ public class HTMLRenderer extends EventReceiverImpl { else if (addrs > 1) _postBodyBuffer.append(addrs).append(" addresses "); + if (_entry != null) { + List replies = _archive.getIndex().getReplies(_entry.getURI()); + if ( (replies != null) && (replies.size() > 0) ) { + if (replies.size() == 1) + _postBodyBuffer.append("1 reply "); + else + _postBodyBuffer.append(replies.size()).append(" replies "); + } + } + + String inReplyTo = (String)_headers.get(HEADER_IN_REPLY_TO); + if ( (inReplyTo != null) && (inReplyTo.trim().length() > 0) ) + _postBodyBuffer.append(" <a href=\"").append(getPageURL(sanitizeTagParam(inReplyTo))).append("\">(view parent)</a>\n"); + _postBodyBuffer.append("</td></tr>\n"); } else { _postBodyBuffer.append("<tr class=\"syndieEntryAttachmentsCell\">\n"); @@ -486,6 +500,35 @@ public class HTMLRenderer extends EventReceiverImpl { _postBodyBuffer.append("<br />\n"); } + + if (_entry != null) { + List replies = _archive.getIndex().getReplies(_entry.getURI()); + if ( (replies != null) && (replies.size() > 0) ) { + _postBodyBuffer.append("<b>Replies:</b> "); + for (int i = 0; i < replies.size(); i++) { + BlogURI reply = (BlogURI)replies.get(i); + _postBodyBuffer.append("<a href=\""); + _postBodyBuffer.append(getPageURL(reply.getKeyHash(), null, reply.getEntryId(), -1, -1, true, _user.getShowImages())); + _postBodyBuffer.append("\">"); + BlogInfo replyAuthor = _archive.getBlogInfo(reply); + if (replyAuthor != null) { + _postBodyBuffer.append(sanitizeString(replyAuthor.getProperty(BlogInfo.NAME))); + } else { + _postBodyBuffer.append(reply.getKeyHash().toBase64().substring(0,16)); + } + _postBodyBuffer.append(" on "); + _postBodyBuffer.append(getEntryDate(reply.getEntryId())); + _postBodyBuffer.append("</a> "); + } + _postBodyBuffer.append("<br />"); + } + } + + String inReplyTo = (String)_headers.get(HEADER_IN_REPLY_TO); + if ( (inReplyTo != null) && (inReplyTo.trim().length() > 0) ) { + _postBodyBuffer.append(" <a href=\"").append(getPageURL(sanitizeTagParam(inReplyTo))).append("\">(view parent)</a><br />\n"); + } + _postBodyBuffer.append("</td>\n</form>\n</tr>\n"); } _postBodyBuffer.append("</table>\n"); @@ -525,6 +568,9 @@ public class HTMLRenderer extends EventReceiverImpl { } private void renderMetaCell() { + String tags[] = (_entry != null ? _entry.getTags() : null); + if ( (tags != null) && (tags.length > 0) ) + _preBodyBuffer.append("<form action=\"index.jsp\">"); _preBodyBuffer.append("<td nowrap=\"true\" align=\"right\" valign=\"top\" class=\"syndieEntryMetaCell\">\n"); BlogInfo info = null; if (_entry != null) @@ -540,11 +586,16 @@ public class HTMLRenderer extends EventReceiverImpl { } else { _preBodyBuffer.append("[unknown blog]"); } - String tags[] = (_entry != null ? _entry.getTags() : null); if ( (tags != null) && (tags.length > 0) ) { _preBodyBuffer.append(" Tags: "); - _preBodyBuffer.append("<i>"); + _preBodyBuffer.append("<select name=\"selector\">"); for (int i = 0; tags != null && i < tags.length; i++) { + _preBodyBuffer.append("<option value=\"blogtag://"); + _preBodyBuffer.append(_entry.getURI().getKeyHash().toBase64()); + _preBodyBuffer.append('/').append(Base64.encode(tags[i])).append("\">"); + _preBodyBuffer.append(sanitizeString(tags[i])); + _preBodyBuffer.append("</option>\n"); + /* _preBodyBuffer.append("<a href=\""); _preBodyBuffer.append(getPageURL(_entry.getURI().getKeyHash(), tags[i], -1, -1, -1, (_user != null ? _user.getShowExpanded() : false), (_user != null ? _user.getShowImages() : false))); _preBodyBuffer.append("\">"); @@ -552,23 +603,32 @@ public class HTMLRenderer extends EventReceiverImpl { _preBodyBuffer.append("</a>"); if (i + 1 < tags.length) _preBodyBuffer.append(", "); + */ } - _preBodyBuffer.append("</i>"); + _preBodyBuffer.append("</select>"); + _preBodyBuffer.append("<input type=\"submit\" value=\"View\" />\n"); + //_preBodyBuffer.append("</i>"); } _preBodyBuffer.append(" "); + /* + String inReplyTo = (String)_headers.get(HEADER_IN_REPLY_TO); + if ( (inReplyTo != null) && (inReplyTo.trim().length() > 0) ) + _preBodyBuffer.append(" <a href=\"").append(getPageURL(sanitizeTagParam(inReplyTo))).append("\">In reply to</a>\n"); + */ + if (_entry != null) _preBodyBuffer.append(getEntryDate(_entry.getURI().getEntryId())); else _preBodyBuffer.append(getEntryDate(new Date().getTime())); - String inReplyTo = (String)_headers.get(HEADER_IN_REPLY_TO); - if ( (inReplyTo != null) && (inReplyTo.trim().length() > 0) ) - _preBodyBuffer.append(" <a href=\"").append(getPageURL(sanitizeTagParam(inReplyTo))).append("\">In reply to</a>\n"); if ( (_user != null) && (_user.getAuthenticated()) ) _preBodyBuffer.append(" <a href=\"").append(getPostURL(_user.getBlog(), true)).append("\">Reply</a>\n"); - _preBodyBuffer.append("\n</td></tr>\n"); + _preBodyBuffer.append("\n</td>"); + if ( (tags != null) && (tags.length > 0) ) + _preBodyBuffer.append("</form>"); + _preBodyBuffer.append("</tr>\n"); } - private final SimpleDateFormat _dateFormat = new SimpleDateFormat("yyyy/MM/dd"); + private final SimpleDateFormat _dateFormat = new SimpleDateFormat("yyyy/MM/dd", Locale.UK); private final String getEntryDate(long when) { synchronized (_dateFormat) { try { @@ -656,13 +716,19 @@ public class HTMLRenderer extends EventReceiverImpl { } } - public String getPageURL(String selector) { - return getPageURL(_user, selector); - } - - public static String getPageURL(User user, String selector) { - ArchiveViewerBean.Selector sel = new ArchiveViewerBean.Selector(selector); - return getPageURL(sel.blog, sel.tag, sel.entry, sel.group, -1, -1, user.getShowExpanded(), user.getShowImages()); + public String getPageURL(String selector) { return getPageURL(_user, selector); } + public static String getPageURL(User user, String selector) { return getPageURL(user, selector, -1, -1); } + public static String getPageURL(User user, String selector, int numPerPage, int pageNum) { + StringBuffer buf = new StringBuffer(128); + buf.append("index.jsp?"); + buf.append("selector=").append(sanitizeTagParam(selector)).append("&"); + if ( (pageNum >= 0) && (numPerPage > 0) ) { + buf.append(ArchiveViewerBean.PARAM_PAGE_NUMBER).append('=').append(pageNum).append('&'); + buf.append(ArchiveViewerBean.PARAM_NUM_PER_PAGE).append('=').append(numPerPage).append('&'); + } + buf.append(ArchiveViewerBean.PARAM_EXPAND_ENTRIES).append('=').append(user.getShowExpanded()).append('&'); + buf.append(ArchiveViewerBean.PARAM_SHOW_IMAGES).append('=').append(user.getShowImages()).append('&'); + return buf.toString(); } public static String getPageURL(Hash blog, String tag, long entryId, int numPerPage, int pageNum, boolean expandEntries, boolean showImages) { diff --git a/apps/syndie/java/src/net/i2p/syndie/web/ArchiveServlet.java b/apps/syndie/java/src/net/i2p/syndie/web/ArchiveServlet.java new file mode 100644 index 0000000000000000000000000000000000000000..28b5b9c7baec0ab343c50165c07afd17b69c3a1a --- /dev/null +++ b/apps/syndie/java/src/net/i2p/syndie/web/ArchiveServlet.java @@ -0,0 +1,178 @@ +package net.i2p.syndie.web; + +import java.io.*; +import java.util.*; + +import javax.servlet.http.HttpServlet; +import javax.servlet.http.HttpServletRequest; +import javax.servlet.http.HttpServletResponse; +import javax.servlet.ServletException; + +import net.i2p.data.*; +import net.i2p.syndie.*; +import net.i2p.syndie.data.*; + +/** + * + */ +public class ArchiveServlet extends HttpServlet { + + public void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException { + String path = req.getPathInfo(); + if ( (path == null) || (path.trim().length() <= 1) ) { + renderRootIndex(resp); + return; + } else if (path.endsWith(Archive.INDEX_FILE)) { + renderSummary(resp); + } else { + String blog = getBlog(path); + if (path.endsWith(Archive.METADATA_FILE)) { + renderMetadata(blog, resp); + } else if (path.endsWith(".snd")) { + renderEntry(blog, getEntry(path), resp); + } else { + renderBlogIndex(blog, resp); + } + } + } + + private String getBlog(String path) { + System.err.println("Blog: [" + path + "]"); + int start = 0; + int end = -1; + int len = path.length(); + for (int i = 0; i < len; i++) { + if (path.charAt(i) != '/') { + start = i; + break; + } + } + for (int j = start + 1; j < len; j++) { + if (path.charAt(j) == '/') { + end = j; + break; + } + } + if (end < 0) end = len; + String rv = path.substring(start, end); + System.err.println("Blog: [" + path + "] rv: [" + rv + "]"); + return rv; + } + + private long getEntry(String path) { + int start = path.lastIndexOf('/'); + if (start < 0) return -1; + if (!(path.endsWith(".snd"))) return -1; + String rv = path.substring(start+1, path.length()-".snd".length()); + System.err.println("Entry: [" + path + "] rv: [" + rv + "]"); + try { + return Long.parseLong(rv); + } catch (NumberFormatException nfe) { + return -1; + } + } + + private void renderRootIndex(HttpServletResponse resp) throws ServletException, IOException { + resp.setContentType("text/html"); + OutputStream out = resp.getOutputStream(); + out.write("<a href=\"archive.txt\">archive.txt</a><br />\n".getBytes()); + ArchiveIndex index = BlogManager.instance().getArchive().getIndex(); + Set blogs = index.getUniqueBlogs(); + for (Iterator iter = blogs.iterator(); iter.hasNext(); ) { + Hash blog = (Hash)iter.next(); + String s = blog.toBase64(); + out.write(("<a href=\"" + s + "/\">" + s + "</a><br />\n").getBytes()); + } + out.close(); + } + + private void renderSummary(HttpServletResponse resp) throws ServletException, IOException { + resp.setContentType("text/plain"); + OutputStream out = resp.getOutputStream(); + ArchiveIndex index = BlogManager.instance().getArchive().getIndex(); + out.write(index.toString().getBytes()); + out.close(); + } + + private void renderMetadata(String blog, HttpServletResponse resp) throws ServletException, IOException { + byte b[] = Base64.decode(blog); + if ( (b == null) || (b.length != Hash.HASH_LENGTH) ) { + resp.sendError(404, "Invalid blog requested"); + return; + } + Hash h = new Hash(b); + BlogInfo info = BlogManager.instance().getArchive().getBlogInfo(h); + if (info == null) { + resp.sendError(404, "Blog does not exist"); + return; + } + resp.setContentType("application/x-syndie-meta"); + OutputStream out = resp.getOutputStream(); + info.write(out); + out.close(); + } + + private void renderBlogIndex(String blog, HttpServletResponse resp) throws ServletException, IOException { + byte b[] = Base64.decode(blog); + if ( (b == null) || (b.length != Hash.HASH_LENGTH) ) { + resp.sendError(404, "Invalid blog requested"); + return; + } + Hash h = new Hash(b); + + BlogInfo info = BlogManager.instance().getArchive().getBlogInfo(h); + if (info == null) { + resp.sendError(404, "Blog does not exist"); + return; + } + resp.setContentType("text/html"); + OutputStream out = resp.getOutputStream(); + out.write("<a href=\"..\">..</a><br />\n".getBytes()); + out.write(("<a href=\"" + Archive.METADATA_FILE + "\">" + Archive.METADATA_FILE + "</a><br />\n").getBytes()); + List entries = new ArrayList(64); + BlogManager.instance().getArchive().getIndex().selectMatchesOrderByEntryId(entries, h, null); + for (int i = 0; i < entries.size(); i++) { + BlogURI entry = (BlogURI)entries.get(i); + out.write(("<a href=\"" + entry.getEntryId() + ".snd\">" + entry.getEntryId() + ".snd</a><br />\n").getBytes()); + } + out.close(); + } + + private void renderEntry(String blog, long entryId, HttpServletResponse resp) throws ServletException, IOException { + byte b[] = Base64.decode(blog); + if ( (b == null) || (b.length != Hash.HASH_LENGTH) ) { + resp.sendError(404, "Invalid blog requested"); + return; + } + Hash h = new Hash(b); + BlogInfo info = BlogManager.instance().getArchive().getBlogInfo(h); + if (info == null) { + resp.sendError(404, "Blog does not exist"); + return; + } + File root = BlogManager.instance().getArchive().getArchiveDir(); + File blogDir = new File(root, blog); + if (!blogDir.exists()) { + resp.sendError(404, "Blog does not exist"); + return; + } + File entry = new File(blogDir, entryId + ".snd"); + if (!entry.exists()) { + resp.sendError(404, "Entry does not exist"); + return; + } + resp.setContentType("application/x-syndie-post"); + dump(entry, resp); + } + + private void dump(File source, HttpServletResponse resp) throws ServletException, IOException { + FileInputStream in = new FileInputStream(source); + OutputStream out = resp.getOutputStream(); + byte buf[] = new byte[1024]; + int read = 0; + while ( (read = in.read(buf)) != -1) + out.write(buf, 0, read); + out.close(); + in.close(); + } +} diff --git a/apps/syndie/java/src/net/i2p/syndie/web/ArchiveViewerBean.java b/apps/syndie/java/src/net/i2p/syndie/web/ArchiveViewerBean.java index b9d6a62b63f055cc0f32e5d29049a2a27732ed42..85c653332c317d75dd8ef2922335e6708709916f 100644 --- a/apps/syndie/java/src/net/i2p/syndie/web/ArchiveViewerBean.java +++ b/apps/syndie/java/src/net/i2p/syndie/web/ArchiveViewerBean.java @@ -25,8 +25,8 @@ public class ArchiveViewerBean { return getEntryTitleDate(name, entryId); } - private static final SimpleDateFormat _dateFormat = new SimpleDateFormat("yyyy/MM/dd"); - private static final String getEntryTitleDate(String blogName, long when) { + private static final SimpleDateFormat _dateFormat = new SimpleDateFormat("yyyy/MM/dd", Locale.UK); + public static final String getEntryTitleDate(String blogName, long when) { synchronized (_dateFormat) { try { String str = _dateFormat.format(new Date(when)); @@ -115,6 +115,22 @@ public class ArchiveViewerBean { Archive archive = BlogManager.instance().getArchive(); ArchiveIndex index = archive.getIndex(); + + for (int i = 0; i < index.getNewestBlogCount(); i++) { + Hash cur = index.getNewestBlog(i); + String blog = Base64.encode(cur.getData()); + out.write("<option value=\"blog://" + blog + "\">"); + out.write("New blog: "); + BlogInfo info = archive.getBlogInfo(cur); + String name = info.getProperty(BlogInfo.NAME); + if (name != null) + name = HTMLRenderer.sanitizeString(name); + else + name = Base64.encode(cur.getData()); + out.write(name); + out.write("</option>\n"); + } + List allTags = new ArrayList(); // perhaps sort this by name (even though it isnt unique...) Set blogs = index.getUniqueBlogs(); @@ -178,7 +194,7 @@ public class ArchiveViewerBean { return user.getDefaultSelector(); } - public static void renderBlogs(User user, Map parameters, Writer out) throws IOException { + public static void renderBlogs(User user, Map parameters, Writer out, String afterPagination) throws IOException { String blogStr = getString(parameters, PARAM_BLOG); Hash blog = null; if (blogStr != null) blog = new Hash(Base64.decode(blogStr)); @@ -211,7 +227,7 @@ public class ArchiveViewerBean { boolean showImages = getBool(parameters, PARAM_SHOW_IMAGES, (user != null ? user.getShowImages() : false)); boolean regenerateIndex = getBool(parameters, PARAM_REGENERATE_INDEX, false); try { - renderBlogs(user, blog, tag, entryId, group, numPerPage, pageNum, expandEntries, showImages, regenerateIndex, out); + renderBlogs(user, blog, tag, entryId, group, numPerPage, pageNum, expandEntries, showImages, regenerateIndex, sel, out, afterPagination); } catch (IOException ioe) { ioe.printStackTrace(); throw ioe; @@ -263,8 +279,8 @@ public class ArchiveViewerBean { } } - public static void renderBlogs(User user, Hash blog, String tag, long entryId, String group, int numPerPage, int pageNum, - boolean expandEntries, boolean showImages, boolean regenerateIndex, Writer out) throws IOException { + private static void renderBlogs(User user, Hash blog, String tag, long entryId, String group, int numPerPage, int pageNum, + boolean expandEntries, boolean showImages, boolean regenerateIndex, String selector, Writer out, String afterPagination) throws IOException { Archive archive = BlogManager.instance().getArchive(); if (regenerateIndex) archive.regenerateIndex(); @@ -293,7 +309,11 @@ public class ArchiveViewerBean { pages++; out.write("<i>"); if (pageNum > 0) { - String prevURL = HTMLRenderer.getPageURL(blog, tag, entryId, group, numPerPage, pageNum-1, expandEntries, showImages); + String prevURL = null; + if ( (selector == null) || (selector.trim().length() <= 0) ) + prevURL = HTMLRenderer.getPageURL(blog, tag, entryId, group, numPerPage, pageNum-1, expandEntries, showImages); + else + prevURL = HTMLRenderer.getPageURL(user, selector, numPerPage, pageNum-1); System.out.println("prevURL: " + prevURL); out.write(" <a href=\"" + prevURL + "\"><<</a>"); } else { @@ -301,7 +321,11 @@ public class ArchiveViewerBean { } out.write("Page " + (pageNum+1) + " of " + pages); if (pageNum + 1 < pages) { - String nextURL = HTMLRenderer.getPageURL(blog, tag, entryId, group, numPerPage, pageNum+1, expandEntries, showImages); + String nextURL = null; + if ( (selector == null) || (selector.trim().length() <= 0) ) + nextURL = HTMLRenderer.getPageURL(blog, tag, entryId, group, numPerPage, pageNum+1, expandEntries, showImages); + else + nextURL = HTMLRenderer.getPageURL(user, selector, numPerPage, pageNum+1); System.out.println("nextURL: " + nextURL); out.write(" <a href=\"" + nextURL + "\">>></a>"); } else { @@ -311,6 +335,7 @@ public class ArchiveViewerBean { } } + /* out.write(" <i>"); if (showImages) @@ -328,6 +353,10 @@ public class ArchiveViewerBean { "\">Expand details</a>"); out.write("</i>"); + */ + + if (afterPagination != null) + out.write(afterPagination); if (entries.size() <= 0) end = -1; System.out.println("Entries.size: " + entries.size() + " start=" + start + " end=" + end); diff --git a/apps/syndie/java/src/net/i2p/syndie/web/RemoteArchiveBean.java b/apps/syndie/java/src/net/i2p/syndie/web/RemoteArchiveBean.java index d5959e6867a1f4c5221a65eeae84c9329c4ccdb5..d2f0a64c94c1299e8ad6885db4300bdb8e1eb1bb 100644 --- a/apps/syndie/java/src/net/i2p/syndie/web/RemoteArchiveBean.java +++ b/apps/syndie/java/src/net/i2p/syndie/web/RemoteArchiveBean.java @@ -7,6 +7,7 @@ import net.i2p.I2PAppContext; import net.i2p.data.*; import net.i2p.util.EepGet; import net.i2p.util.EepGetScheduler; +import net.i2p.util.EepPost; import net.i2p.syndie.data.*; import net.i2p.syndie.sml.*; import net.i2p.syndie.*; @@ -39,6 +40,8 @@ public class RemoteArchiveBean { public String getRemoteSchema() { return _remoteSchema; } public String getRemoteLocation() { return _remoteLocation; } public ArchiveIndex getRemoteIndex() { return _remoteIndex; } + public String getProxyHost() { return _proxyHost; } + public int getProxyPort() { return _proxyPort; } public boolean getFetchIndexInProgress() { return _fetchIndexInProgress; } public String getStatus() { StringBuffer buf = new StringBuffer(); @@ -46,7 +49,7 @@ public class RemoteArchiveBean { buf.append(_statusMessages.remove(0)).append("\n"); return buf.toString(); } - + public void fetchMetadata(User user, Map parameters) { String meta = ArchiveViewerBean.getString(parameters, "blog"); if (meta == null) return; @@ -146,33 +149,45 @@ public class RemoteArchiveBean { scheduler.fetch(); } - public void fetchIndex(User user, String schema, String location) { + public void fetchIndex(User user, String schema, String location, String proxyHost, String proxyPort) { _fetchIndexInProgress = true; _remoteIndex = null; _remoteLocation = location; _remoteSchema = schema; _proxyHost = null; _proxyPort = -1; - if ("eep".equals(_remoteSchema)) { - _proxyHost = user.getEepProxyHost(); - _proxyPort = user.getEepProxyPort(); - } else if ("web".equals(_remoteSchema)) { - _proxyHost = user.getWebProxyHost(); - _proxyPort = user.getWebProxyPort(); - } else if ("tor".equals(_remoteSchema)) { - _proxyHost = user.getTorProxyHost(); - _proxyPort = user.getTorProxyPort(); + + if ( (schema == null) || (schema.trim().length() <= 0) || + (location == null) || (location.trim().length() <= 0) ) { + _statusMessages.add("Location must be specified"); + _fetchIndexInProgress = false; + return; + } + + if ("web".equals(schema)) { + if ( (proxyHost != null) && (proxyHost.trim().length() > 0) && + (proxyPort != null) && (proxyPort.trim().length() > 0) ) { + _proxyHost = proxyHost; + try { + _proxyPort = Integer.parseInt(proxyPort); + } catch (NumberFormatException nfe) { + _statusMessages.add("Proxy port " + HTMLRenderer.sanitizeString(proxyPort) + " is invalid"); + _fetchIndexInProgress = false; + return; + } + } } else { _statusMessages.add(new String("Remote schema " + HTMLRenderer.sanitizeString(schema) + " currently not supported")); _fetchIndexInProgress = false; return; } - - _statusMessages.add("Fetching index from " + HTMLRenderer.sanitizeString(_remoteLocation)); + + _statusMessages.add("Fetching index from " + HTMLRenderer.sanitizeString(_remoteLocation) + + (_proxyHost != null ? " via " + HTMLRenderer.sanitizeString(_proxyHost) + ":" + _proxyPort : "")); File archiveFile = new File(BlogManager.instance().getTempDir(), user.getBlog().toBase64() + "_remoteArchive.txt"); archiveFile.delete(); - EepGet eep = new EepGet(I2PAppContext.getGlobalContext(), ((_proxyHost != null) && (_proxyPort > 0)), - _proxyHost, _proxyPort, 0, archiveFile.getAbsolutePath(), location); + EepGet eep = new EepGet(I2PAppContext.getGlobalContext(), ((_proxyHost != null) && (_proxyPort > 0)), + _proxyHost, _proxyPort, 0, archiveFile.getAbsolutePath(), location); eep.addStatusListener(new IndexFetcherStatusListener(archiveFile)); eep.fetch(); } @@ -185,7 +200,7 @@ public class RemoteArchiveBean { public void attemptFailed(String url, long bytesTransferred, long bytesRemaining, int currentAttempt, int numRetries, Exception cause) { _statusMessages.add("Attempt " + currentAttempt + " failed after " + bytesTransferred + (cause != null ? cause.getMessage() : "")); } - + public void bytesTransferred(long alreadyTransferred, int currentWrite, long bytesTransferred, long bytesRemaining, String url) {} public void transferComplete(long alreadyTransferred, long bytesTransferred, long bytesRemaining, String url, String outputFile) { _statusMessages.add("Fetch of " + HTMLRenderer.sanitizeString(url) + " successful"); @@ -210,7 +225,7 @@ public class RemoteArchiveBean { public void attemptFailed(String url, long bytesTransferred, long bytesRemaining, int currentAttempt, int numRetries, Exception cause) { _statusMessages.add("Attempt " + currentAttempt + " failed after " + bytesTransferred + (cause != null ? cause.getMessage() : "")); } - + public void bytesTransferred(long alreadyTransferred, int currentWrite, long bytesTransferred, long bytesRemaining, String url) {} public void transferComplete(long alreadyTransferred, long bytesTransferred, long bytesRemaining, String url, String outputFile) { _statusMessages.add("Fetch of " + HTMLRenderer.sanitizeString(url) + " successful"); @@ -244,7 +259,7 @@ public class RemoteArchiveBean { public void attemptFailed(String url, long bytesTransferred, long bytesRemaining, int currentAttempt, int numRetries, Exception cause) { _statusMessages.add("Attempt " + currentAttempt + " failed after " + bytesTransferred + (cause != null ? cause.getMessage() : "")); } - + public void bytesTransferred(long alreadyTransferred, int currentWrite, long bytesTransferred, long bytesRemaining, String url) {} public void transferComplete(long alreadyTransferred, long bytesTransferred, long bytesRemaining, String url, String outputFile) { _statusMessages.add("Fetch of " + HTMLRenderer.sanitizeString(url) + " successful"); @@ -285,6 +300,47 @@ public class RemoteArchiveBean { } } + + public void postSelectedEntries(User user, Map parameters) { + String entries[] = ArchiveViewerBean.getStrings(parameters, "localentry"); + if ( (entries == null) || (entries.length <= 0) ) return; + List uris = new ArrayList(entries.length); + for (int i = 0; i < entries.length; i++) + uris.add(new BlogURI(entries[i])); + + post(uris, user); + } + + private void post(List blogURIs, User user) { + List files = new ArrayList(blogURIs.size()+1); + Set meta = new HashSet(4); + Map uploads = new HashMap(files.size()); + String importURL = getImportURL(); + _statusMessages.add("Uploading through " + HTMLRenderer.sanitizeString(importURL)); + for (int i = 0; i < blogURIs.size(); i++) { + BlogURI uri = (BlogURI)blogURIs.get(i); + File blogDir = new File(BlogManager.instance().getArchive().getArchiveDir(), uri.getKeyHash().toBase64()); + BlogInfo info = BlogManager.instance().getArchive().getBlogInfo(uri); + if (!meta.contains(uri.getKeyHash())) { + uploads.put("blogmeta" + meta.size(), new File(blogDir, Archive.METADATA_FILE)); + meta.add(uri.getKeyHash()); + _statusMessages.add("Scheduling upload of the blog metadata for " + HTMLRenderer.sanitizeString(info.getProperty(BlogInfo.NAME))); + } + uploads.put("blogpost" + i, new File(blogDir, uri.getEntryId() + ".snd")); + _statusMessages.add("Scheduling upload of " + HTMLRenderer.sanitizeString(info.getProperty(BlogInfo.NAME)) + + ": " + getEntryDate(uri.getEntryId())); + } + EepPost post = new EepPost(); + post.postFiles(importURL, _proxyHost, _proxyPort, uploads, new Runnable() { public void run() { _statusMessages.add("Upload complete"); } }); + } + + private String getImportURL() { + String loc = _remoteLocation.trim(); + int archiveRoot = loc.lastIndexOf('/'); + int syndieRoot = loc.lastIndexOf('/', archiveRoot-1); + return loc.substring(0, syndieRoot + 1) + "import.jsp"; + } + public void renderDeltaForm(User user, ArchiveIndex localIndex, Writer out) throws IOException { Archive archive = BlogManager.instance().getArchive(); StringBuffer buf = new StringBuffer(512); @@ -305,7 +361,9 @@ public class RemoteArchiveBean { } int newEntries = 0; + int localNew = 0; out.write("<table border=\"1\" width=\"100%\">\n"); + List entries = new ArrayList(); for (Iterator iter = remoteBlogs.iterator(); iter.hasNext(); ) { Hash blog = (Hash)iter.next(); buf = new StringBuffer(1024); @@ -314,13 +372,13 @@ public class RemoteArchiveBean { BlogInfo info = archive.getBlogInfo(blog); if (info != null) { buf.append("<a href=\"" + HTMLRenderer.getPageURL(blog, null, -1, -1, -1, user.getShowExpanded(), user.getShowImages()) + "\"><b>" + HTMLRenderer.sanitizeString(info.getProperty(BlogInfo.NAME)) + "</b></a>: " + - HTMLRenderer.sanitizeString(info.getProperty(BlogInfo.DESCRIPTION)) + "\n"); + HTMLRenderer.sanitizeString(info.getProperty(BlogInfo.DESCRIPTION)) + "\n"); } else { buf.append("<b>" + blog.toBase64() + "</b>\n"); } buf.append("</td></tr>\n"); buf.append("<tr><td> </td><td nowrap=\"true\"><b>Posted on</b></td><td nowrap=\"true\"><b>#</b></td><td nowrap=\"true\"><b>Size</b></td><td width=\"90%\" nowrap=\"true\"><b>Tags</b></td></tr>\n"); - List entries = new ArrayList(); + entries.clear(); _remoteIndex.selectMatchesOrderByEntryId(entries, blog, null); for (int i = 0; i < entries.size(); i++) { BlogURI uri = (BlogURI)entries.get(i); @@ -330,8 +388,8 @@ public class RemoteArchiveBean { newEntries++; shownEntries++; } else { - String page = HTMLRenderer.getPageURL(blog, null, uri.getEntryId(), -1, -1, - user.getShowExpanded(), user.getShowImages()); + String page = HTMLRenderer.getPageURL(blog, null, uri.getEntryId(), -1, -1, + user.getShowExpanded(), user.getShowImages()); buf.append("<td><a href=\"" + page + "\">(local)</a></td>\n"); } buf.append("<td>" + getDate(uri.getEntryId()) + "</td>\n"); @@ -345,9 +403,70 @@ public class RemoteArchiveBean { buf.append("</td>\n"); buf.append("</tr>\n"); } + + // now for posts in known blogs that we have and they don't + entries.clear(); + localIndex.selectMatchesOrderByEntryId(entries, blog, null); + buf.append("<tr><td colspan=\"5\">Entries we have, but the remote Syndie doesn't:</td></tr>\n"); + for (int i = 0; i < entries.size(); i++) { + BlogURI uri = (BlogURI)entries.get(i); + if (!_remoteIndex.getEntryIsKnown(uri)) { + buf.append("<tr>\n"); + buf.append("<td><input type=\"checkbox\" name=\"localentry\" value=\"" + uri.toString() + "\" /></td>\n"); + shownEntries++; + newEntries++; + localNew++; + buf.append("<td>" + getDate(uri.getEntryId()) + "</td>\n"); + buf.append("<td>" + getId(uri.getEntryId()) + "</td>\n"); + buf.append("<td>" + localIndex.getBlogEntrySizeKB(uri) + "KB</td>\n"); + buf.append("<td>"); + for (Iterator titer = new TreeSet(localIndex.getBlogEntryTags(uri)).iterator(); titer.hasNext(); ) { + String tag = (String)titer.next(); + buf.append("<a href=\"" + HTMLRenderer.getPageURL(blog, tag, -1, -1, -1, user.getShowExpanded(), user.getShowImages()) + "\">" + tag + "</a> \n"); + } + buf.append("</td>\n"); + buf.append("</tr>\n"); + } + } + if (shownEntries > 0) // skip blogs we have already syndicated out.write(buf.toString()); } + + // now for posts in blogs we have and they don't + int newBefore = localNew; + buf.setLength(0); + buf.append("<tr><td colspan=\"5\">Blogs the remote Syndie doesn't have</td></tr>\n"); + for (Iterator iter = localBlogs.iterator(); iter.hasNext(); ) { + Hash blog = (Hash)iter.next(); + if (remoteBlogs.contains(blog)) { + System.err.println("Remote index has " + blog.toBase64()); + continue; + } + + entries.clear(); + localIndex.selectMatchesOrderByEntryId(entries, blog, null); + + for (int i = 0; i < entries.size(); i++) { + BlogURI uri = (BlogURI)entries.get(i); + buf.append("<tr>\n"); + buf.append("<td><input type=\"checkbox\" name=\"localentry\" value=\"" + uri.toString() + "\" /></td>\n"); + buf.append("<td>" + getDate(uri.getEntryId()) + "</td>\n"); + buf.append("<td>" + getId(uri.getEntryId()) + "</td>\n"); + buf.append("<td>" + localIndex.getBlogEntrySizeKB(uri) + "KB</td>\n"); + buf.append("<td>"); + for (Iterator titer = new TreeSet(localIndex.getBlogEntryTags(uri)).iterator(); titer.hasNext(); ) { + String tag = (String)titer.next(); + buf.append("<a href=\"" + HTMLRenderer.getPageURL(blog, tag, -1, -1, -1, user.getShowExpanded(), user.getShowImages()) + "\">" + tag + "</a> \n"); + } + buf.append("</td>\n"); + buf.append("</tr>\n"); + localNew++; + } + } + if (localNew > newBefore) + out.write(buf.toString()); + out.write("</table>\n"); if (newEntries > 0) { out.write("<input type=\"submit\" name=\"action\" value=\"Fetch selected entries\" /> \n"); @@ -355,14 +474,30 @@ public class RemoteArchiveBean { } else { out.write(HTMLRenderer.sanitizeString(_remoteLocation) + " has no new posts to offer us\n"); } + if (localNew > 0) { + out.write("<input type=\"submit\" name=\"action\" value=\"Post selected entries\" /> \n"); + } } - private final SimpleDateFormat _dateFormat = new SimpleDateFormat("yyyy/MM/dd"); + private final SimpleDateFormat _dateFormat = new SimpleDateFormat("yyyy/MM/dd", Locale.UK); private String getDate(long when) { synchronized (_dateFormat) { return _dateFormat.format(new Date(when)); } } - + private final String getEntryDate(long when) { + synchronized (_dateFormat) { + try { + String str = _dateFormat.format(new Date(when)); + long dayBegin = _dateFormat.parse(str).getTime(); + return str + "." + (when - dayBegin); + } catch (ParseException pe) { + pe.printStackTrace(); + // wtf + return "unknown"; + } + } + } + private long getId(long id) { synchronized (_dateFormat) { try { diff --git a/apps/syndie/jsp/_bodyindex.jsp b/apps/syndie/jsp/_bodyindex.jsp index 97fe883c84b9008782a48e648f4fcc953959fa3e..3ab14f9888d91b775a358ba4cc64ca60c348cfbf 100644 --- a/apps/syndie/jsp/_bodyindex.jsp +++ b/apps/syndie/jsp/_bodyindex.jsp @@ -1,9 +1,7 @@ <%@page import="net.i2p.syndie.web.ArchiveViewerBean, net.i2p.syndie.*" %> -<jsp:useBean scope="session" class="net.i2p.syndie.User" id="user" /> -<form action="index.jsp"> +<jsp:useBean scope="session" class="net.i2p.syndie.User" id="user" /><table border="0" width="100%"> +<tr><form action="index.jsp"><td nowrap="true"> <b>Blogs:</b> <%ArchiveViewerBean.renderBlogSelector(user, request.getParameterMap(), out);%> <input type="submit" value="Refresh" /> -<input type="submit" name="action" value="<%=ArchiveViewerBean.SEL_ACTION_SET_AS_DEFAULT%>" /></form> -<hr /> - -<%ArchiveViewerBean.renderBlogs(user, request.getParameterMap(), out); out.flush(); %> \ No newline at end of file +<input type="submit" name="action" value="<%=ArchiveViewerBean.SEL_ACTION_SET_AS_DEFAULT%>" /> +<%ArchiveViewerBean.renderBlogs(user, request.getParameterMap(), out, "</td></form></tr><tr><td align=\"left\" valign=\"top\">");%></td></tr></table> \ No newline at end of file diff --git a/apps/syndie/jsp/_leftnav.jsp b/apps/syndie/jsp/_leftnav.jsp index a453f375396db2973c85a6349249654188670c12..2b342a5f539785917ac1a7524d0333600d18c8a4 100644 --- a/apps/syndie/jsp/_leftnav.jsp +++ b/apps/syndie/jsp/_leftnav.jsp @@ -1,28 +1,3 @@ <%@page import="net.i2p.syndie.web.ArchiveViewerBean, net.i2p.syndie.*, net.i2p.data.Base64" %> <jsp:useBean scope="session" class="net.i2p.syndie.User" id="user" /> -<jsp:useBean scope="session" class="net.i2p.syndie.data.ArchiveIndex" id="archive" /> -<%if (request.getRequestURI().indexOf("register.jsp") == -1) {%> -<jsp:include page="_leftnavprotected.jsp" /> -<hr /> -<% } %> -<u>Local archive:</u><br /> -<b>Posts:</b> <jsp:getProperty name="archive" property="newEntries" />/<jsp:getProperty name="archive" property="allEntries" /><br /> -<b>Blogs:</b> <jsp:getProperty name="archive" property="newBlogs" />/<jsp:getProperty name="archive" property="allBlogs" /><br /> -<b>Size:</b> <jsp:getProperty name="archive" property="newSizeStr" />/<jsp:getProperty name="archive" property="totalSizeStr" /><br /> -<i>(new/total)</i> -<hr /> -<u>Latest blogs:</u><br /> -<%!int i = 0; %> -<%for (i = 0; i < archive.getNewestBlogCount(); i++) { - String keyHash = Base64.encode(archive.getNewestBlog(i).getData()); -%><a href="viewmetadata.jsp?<%=ArchiveViewerBean.PARAM_BLOG%>=<%=keyHash%>"> -<%=ArchiveViewerBean.getBlogName(keyHash)%></a><br /> -<% } %> -<hr /> -<u>Latest posts:</u><br /> -<%for (i = 0; i < archive.getNewestBlogEntryCount(); i++) { - String keyHash = Base64.encode(archive.getNewestBlogEntry(i).getKeyHash().getData()); - long entryId = archive.getNewestBlogEntry(i).getEntryId(); -%><a href="index.jsp?<%=ArchiveViewerBean.PARAM_BLOG%>=<%=keyHash%>&<%=ArchiveViewerBean.PARAM_ENTRY%>=<%=entryId%>&<%=ArchiveViewerBean.PARAM_EXPAND_ENTRIES%>=true"> -<%=ArchiveViewerBean.getEntryTitle(keyHash, entryId)%></a><br /> -<% } %> +<jsp:useBean scope="session" class="net.i2p.syndie.data.TransparentArchiveIndex" id="archive" /> \ No newline at end of file diff --git a/apps/syndie/jsp/_leftnavprotected.jsp b/apps/syndie/jsp/_leftnavprotected.jsp deleted file mode 100644 index e85fbaff4e632b96bbebdc1b953c25f3db127991..0000000000000000000000000000000000000000 --- a/apps/syndie/jsp/_leftnavprotected.jsp +++ /dev/null @@ -1,41 +0,0 @@ -<%@page import="net.i2p.syndie.*, net.i2p.syndie.sml.*" %> -<jsp:useBean scope="session" class="net.i2p.syndie.User" id="user" /> -<jsp:useBean scope="session" class="net.i2p.syndie.data.ArchiveIndex" id="archive" /> -<% -if ("true".equals(request.getParameter("logout"))) { - user.invalidate(); -} -String login = request.getParameter("login"); -String pass = request.getParameter("password"); -String loginSubmit = request.getParameter("Login"); -if ( (login != null) && (pass != null) && (loginSubmit != null) && (loginSubmit.equals("Login")) ) { - String loginResult = BlogManager.instance().login(user, login, pass); - out.write("<b>" + loginResult + "</b>"); -} - -if (user.getAuthenticated()) {%> -<b><u><jsp:getProperty property="username" name="user" />:</u></b><br /> -<a href="<%=HTMLRenderer.getPageURL(user.getBlog(), null, -1, -1, -1, user.getShowExpanded(), user.getShowImages())%>">My blog</a><br /> -<a href="<%=HTMLRenderer.getPageURL(user.getBlog(), null, user.getMostRecentEntry(), -1, -1, true, user.getShowImages())%>">Last post</a><br /> -<a href="<%=HTMLRenderer.getPostURL(user.getBlog())%>">Post</a><br /> -<a href="<%=HTMLRenderer.getMetadataURL(user.getBlog())%>">Metadata</a><br /> -<a href="index.jsp?logout=true">Logout</a><br /> -<!-- -<hr /> -<u>Remote Archives:</u><br /> -<a href="viewarchive.jsp?url=eep://politics.i2p/archive/">politics.i2p</a><br /> -<a href="viewarchive.jsp?url=freenet://SSK@.../TFE/archive/">TFE</a><br /> ---> -<%} else {%> -<form action="<%=request.getRequestURI() + "?" + (request.getQueryString() != null ? request.getQueryString() : "")%>"> -<b>Login:</b> <input type="text" name="login" size="8" /><br /> -<b>Pass:</b> <input type="password" name="password" size="8" /><br /><% -java.util.Enumeration params = request.getParameterNames(); -while (params.hasMoreElements()) { - String p = (String)params.nextElement(); - String val = request.getParameter(p); - %><input type="hidden" name="<%=p%>" value="<%=val%>" /><% -}%> -<input type="submit" name="Login" value="Login" /><br /></form> -<a href="register.jsp">Register</a><br /> -<% } %> diff --git a/apps/syndie/jsp/_toplogo.jsp b/apps/syndie/jsp/_toplogo.jsp index 12f035026996f7eab105444c80c289e69c37ba37..07d6f60abce908cafd0d16a8c7881c1cb6e4e307 100644 --- a/apps/syndie/jsp/_toplogo.jsp +++ b/apps/syndie/jsp/_toplogo.jsp @@ -1,7 +1,5 @@ <%@page import="net.i2p.syndie.BlogManager" %> <jsp:useBean scope="session" class="net.i2p.syndie.User" id="user" /> -<jsp:useBean scope="session" class="net.i2p.syndie.data.ArchiveIndex" id="archive" /> -<% session.setAttribute("archive", BlogManager.instance().getArchive().getIndex()); %> <!-- <center>[syndiemedia]</center> --> \ No newline at end of file diff --git a/apps/syndie/jsp/_topnav.jsp b/apps/syndie/jsp/_topnav.jsp index 151bd30baf351bd81ab99f5182c3a337a712bf3f..43a2055d2d2497455581e064ed3b94e6a5b8735e 100644 --- a/apps/syndie/jsp/_topnav.jsp +++ b/apps/syndie/jsp/_topnav.jsp @@ -1,3 +1,42 @@ -<td valign="top" align="left" class="syndieTopNavBlogsCell" height="10"><a href="index.jsp">Blogs</a></td> -<td valign="top" align="left" class="syndieTopNavRemoteCell" height="10"><a href="remote.jsp">Remote archives</a></td> -<td valign="top" align="left" class="syndieTopNavManageCell" height="10">Manage</td> \ No newline at end of file +<%@page import="net.i2p.syndie.*, net.i2p.syndie.sml.*, net.i2p.syndie.web.*" %> +<jsp:useBean scope="session" class="net.i2p.syndie.User" id="user" /> +<td valign="top" align="left" class="syndieTopNavBlogsCell" height="10"><a href="index.jsp">Home</a></td> +<td valign="top" align="left" class="syndieTopNavRemoteCell" height="10"> +<a href="remote.jsp">Remote archives</a> +<a href="import.jsp">Import</a> +</td> +<form action="<%=request.getRequestURI() + "?" + (request.getQueryString() != null ? request.getQueryString() : "")%>"> +<td nowrap="true" valign="top" align="right" class="syndieTopNavManageCell" height="10"><% +if ("true".equals(request.getParameter("logout"))) { + user.invalidate(); +} +String login = request.getParameter("login"); +String pass = request.getParameter("password"); +String loginSubmit = request.getParameter("Login"); +if ( (login != null) && (pass != null) && (loginSubmit != null) && (loginSubmit.equals("Login")) ) { + String loginResult = BlogManager.instance().login(user, login, pass); + if (!user.getAuthenticated()) + out.write("<b>" + loginResult + "</b>"); +} +%> +<% if (user.getAuthenticated()) { %> +Logged in as: <b><jsp:getProperty property="username" name="user" />:</b> +<a href="<%=HTMLRenderer.getPageURL(user.getBlog(), null, -1, -1, -1, user.getShowExpanded(), user.getShowImages())%>"><%=HTMLRenderer.sanitizeString(ArchiveViewerBean.getBlogName(user.getBlogStr()))%></a> +<a href="<%=HTMLRenderer.getPostURL(user.getBlog())%>">Post</a> +<a href="<%=HTMLRenderer.getMetadataURL(user.getBlog())%>">Metadata</a> +<a href="index.jsp?logout=true">Logout</a><br /> +<%} else {%> +Login: <input type="text" name="login" size="8" /> +Pass: <input type="password" name="password" size="8" /><% +java.util.Enumeration params = request.getParameterNames(); +while (params.hasMoreElements()) { + String p = (String)params.nextElement(); + String val = request.getParameter(p); + %><input type="hidden" name="<%=p%>" value="<%=val%>" /><% +}%> +<input type="submit" name="Login" value="Login" /> +<a href="register.jsp">Register</a> +<% } %> + +</td> +</form> \ No newline at end of file diff --git a/apps/syndie/jsp/import.jsp b/apps/syndie/jsp/import.jsp index 4fce96933d27d0fc65596f09443d153ef01d264f..3672e79889e8bc0262b073d4e4caae723043045d 100644 --- a/apps/syndie/jsp/import.jsp +++ b/apps/syndie/jsp/import.jsp @@ -1,5 +1,5 @@ <%@page import="net.i2p.data.Base64, net.i2p.syndie.web.*, net.i2p.syndie.sml.*, net.i2p.syndie.data.*, net.i2p.syndie.*, org.mortbay.servlet.MultiPartRequest, java.util.*, java.io.*" %> -<jsp:useBean scope="session" class="net.i2p.syndie.data.ArchiveIndex" id="archive" /> +<jsp:useBean scope="session" class="net.i2p.syndie.data.TransparentArchiveIndex" id="archive" /> <html> <head> <title>SyndieMedia import</title> diff --git a/apps/syndie/jsp/post.jsp b/apps/syndie/jsp/post.jsp index 1015d25310c6baee323c3de12515560b2b453e8d..a7a78cc4b6089c3cae5fe5f1a2470ec70ada320e 100644 --- a/apps/syndie/jsp/post.jsp +++ b/apps/syndie/jsp/post.jsp @@ -113,13 +113,13 @@ if ( (s != null) && (s.trim().length() > 0) ) {%> Attachment 0: <input type="file" name="entryfile0" /><br /> Attachment 1: <input type="file" name="entryfile1" /><br /> Attachment 2: <input type="file" name="entryfile2" /><br /> -Attachment 3: <input type="file" name="entryfile3" /><br /> +Attachment 3: <input type="file" name="entryfile3" /><br /><!-- Attachment 4: <input type="file" name="entryfile4" /><br /> Attachment 5: <input type="file" name="entryfile5" /><br /> Attachment 6: <input type="file" name="entryfile6" /><br /> Attachment 7: <input type="file" name="entryfile7" /><br /> Attachment 8: <input type="file" name="entryfile8" /><br /> -Attachment 9: <input type="file" name="entryfile9" /><br /> +Attachment 9: <input type="file" name="entryfile9" /><br />--> <hr /> <input type="submit" name="Post" value="Preview..." /> <input type="reset" value="Cancel" /> <% diff --git a/apps/syndie/jsp/remote.jsp b/apps/syndie/jsp/remote.jsp index 8270c6418f3f942683e4614cde66593c65de7c36..4720222d0d96b6ea97a5cf56e1927bb40f35be93 100644 --- a/apps/syndie/jsp/remote.jsp +++ b/apps/syndie/jsp/remote.jsp @@ -1,7 +1,7 @@ <%@page contentType="text/html" import="net.i2p.syndie.web.*" %> <jsp:useBean scope="session" class="net.i2p.syndie.web.RemoteArchiveBean" id="remote" /> <jsp:useBean scope="session" class="net.i2p.syndie.User" id="user" /> -<jsp:useBean scope="session" class="net.i2p.syndie.data.ArchiveIndex" id="archive" /> +<jsp:useBean scope="session" class="net.i2p.syndie.data.TransparentArchiveIndex" id="archive" /> <html> <head> <title>SyndieMedia</title> @@ -17,37 +17,41 @@ <% if (!user.getAuthenticated() || !user.getAllowAccessRemote()) { %>Sorry, you are not allowed to access remote archives from here. Perhaps you should install Syndie yourself?<% -} else { - %>Import from: +} else { %>Import from: <select name="schema"> - <option value="web">Web</option> - <option value="eep">I2P</option> - <option value="tor">TOR</option> - <option value="freenet">Freenet</option> + <option value="web">I2P/TOR/Freenet</option> <option value="mnet">MNet</option> <option value="feedspace">Feedspace</option> <option value="usenet">Usenet</option> -</select> -<input name="location" size="60" /> <input type="submit" name="action" value="Continue..." /> +</select> +Proxy <input type="text" size="10" name="proxyhost" value="localhost" />:<input type="text" size="4" name="proxyport" value="4444" /> +<input name="location" size="40" /> <input type="submit" name="action" value="Continue..." /><br /> <% String action = request.getParameter("action"); if ("Continue...".equals(action)) { - remote.fetchIndex(user, request.getParameter("schema"), request.getParameter("location")); + remote.fetchIndex(user, request.getParameter("schema"), request.getParameter("location"), request.getParameter("proxyhost"), request.getParameter("proxyport")); } else if ("Fetch metadata".equals(action)) { remote.fetchMetadata(user, request.getParameterMap()); } else if ("Fetch selected entries".equals(action)) { remote.fetchSelectedEntries(user, request.getParameterMap()); } else if ("Fetch all new entries".equals(action)) { remote.fetchAllEntries(user, request.getParameterMap()); + } else if ("Post selected entries".equals(action)) { + remote.postSelectedEntries(user, request.getParameterMap()); } String msgs = remote.getStatus(); if ( (msgs != null) && (msgs.length() > 0) ) { %><pre><%=msgs%> -<a href="remote.jsp">Refresh</a></pre><br /><% } +<a href="remote.jsp">Refresh</a></pre><br /><% + } if (remote.getFetchIndexInProgress()) { %><b>Please wait while the index is being fetched from <%=remote.getRemoteLocation()%></b>. <% } else if (remote.getRemoteIndex() != null) { // remote index is NOT null! - %><b><%=remote.getRemoteLocation()%></b>:<br /> + %><b><%=remote.getRemoteLocation()%></b> +<a href="remote.jsp?schema=<%=remote.getRemoteSchema()%>&location=<%=remote.getRemoteLocation()%><% +if (remote.getProxyHost() != null && remote.getProxyPort() > 0) { + %>&proxyhost=<%=remote.getProxyHost()%>&proxyport=<%=remote.getProxyPort()%><% +} %>&action=Continue...">(refetch)</a>:<br /> <%remote.renderDeltaForm(user, archive, out);%> <textarea style="font-size:8pt" rows="5" cols="120"><%=remote.getRemoteIndex()%></textarea><% } diff --git a/apps/syndie/jsp/web.xml b/apps/syndie/jsp/web.xml index 53125bbbe7ebe8b2c0f664667f5fae8bdf22e717..2251d2107719fc21b6e583ccb700b043dda080f6 100644 --- a/apps/syndie/jsp/web.xml +++ b/apps/syndie/jsp/web.xml @@ -4,11 +4,21 @@ "http://java.sun.com/j2ee/dtds/web-app_2.2.dtd"> <web-app> + <servlet> + <servlet-name>net.i2p.syndie.web.ArchiveServlet</servlet-name> + <servlet-class>net.i2p.syndie.web.ArchiveServlet</servlet-class> + </servlet> + <!-- precompiled servlets --> + <servlet-mapping> <servlet-name>net.i2p.syndie.jsp.index_jsp</servlet-name> <url-pattern>/</url-pattern> </servlet-mapping> + <servlet-mapping> + <servlet-name>net.i2p.syndie.web.ArchiveServlet</servlet-name> + <url-pattern>/archive/*</url-pattern> + </servlet-mapping> <session-config> <session-timeout>