- Conditional Tags
- Setting request attributes
- Reading and using User Attributes
To address these issues I decided to build a custom DS tag that would call the weather feed and store a cached instance for each location requested.
If 1000 people access the site within a 2 hour period (500 from Toronto and 500 from New York), the IOLet will save an exact copy of the XML for both Toronto and New York. The first person from New York that access the site will download a copy where as the next 499 people will access that local copy. (Same for Toronto)
The Basic Idea
Technologies:
- EHCache
- Log4j
weather.xml
<rde-dm:iolet name="weatherfeed" method="GetWeatherOutput" tag="notag"> <location>[#user:preferences.widget.weather.location#]</location> </rde-dm:iolet>
Here you find the magic.
WeatherIOLet.class
public class WeatherIOLet extends IoletAdapter { public String location; public String BuildErrorResponse(String error) { String xml = "<error>" + " <note>"+error+"</note>" + "</error>"; return xml; } public IoletResponse doGetWeather(CoaSession session, IoletRequest request) { //CoaUser sessionUser = session.getCoaUser(); // can we get other Attributes from the the session User? if (request.getParameter("location") == null || request.getParameter("location").toString().equals("")) { // error IoletResponse response = IoletResponseFactory.getIoletResponse(this.BuildErrorResponse("Location not specified")); response.setParameter(IoletResponse.XML_TEXT, Boolean.TRUE); return response; } else { this.location = (String) request.getParameter("location"); WeatherService ws = new WeatherService(this.location); IoletResponse response = IoletResponseFactory.getIoletResponse(ws.getXML()); response.setParameter(IoletResponse.XML_TEXT, Boolean.TRUE); return response; } } }
The WeatherXMLDAO is always called by the IOLet - each time the user accesses a page with the weather widget on it, this class will check the cache for a valid instance and if one exists it will return the full XML back to the IOLet. If the cache is empty or out of date (determined by the cache - I defined at 2 hours) then it will make a call to the WeatherCacheItem class which will make a URL request against the weather service.
WeatherXMLDAO.class
public class WeatherXMLDAO implements XMLDAOIF { ... public WeatherXMLDAO(String cache_name) { super(); this.cache_name = cache_name; this.cache_init(); } @Override public String getXMLOutput() { // can not pass in parameters as it may be to difficult to pull out - maybe use HashMap with a key/value pairs _logger.debug("grabbing XML from cache"); WeatherCacheItem wci = new WeatherCacheItem(wr.getParams()); SelfPopulatingCache spc = new SelfPopulatingCache(cache, wci); if (spc.isElementInMemory(wr.getLocation())) { _logger.info("Item is in memory "); } return (String) spc.get(wr.getLocation()).getObjectValue(); } private void cache_init() { _logger.info("initializing cache: " + this.cache_name); if (this.cache == null) { _logger.debug("Cache does not exist: Creating new cache object"); CacheManager manager = CacheManager.getInstance(); cache = manager.getEhcache(this.cache_name); } } }
WeatherCacheItem.class
public class WeatherCacheItem implements CacheEntryFactory {
...
public Object createEntry(Object arg0) throws Exception {
String url = "http://" + props.getProperty("weather.host") + props.getProperty("weather.uri") + this.params;
URL u = null;
try {
u = new URL(url);
URLConnection conn = u.openConnection();
_logger.debug("Opening Weather URL: " + url );
BufferedReader br = new BufferedReader(new InputStreamReader(conn.getInputStream(),"ISO-8859-1"));
sb = new StringBuilder(16384);
try {
String line;
while ((line = br.readLine()) != null) {
sb.append(line);
sb.append('\n');
}
} finally {
br.close();
}
_logger.debug("Weather XML Output: " + sb.toString());
} catch (MalformedURLException mue) {
_logger.fatal("Malformed Exception: " + mue.toString());
} catch (IOException ioe) {
_logger.fatal("IO Exception: " + ioe.toString());
}
return sb.toString();
}
}
Enhance the Idea
To enhance this functionality would be to optimize the code to support additional external data feeds and apply the same caching process and described above. I have enhanced the application to support RSS external news feeds, stock feeds, and Career Opportunity feeds. As each feed requires a different set of requirements, I have enhanced how the process works slightly.I built a "codec" to interpret and decode a number of parameters required for each feed. These parameters further defined the number of cache stores.
WeatherRequest.class
public class WeatherRequest implements RequestIF { String location; public WeatherRequest(String location) { this.location = location; } public String getLocation() { return this.location; } @Override public String getParams() { Configurator props = Configurator.getInstance(); String param = props.getProperty("weather.params.location").replaceFirst("\\[\\#location\\#\\]", this.getLocation()); return param; } }
ehcache.xml
<cache name="weather_cache" maxElementsInMemory="10000" maxElementsOnDisk="1000" eternal="false" overflowToDisk="true" diskSpoolBufferSizeMB="20" timeToIdleSeconds="7200" timeToLiveSeconds="10800" memoryStoreEvictionPolicy="LFU" /> <cache name="stock_cache" maxElementsInMemory="100" maxElementsOnDisk="1000" eternal="false" overflowToDisk="true" diskSpoolBufferSizeMB="20" timeToIdleSeconds="180" timeToLiveSeconds="300" memoryStoreEvictionPolicy="LFU" />