What does one do when presented with more snow days than expected? My friends in Colorado would probably do something outrageous like skiing, but I found it to be a great opportunity to catch up on some of my recreational coding. Specifically, I wanted to revisit the Node/PostGIS work I blogged about earlier.
As fun as that project was, it was fairly limited in scope and I wanted to spread my wings a little more with Node. So I decided to build a more general-purpose wrapper around PostGIS. Lately, I’ve become a bit obsessed with the idea that PostGIS may be the only GIS tool you really need in terms of data storage, management, and analytics. That’s probably a topic for another post but exploring that concept was a perfect premise for my continued explorations with Node.
I have been circling back to Node over the past few months to continue building my comfort level with it. I tend to eschew frameworks when i have learning something new because I want to get comfortable with the core before I start layering on abstraction. That was my approach with the tile viewer tool I built a while back. For the recent post centered on Amazon RDS, I added Express into the mix, which has been a big help.
This time around, I wanted to dig a little deeper with the node-postgres module and also make the application more modular. I wanted to build around a few core principles:
- Keep it RESTful (or as close to it as I could)
- GeoJSON in/GeoJSON out (so….vector only for now)
- Let PostGIS do the heavy lifting
Getting Started
This time around, I elected to use my local PostgreSQL/PostGIS instance rather than Amazon RDS. This was mainly so I could keep my development isolated on one machine. I already had the basic infrastructure in place from my last time around, so I was able to quickly dive into the meat of things. I decided to scope my initial effort at the following:
- Return the contents of an entire table as GeoJSON, with the ability to choose between features (geometries and attributes) in a GeoJSON feature collection or just geometries in a GeoJSON geometry collection. This should support any table in the database.
- Return those records in a table that intersect a geometry passed in as a parameter. The input geometry would be in GeoJSON format.
- Return a JSON representation of a table’s schema.
- Return a list of tables from the server. This is necessary in order to support the ability to query any available table.
- Implement some simple, but application-specific logic to demonstrate extensibility.
With these goals in mind, I decided to first tackle the issue of extensibility. I wanted to make it as painless as possible and the strategy described in this post seemed to fit the bill. I just had to add the following code block to my server.js (straight from the post):
This will load any .js file in the controllers directory into the application. If they are written to the pattern expected by Express, new resource paths are exposed to the application. The post above describes a simple MVC implementation. Astute readers will note that my take is all “C” without “M” or “V.” I plan to refactor that later but it it was easier for me to keep track of things on this pass with code in one place.
Getting Data
With modularity out of the way, it was time work on the basic structure for getting data from the database. In core.js, I defined a route with a URL template like ‘/vector/:schema/:table/:geom’. This would translate into something like http://localhost/vector/public/parcels/features, which would fetch a GeoJSON feature collection containing the contents of the parcels table. To do that, I need to know the name of the spatial column in the table, which the following helps me retrieve:
The next code block shows how I capture the name of the spatial column and structure the main query, depending on the choice of features or geometry:
As can be seen above, the query will transform the output geometry to WGS84 and convert it to GeoJSON for me. So I’m sticking my third principle by leaning on PostGIS functions here. I plan to stick to GeoJSON’s default spatial reference of WGS84 for now. To roll up the query results into the appropriate GeoJSON object and return it, I handled the ‘row’ and ‘end’ events.
I wrote a helper function to roll up GeoJSON features:
So that’s basic data retrieval. How about that spatial intersect?
A Simple Spatial Query
One thing I failed to mention in the above section, is that all of that is exposed through an HTTP GET request. For this next function, I’m going to use a POST. I went back and forth on that but came down on the side of POST due to the potential for a user to send a very verbose input shape. The function is designed to accept JSON as the body of the request, which would be done in curl like this:
The above action returns the state of Kansas (I knew you were wondering). To make this happen, there are only three things that are different. First, the URL is defined a POST and, second, the code needs to capture the input shape. The first few lines are:
I stringify the JSON since I have to insert it into my SQL. This brings me to the third difference here, the query. This time, I am using ST_INTERSECTS to filter down the response. So, depending on the choice of features or geometry, the query will be similar to:
“select st_asgeojson(st_transform(” + spatialcol + “,4326)) as geojson, * from ” + fullname + ” where ST_INTERSECTS(” + spatialcol + “, ST_SetSRID(ST_GeomFromGeoJSON(‘” + queryshape + “’),4326));”
The rest of the process is similar to the basic query above. With a well-exercised data access pattern in place, querying table schema and layer lists become trivial. Since GeoJSON doesn’t cover these topics, I had to roll my own. I won’t detail the output but the queries are below.
So this gives me everything I need for an all-purpose interface into PostGIS from Node. I could spend the rest of the year similarly wrapping the hundreds of spatial functions in PostGIS but the real power of extensibility is the ability to tailor an application for one’s own needs, based upon one’s detailed understanding of their own data and logic.
Adding Some Customization
To do this, I fell back to the data for Leonardtown, Maryland that have used in a couple of previous posts. I am simply going to expose the ability to query residential or commercial buildings from the data set. For this, all of the prep work is done at the top of the function by simply preparing a WHERE clause.
The primary difference here are that I am using a GET with a query string since I’m not concerned with data size and that I’m building a WHERE clause on a specific column name. What’s not shown is that, farther down, I don’t need to query the name of the spatial column so I can cut out that step. I can do this because I understand my own data so I can be more succinct that if I were writing a more generic function. Using this approach I can also write more complex custom logic in my database, call it from Node, and send the response. In other words, standard web application behavior.
In order to expose this application-specific logic, I just needed expose it in a separate leonardtown.js file and drop it into the ‘controllers’ directory.
Wrapping Up
This post was bit longer than usual but there was lot of ground to cover. I feel like I’m getting more comfortable with the Node ecosystem though I’m still a bit wobbly. My next step is probably to dive a little deeper into the MVC side of things with something like Sails. Having a familiar face like PostGIS on the back end is helping me as I figure out how to perform more meaningful tasks with Node and its related tools.
If you want to check out the full code for this application, it is here: https://github.com/geobabbler/node-gis-server
Someone has been working on a version of this. http://cugos.org/presentations/archive/fallfling2013/ryan_whitley_fallfling2013.pdf –> https://github.com/spatialdev/PGRestAPI Collaboration?
I love the idea of having a fast, simple RESTful geoservice with the fewest moving parts possible. This would run happily in a super tiny $5/month cloud slice. Try that with the leading brand! Please keep going with this. It feels like it needs to be it’s own project..
Thank you. You are kind to say so. I am interested in a thin, extensible, RESTish layer to stand on top of PostGIS and I do plan to keep moving forward with it. I am learning more with each pass and it’s been fun so far. I will say that pull requests are always welcome as I won’t claim to have all the answers. 🙂
Very interesting. I will read it more thoroughly later. In PostGIS trunk you also have “tiny wkb” as a possible format http://postgis.net/docs/manual-dev/ST_AsTWKB.html
I made a small demo some time ago demonstrating “streaming” geometries in tiny wkb format with node from postgis. Note that the streaming is from the center. If you zoom in somewhere it starts in the center of the map.
http://sandbox.jordogskog.no/twkb_node/
Thank you for sharing your work. I have made used of twkb on other tasks. I do a fair bit of real-time processing in my regular work and prefer to pass binary in those situations due to its compact nature. I do that work with more traditional, non-web tool sets, so I appreciate you sharing this example with web sockets. It’s good stuff.
You mean you use wkb, right? Or have you tested twkb? Then I get really exited 🙂 A twkb parser in QGIS for instance would be very interesting since it would reduce the amount of data between PostGIS and QGIS very much. Hopefully twkb will be natively read by OpenLayers 3 https://github.com/openlayers/ol3/pull/1718
Yes, you’re right, thank you. It’s been WKB. I am interested in seeing how I can apply twkb to some of what I’ve been doing. Most of my regular work is in .Net so I’ll need to tackle a parser for that. I think I need more coffee. 🙂