/* * RssItem.java * * Copyright (C) 2005-2006 Tommi Laukkanen * http://www.substanceofcode.com * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA * */ // Expand to define logging define //#define DNOLOGGING package com.substanceofcode.rssreader.businessentities; import java.io.UnsupportedEncodingException; import java.util.Date; import com.substanceofcode.utils.Base64; import com.substanceofcode.utils.StringUtil; import com.substanceofcode.utils.CauseException; import com.substanceofcode.utils.CauseMemoryException; //#ifdef DLOGGING import net.sf.jlogmicro.util.logging.Logger; import net.sf.jlogmicro.util.logging.Level; //#endif /** * RssItem class is a data store for a single item in RSS feed. * One item consist of title, link, description and optional date. * * @author Tommi Laukkanen * @version 1.1 */ public class RssItem { public static final int TITLE_OFFSET = 0; public static final int DATE_OFFSET = 2; public static final int UNREAD_ITEM = 4; public static final int DESC_OFFSET = 5; public static final char CONE = (char)1; //#ifdef DLOGGING private Logger logger = Logger.getLogger("RssItem"); //#endif // Use protected so that sub classes can access these including the // backward store compatibility classes. protected String m_title = ""; // The RSS item title protected String m_link = ""; // The RSS item link protected String m_desc = ""; // The RSS item description protected Date m_date = null; protected String m_enclosure = ""; // The RSS item enclosure protected boolean m_unreadItem = false; //#ifdef DLOGGING private boolean fineLoggable = logger.isLoggable(Level.FINE); private boolean finestLoggable = logger.isLoggable(Level.FINEST); //#endif /** Creates a new instance of RssItem. Used by this class and RssItem and later the fields are initalized. */ protected RssItem() { } /** Creates a new instance of RssItem */ public RssItem(String title, String link, String desc, Date pubDate, String enclosure, boolean unreadItem) { m_title = title; m_link = link; m_desc = desc; m_date = pubDate; m_enclosure = enclosure; m_unreadItem = unreadItem; } public RssItem(RssItem item) { this(item.m_title, item.m_link, item.m_desc, item.m_date, item.m_enclosure, item.m_unreadItem); } /** Get RSS item title */ public String getTitle(){ return m_title; } /** Get RSS item link address */ public String getLink(){ return m_link; } /** Get RSS item description */ public String getDescription(){ return m_desc; } /** Get RSS item publication date */ public Date getDate() { return m_date; } /** Serialize the object When we serialize we don't do anything special for itunes as the store to memory will be deserialized only by the iTunes capable version. */ public String unencodedSerialize() { String dateString; if(m_date==null){ dateString = ""; } else { // We use base 16 (hex) for the date so that we can save some // space for toString. dateString = Long.toString( m_date.getTime(), 16 ); } String title = m_title.replace('|', CONE); String preData = title + "|" + m_link + "|" + dateString + "|" + m_enclosure + "|" + (m_unreadItem ? "1" : "0") + "|" + m_desc; return (preData); } /** Serialize the object this serialize does not need to know if Itunes is capable/enabled given that no fields were added to make it capable/enabled */ public String serialize() { String preData = unencodedSerialize(); Base64 b64 = new Base64(); String encodedSerializedData = null; try { encodedSerializedData = b64.encode( preData.getBytes("UTF-8") ); } catch (UnsupportedEncodingException e) { encodedSerializedData = b64.encode( preData.getBytes() ); } return encodedSerializedData; } /** Initialize fields in the class from data. startIndex - Starting index in nodes of RssItem iTunesCapable - True if the data can support Itunes (but may not actually have Itunes data) or may not be turned on by the user. So, the serializaion/deserialization will account for iTunes fields except if not enabled, the will have empty values. If itunes capable we use base 16 (hex) for the date so that we can save some space for toString. hasPipe - True if the data has a pipe in at least one item nodes - (elements in an array). **/ protected void init(int startIndex, boolean iTunesCapable, boolean hasPipe, String [ ] nodes) throws CauseMemoryException, CauseException { try { /* Node count should be 6: * title | link | date | enclosure | unreadItem | desc */ int TITLE = TITLE_OFFSET; //#ifdef DLOGGING if (finestLoggable) {logger.finest("startIndex,nodes.length,first nodes=" + startIndex + "," + nodes.length + "|" + nodes[ startIndex + TITLE ]);} //#endif m_title = nodes[startIndex + TITLE]; if (hasPipe) { if (iTunesCapable) { m_title = m_title.replace(CONE, '|'); } else { m_title = m_title.replace('\n', '|'); } } int LINK = 1; m_link = nodes[startIndex + LINK]; int DATE = DATE_OFFSET; String dateString = nodes[startIndex + DATE]; if(dateString.length()>0) { if (iTunesCapable) { m_date = new Date(Long.parseLong(dateString, 16)); } else { m_date = new Date(Long.parseLong(dateString)); } } else { m_date = null; } int ENCLOSURE = 3; m_enclosure = nodes[startIndex + ENCLOSURE]; int UNREADITEM = UNREAD_ITEM; String cunreadItem = nodes[startIndex + UNREADITEM]; m_unreadItem = (cunreadItem.equals("1")); // If description has '|', we need to join. int DESC = DESC_OFFSET; if (DESC + startIndex < (nodes.length - 1)) { m_desc = StringUtil.join(nodes, "|", startIndex + DESC); } else { m_desc = nodes[startIndex + DESC]; } } catch(Exception e) { CauseException ce = new CauseException( "Internal error while RssItem init ", e); //#ifdef DLOGGING logger.severe(ce.getMessage(), e); //#endif System.err.println(ce.getMessage() + " " + e.toString()); e.printStackTrace(); throw ce; } catch(OutOfMemoryError e) { CauseMemoryException ce = new CauseMemoryException( "Out of memory error while RssItem init ", e); //#ifdef DLOGGING logger.severe(ce.getMessage(), e); //#endif System.err.println(ce.getMessage() + " " + e.toString()); e.printStackTrace(); throw ce; } } /** Deserialize the object **/ public static RssItem deserialize(String encodedData) throws CauseMemoryException, CauseException { try { // Base64 decode Base64 b64 = new Base64(); byte[] decodedData = b64.decode(encodedData); String data; try { data = new String( decodedData, "UTF-8" ); } catch (UnsupportedEncodingException e) { data = new String( decodedData ); } return unencodedDeserialize(data); } catch(CauseException e) { throw e; } catch(Exception e) { CauseException ce = new CauseException( "Internal error while RssItem deserialize ", e); //#ifdef DLOGGING Logger logger = Logger.getLogger("RssItem"); logger.severe(ce.getMessage(), e); //#endif System.err.println(ce.getMessage() + " " + e.toString()); e.printStackTrace(); throw ce; } catch(OutOfMemoryError e) { CauseMemoryException ce = new CauseMemoryException( "Out of memory error while RssItem deserialize ", e); //#ifdef DLOGGING Logger logger = Logger.getLogger("RssItem"); logger.severe(ce.getMessage(), e); //#endif System.err.println(ce.getMessage() + " " + e.toString()); e.printStackTrace(); throw ce; } } /** This is only used to deserialize data from a previous version but not the initial version as that version has a bug in getting the items. */ public static RssItem unencodedDeserialize(String data) throws CauseMemoryException, CauseException { RssItem item = new RssItem(); try { boolean hasPipe = (data.indexOf('\n') >= 0); String[] nodes = StringUtil.split( data, '|'); item.init(0, false, hasPipe, nodes); return item; } catch(Exception e) { CauseException ce = new CauseException( "Internal error while RssItem unencodedDeserialize ", e); //#ifdef DLOGGING Logger logger = Logger.getLogger("RssItem"); logger.severe(ce.getMessage(), e); //#endif System.err.println(ce.getMessage() + " " + e.toString()); e.printStackTrace(); throw ce; } catch(OutOfMemoryError e) { CauseMemoryException ce = new CauseMemoryException( "Out of memory error while RssItem unencodedDeserialize ", e); //#ifdef DLOGGING Logger logger = Logger.getLogger("RssItem"); logger.severe(ce.getMessage(), e); //#endif System.err.println(ce.getMessage() + " " + e.toString()); e.printStackTrace(); throw ce; } } /* Copy to item. */ public RssItem copyTo(RssItem item) { item.m_title = m_title; item.m_link = m_link; item.m_desc = m_desc; item.m_date = m_date; item.m_enclosure = m_enclosure;; item.m_unreadItem = m_unreadItem; return item; } /* Compare item. */ public boolean equals(RssItem item) { if (!item.m_title.equals(m_title)) { //#ifdef DLOGGING if (finestLoggable) {logger.finest("unequal item.m_title,this=" + item.m_title + "," + m_title);} //#endif return false; } if (!item.m_link.equals(m_link)) { //#ifdef DLOGGING if (finestLoggable) {logger.finest("unequal item.m_link,this=" + item.m_link + "," + m_link);} //#endif return false; } if (!item.m_desc.equals(m_desc)) { //#ifdef DLOGGING if (finestLoggable) {logger.finest("unequal item.m_desc,this=" + item.m_desc + "," + m_desc);} //#endif return false; } if ((item.m_date == null) && (this.m_date == null)) { } else if ((item.m_date != null) && (this.m_date != null)) { if (item.m_date.equals(this.m_date)) { } else { //#ifdef DLOGGING if (finestLoggable) {logger.finest("unequal dates=" + item.m_date + "," + m_date);} //#endif return false; } } else { //#ifdef DLOGGING if (finestLoggable) {logger.finest("unequal dates=" + item.m_date + "," + m_date);} //#endif return false; } if (!item.m_enclosure.equals(m_enclosure)) { //#ifdef DLOGGING if (finestLoggable) {logger.finest("unequal item.m_enclosure,this=" + item.m_enclosure + "," + m_enclosure);} //#endif return false; } if (item.m_unreadItem != m_unreadItem) { //#ifdef DLOGGING if (finestLoggable) {logger.finest("unequal item.m_unreadItem,this=" + item.m_unreadItem + "," + m_unreadItem);} //#endif return false; } return true; } public void setUnreadItem(boolean unreadItem) { this.m_unreadItem = unreadItem; } public boolean isUnreadItem() { return (m_unreadItem); } public void setEnclosure(String enclosure) { this.m_enclosure = enclosure; } public String getEnclosure() { return (m_enclosure); } /** convert the object to string */ public String toString() { String preData = m_title + "|" + m_link + "|" + m_date + "|" + m_enclosure + "|" + m_unreadItem + "|" + m_desc; return (preData); } }