This is typically done via a GeoProcessing task where the data is queried and stored into an intermediate FeatureClass that is further processed by a Kernel Density that produced a raster layer that is finally visualized. As you can tell, this is not interactive nor relatively fast.
Since the traditional means of retrieving the data from a relational database is not fast enough and 11 million records is not such a big set after all, I decided to put the whole thing in memory. BTW, this is a meme that has been trending for while now, and the most vocal about it is SAP HANA.
I decided to use Terracotta's BigMemory to hold the data in the "off-heap" and use its EHCache query capability to aggregate and fetch the data. Now despite the name, I used the cache "eternal" capabilities to forever hold the data elements.
I was given the data in CSV format, so I wrote a simple CSV reader/tokenizer that bulk loads the data into BigMemory using multiple threads. The inter thread communication was handled by the LMAX disruptor, letting me focus on the loading, parsing and putting into BigMemory.
To access this BigMemory from a webmap as a layer, I decided to write a middleware that implements the ArcGIS Rest JSON interface for an ArcGISDynamicLayer. Now, I did not write the whole interface, just the layer metadata and the export image endpoints, in such that you can access the layer as:
http://host/arcgis/rest/services/Service/MapServer
The implementation of the Rest endpoint was easily achievable using the Spring MVC framework which enables me to export POJO methods as ArcGIS Rest endpoints:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
@RequestMapping(value = "/rest/services/InfoUSA/MapServer", method = RequestMethod.GET) | |
public void doMapServer( | |
final HttpServletResponse response | |
) throws IOException | |
{ | |
response.setStatus(HttpServletResponse.SC_OK); | |
response.setContentType("application/json"); | |
response.setContentLength(m_byteArrayOutputStream.size()); | |
m_byteArrayOutputStream.writeTo(response.getOutputStream()); | |
response.getOutputStream().flush(); | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
@RequestMapping(value = "/rest/services/InfoUSA/MapServer/export", method = {RequestMethod.GET, RequestMethod.POST}) | |
public void doExport( | |
@RequestParam("bbox") final String bbox, | |
@RequestParam(value = "size", required = false) final String size, | |
@RequestParam(value = "layerDefs", required = false) final String layerDefs, | |
@RequestParam(value = "transparent", required = false) final String transparent, | |
final HttpServletResponse response | |
) throws IOException |
To run this PoC, I started 2 extra large Amazon EC2 instances to run BigMemory with 15GB of off heap RAM (just because I can :-) and started a medium instance to run tomcat with the Spring MVC Rest interface that communicated with two large instances.
The PoC was very well received and extended it to make the layer time aware enabling the client to slide though time aware features.
Like usual, you can download all the source code from here, here and here.
2 comments:
I'm unable to run this application , since i'm missing the com.esri.webmercator.jar. Please let me know , where can i find the jar.
https://github.com/mraad/WebMercator
Post a Comment