Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/itzmeanjan/mapz
A Geospatial Application
https://github.com/itzmeanjan/mapz
express javascript map mapnik nodejs osm python python3 tile-generation tile-server tiles tms
Last synced: 3 months ago
JSON representation
A Geospatial Application
- Host: GitHub
- URL: https://github.com/itzmeanjan/mapz
- Owner: itzmeanjan
- License: mit
- Created: 2019-04-09T18:05:56.000Z (almost 6 years ago)
- Default Branch: master
- Last Pushed: 2019-04-16T12:24:49.000Z (almost 6 years ago)
- Last Synced: 2024-10-03T15:54:54.605Z (4 months ago)
- Topics: express, javascript, map, mapnik, nodejs, osm, python, python3, tile-generation, tile-server, tiles, tms
- Language: Python
- Homepage:
- Size: 19.8 MB
- Stars: 16
- Watchers: 3
- Forks: 4
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
README
# mapZ
A Geospatial Application, which I'm still working on.Putting :star: might be a bit inspirational to me.
# Documentation
:eyes: : So, you interested in building a mapping application, is it ?
:eyes: : Yes
:eyes: : Okay, I'm gonna take you through each and every step. All you need to do is to follow me. Ready ?
:eyes: : Definitely
## Platform and Tools used
I'm going to use [*Fedora Linux*](https://getfedora.org/) for implementing this whole process.
For database implementation I'll use *PostgreSQL*.
```shell
postgres (PostgreSQL) 10.7
```
GeoSpatial Data to be stored and processed using *PostGIS*.
```shell
Name : postgis
Version : 2.4.3
```
A lot of *Python* scripts are going to be used for automating tile generation and database population procedure.
```shell
Python 3.7.3
```
*NPM* used for installing different *JavaScript* dependencies.
```shell
>> npm --version
6.4.1
```
*Express* app to be written for implementing Tile Map Server.
```shell
>> node --version
v10.15.0
>> npm info express
[email protected]
```
*Mapnik* used for rendering map tiles.
```shell
>> mapnik --version
3.0.20
```
Well that's it :smile:.## Initial SetUp
First thing first, you need world map data ( here we'll be using shapefiles ) to build a map of world. We'll use [GADM](https://gadm.org) as map data source. Here is the [data](https://biogeo.ucdavis.edu/data/gadm3.6/gadm36_levels_shp.zip), which we'll require.So I've written small bash script for you to download and unzip that data. Well you're free to do that on your own too.
```bash
#!/usr/bin/bash
# script downloads shape files from GADM, make sure you're connected to internet
wget https://biogeo.ucdavis.edu/data/gadm3.6/gadm36_levels_shp.zip
# unzips downloaded zip into multiple layered shape files, which will be later on used for inflating features into database
unzip gadm36_levels_shp.zip
```
Now let's setup a [PostgreSQL](https://postgresql.org/) database with [postgis](https://postgis.net/) extension enabled.
Make sure you've installed PostgreSQL database properly on your system. I found [it](http://www.glom.org/wiki/index.php?title=Initial_Postgres_Configuration) helpful.
Login to PostgreSQL.
```bash
>> psql --username=your-user-name-for-postgresql # which is generally postgres
```
Create a database named *world_features*.
```bash
>> create database world_features;
```
Now simply quit i.e. logout from *psql* prompt.
```bash
>> \q
```
Relogin to use newly created database.
```bash
>> psql --username=your-user-name-for-postgresql --dbname=world_features
```
Let's enable *postgis extension* for this database. Make sure you've installed *postgis* first.
```bash
>> create extension postgis;
```
And initial setup is done. Now we gonna automate things :wink:.
## Database Population
:eyes: : That shapefile we downloaded, has 6 layers. So we'll be creating 6 different tables, where we're going to store that huge map data.
:eyes: : :worried:
:eyes: : Hey wait, we're going to automate that whole thing. Now happy ???
:eyes: : :relaxed:
Alright so let's automate. And don't forget to grab a cup of :coffee:, cause this gonna be a bit longer.
See we're going to process almost few hundreds of thousands features ( mostly polygon / multipolygon geometry ), using *geo.Open('/path-to-gadm36_0.shp').GetLayer(0).GetFeature(j).GetGeometryRef().ExportToWkt()*, where j is feature count. So of course this gonna be time consuming.
```python
def app(path='/path-to-file/gadm36_{}.shp', file_id=[0, 1, 2, 3, 4, 5]):
# path, path to gadm shapefiles
# gadm has 6 layers, shape files hold corresponding layer number too
print('[+]Now grab a cup of coffee, cause this gonna be a little longer ...\n')
for i in file_id:
print('[+]Working on `{}`'.format(path.format(i)))
datasource = geo.Open(path.format(i)) # datasource opened
# layer fetched, only a single layer present in a shape file
layer = datasource.GetLayer(0)
tmp = []
for j in range(layer.GetFeatureCount()):
feature = layer.GetFeature(j) # gets feature by id
gid = 'NA'
name = 'NA'
# there might be some fields present in shapefile, which is None
if(feature.items().get('GID_{}'.format(i)) is not None):
# To handle so, I'm adding these two checks, otherwise those might be causing problem during database population
gid = feature.items().get('GID_{}'.format(i))
if(feature.items().get('NAME_{}'.format(i)) is not None):
name = feature.items().get('NAME_{}'.format(i))
tmp.append([gid, name,
feature.GetGeometryRef().ExportToWkt()])
# holds data in temp variable
# data format -- [feature_id, feature_name, outline]
if(inflate_into_db('world_features', 'username', 'password', {i: tmp})):
# finally inflate into database
print('[+]Success')
return
```
Don't forget to change username and password, required for database login, before running [this](https://github.com/itzmeanjan/mapZ/blob/master/fetch_and_push.py) script.
```python
if(inflate_into_db('world_features', 'username', 'password', {i: tmp})):
# finally inflate into database
print('[+]Success')
```
Simply run [this](https://github.com/itzmeanjan/mapZ/blob/master/fetch_and_push.py) script and it'll be done.
```shell
>> python3 fetch_and_push.py
```
And it's done. Wanna check ?
Login to *postgresql*.
```shell
>> psql --username=your-user-name-for-postgresql --dbname=world_features
```
Type in *psql* prompt.
```sql
>> select feature_id, feature_name from world_features_level_0 where feature_id = 'IND';
feature_id | feature_name
------------+--------------
IND | India
(1 row)
```
And this is the structure of table, which the script built for us. All *6 tables* has same structure.
```sql
>> \d world_features_level_0;
Table "public.world_features_level_0"
Column | Type | Collation | Nullable | Default
--------------+-------------------+-----------+----------+---------
feature_id | character varying | | not null |
feature_name | character varying | | not null |
outline | geography | | |
Indexes:
"world_features_level_0_pkey" PRIMARY KEY, btree (feature_id)
"world_features_level_0_index" gist (outline)
```
## Tile Generation
This gonna be way more time consuming than previous one. I'm still working on it. I'll be back :sunglasses:.
## Tile Map Server
The *Tile Map Server*, built using [NodeJS](https://nodejs.org/en/) i.e. [ExpressJS](http://expressjs.com/) Framework resides [here](https://github.com/itzmeanjan/mapZ/tree/master/tms).
Get into *tms* directory and run following command, which will download all dependencies, required for running this express app.
```shell
>> npm install
```
This *Express* app will be working in local network. Make necessary changes, so that it can be discovered from Internet.
```javascript
app.listen(8000, '0.0.0.0', () => {
// tms listens at 0.0.0.0:8000, so that it can be accessed via both localhost and devices present in local network
console.log('[+]Tile Map Server listening at - `0.0.0.0: 8000`\n');
});
```
Tile Map Server will be accepting *GET* request in */tile/:zoom/:row/:col.png* path, where *zoom* is Zoom Level value, *row* is Row ID( tile identifier along X-axis ) and *col* is Column ID( tile identifier along Y-axis ).
```javascript
app.get('/tile/:zoom/:row/:col.png', (req, res) => {
.
. // lots of code
.
});
```
Ready to run Tile Map Server ?
```shell
>> node index.js
```
As you can see on terminal its running. Just head to [this url](http://localhost:8000/tile/0/0/0.png), and you get to see a tile, which you built during that *very long* running tile generation procedure.
![Working Tile Map Server](https://github.com/itzmeanjan/mapZ/blob/master/screenshot_1.png)
## Mapping Application
Let's first talk about server side.
### Server Side
We gonna write an [Express](http://expressjs.com/) App, which will be serving web pages for displaying maps, and for client, will simply reply on browser, which makes this whole venture a bit platform independent.
Mapping application resides [here](https://github.com/itzmeanjan/mapZ/tree/master/app). As I assume, you've already successfully completed previous steps, simply get into *app* directory and run *app/index.js* for launching mapping application server.
```shell
>> cd app
>> node index.js
```
![mapping application server, running](https://github.com/itzmeanjan/mapZ/blob/master/screenshot_2.png)
Back to client.
### Client Side
You might have already found that there's a [*static*](https://github.com/itzmeanjan/mapZ/tree/master/app/static) directory inside *app*, which holds a simple web page, [*index.html*](https://github.com/itzmeanjan/mapZ/blob/master/app/static/index.html), which will be served by *app/index.js*, ( express app ) and browser will be using it for displaying map with the help of a great JavaScript library [*leaflet*](https://leafletjs.com/).
For using *leaflet* in our app, we're going to put following two tags in *head* tag of [*index.html*](https://github.com/itzmeanjan/mapZ/blob/master/app/static/index.html).
```html
```
For displaying a map we need a *div* element with *id* attribute set, in *body* of html. You need to specify *height* for this *div* element.
```html
body {
margin: 0;
padding: 0;
}
html,
body,
#map {
height: 100%;
}
```
![map on client side 1](https://github.com/itzmeanjan/mapZ/blob/master/screenshot_3.png)
![map on client side 2](https://github.com/itzmeanjan/mapZ/blob/master/screenshot_4.png)
![map on client side 3](https://github.com/itzmeanjan/mapZ/blob/master/screenshot_5.png)
Put this script with in *body* of html, which will display map on browser. Of course the heavy lifting is done by *leaflet*.
```javascript
var map = L.map('map', {
maxZoom: 5,
minZoom: 0,
});
L.tileLayer('http://localhost:8000/tile/{z}/{x}/{y}.png', {
attribution: '© 2019 Anjan Roy'
}).addTo(map);
map.setView([22, 83], 3);
```
You see, I've created a map with *maxZoom: 5* and *minZoom: 0*, which will be displayed with in a *div* element, identified by *map*, id attribute.
```javascript
var map = L.map('map', {
maxZoom: 5,
minZoom: 0,
});
```
![map on client side 4](https://github.com/itzmeanjan/mapZ/blob/master/screenshot_6.png)
![map on client side 5](https://github.com/itzmeanjan/mapZ/blob/master/screenshot_7.png)
![map on client side 6](https://github.com/itzmeanjan/mapZ/blob/master/screenshot_8.png)
Next we're going to add a *tileLayer*, used for displaying tiles. And the url for *tms* is *http://localhost:8000/tile/{z}/{x}/{y}.png*, where *z* denotes Zoom Level, *x* denotes tile-id along X-axis and *y* denotes tile-id along Y-axis.
Well, the top-left most tile is identified as *0-0* tile. After that as you move towards right, *x* increases and moving downward increases *y* value.
```javascript
L.tileLayer('http://localhost:8000/tile/{z}/{x}/{y}.png', {
attribution: '© 2019 Anjan Roy'
}).addTo(map);
```
And we add tileLayer to map. and :boom: !!!
![map on client side 7](https://github.com/itzmeanjan/mapZ/blob/master/screenshot_9.png)
![map on client side 8](https://github.com/itzmeanjan/mapZ/blob/master/screenshot_10.png)
![map on client side 9](https://github.com/itzmeanjan/mapZ/blob/master/screenshot_11.png)
![map on client side 10](https://github.com/itzmeanjan/mapZ/blob/master/screenshot_12.png)
![map on client side 11](https://github.com/itzmeanjan/mapZ/blob/master/screenshot_13.png)
![map on client side 12](https://github.com/itzmeanjan/mapZ/blob/master/screenshot_14.png)
Was it hard ???
:eyes: -- Facing some problems ?
:eyes: -- Yes
:eyes: -- Alright, find me [here](https://twitter.com/meanjanry)
## Courtesy
Thanking [Tobaloidee](https://github.com/Tobaloidee) :heart: -ily for designing a logo for this project.