Naksha (नक्शा) is the name of this fork of the XYZ-Hub (pronounced nakshaa or nakśā). It stays a web service for the access and management of geospatial data. This spin-off was done to independently realize needed new features, not planned to be supported in the original XYZ-Hub project. The architecture was modified to allow extensions and plug-ins.
The meaning of Naksha-Hub is “Map-Hub”.
Naksha service, has been designed to primarily offer:
- GeoJson based REST API as service
- GeoJson based Java API as library (interacting directly with database)
- Support Postgres as default Storage system (with inbuilt transactional and history capabilities)
- Support publishing Data Change events (with atleast-once messaging guarantee)
- Additionally, Extension framework to provide ability:
- to integrate REST APIs with your own custom Storage mechanism
- to inject custom business logic as part of REST API request processing pipeline
- to inject custom business logic as part of Data change event processing pipeline (subscription based)
Naksha features are:
- Organize geo datasets in collections.
- Organize access to your data via spaces.
- Listen to changes done to your data via subscriptions.
- Store and manipulate individual geo features (points, line-strings, polygons).
- Retrieve geo features as vector tiles, with or without clipped geometries.
- Search for geo features spatially using a bounding box, radius, or any custom geometry.
- Implement own customer data processing logic as embedded handler or external extension handler.
- Explore geo features by filtering property values.
- Connect with your own data sources.
- Use default PostgresQL implementation with your own database.
- Build a real-time geo-data pipeline with processors.
Naksha uses GeoJSON as the main geospatial data exchange format.
- Java 17+
- Gradle 7.2+
- Postgres 14+ with PostGIS 2.5+
Clone and install the project using:
git clone https://github.com/heremaps/naksha.git
cd naksha
gradle clean build
Naksha relies strongly on Postgres - you need running instance if you want to run Naksha locally.
Supported Postgres version is 14+ (the newer the better).
Apart from bare Postgres, Naksha needs PostGIS extension which is mandatory. Apart from that, optional, but highly recommended extensions that are supported include: pg_hint_plan
and pg_stat_statements
.
You can use standalone instance installed directly on your host machine but there's also a docker image that hosts Postgres 16 with all the extensions mentioned above already installed. You can find its definition in this Dockerfile.
To use the containerized Postgres with your locally runnning Naksha:
-
Navigate to Dockerfile directory:
cd here-naksha-app-service/src/test/psql_container
-
Build the image:
docker build --no-cache -t <IMAGE_ID> .
-
Run the container (supplied options omit all auth - use it only locally, tweak if needed):
docker run -p 5432:5432 -e POSTGRES_PASSWORD=postgres -e POSTGRES_INITDB_ARGS="--auth-host=trust --auth-local=trust" -e POSTGRES_HOST_AUTH_METHOD=trust <IMAGE_ID>
-
Now your database should be available on
localhost
with port5432
- you can start Naksha the same way as described in Run App section. -
[optional extension run] Now you can run naksha jar and include example config with additional extensions enabled like so:
java -jar build/libs/naksha-2.0.6-all.jar test-config-with-extensions 'jdbc:postgresql://localhost:5432/postgres?user=postgres&password=pswd&schema=naksha&app=naksha_local&id=naksha_admin_db'
The service could also be started directly from a fat jar. In this case Postgres and the other optional dependencies need to be started separately.
To build the fat jar, at the root project directory, run one of the following:
# Using machine installed gradle (through apt, brew,... package managers)
gradle shadowJar
# Using gradle wrapper
./gradlew shadowJar
The jar can be found under build/libs/
.
To ramp up Naksha with the jar, run:
java -jar <jar-file> <config-id> <database-url>
# Example 1 : Start service with test config against default Database URL (useful for local env)
java -jar build/libs/naksha-2.0.6-all.jar test-config
# Example 2 : Start service with given custom config (using NAKSHA_CONFIG_PATH) and custom database URL (useful for cloud env)
java -jar build/libs/naksha-2.0.6-all.jar cloud-config 'jdbc:postgresql://localhost:5432/postgres?user=postgres&password=pswd&schema=naksha&app=naksha_local&id=naksha_admin_db'
# Example 3 : Start service with given custom config (using NAKSHA_CONFIG_PATH) and default (local) database URL
java -jar build/libs/naksha-2.0.6-all.jar custom-config
Then use a web browser to connect to localhost:8080
, an OK message should be displayed if the service is up and running.
To run Naksha in container, take a look at these docs
Once application is UP, the OpenAPI specification is accessible at http(s)://{host}:{port}/hub/swagger/index.html
, by default at http://localhost:8080/hub/swagger/index.html
The service persists out of modules with a bootstrap code to start the service. Service provides default configuration in default-config.json.
The custom (external) configuration file can be supplied by modifying environment variable or by creating the default-config.json
file in the corresponding configuration folder.
The exact configuration folder is platform dependent, but generally follows the XGD user configuration directory, standard, so on Linux being by default ~/.config/naksha/
. For Windows the files will reside in the CSIDL_PROFILE folder, by default C:\Users\{username}\.config\naksha\
.
Next to this, an explicit location can be specified via the environment variable NAKSHA_CONFIG_PATH
, this path will not be extended by the naksha/
folder, so you can directly specify where to keep the config files. This is important when you want to start multiple versions of the service: NAKSHA_CONFIG_PATH=~/.config/naksha/ java -jar naksha.jar {arguments}
.
In the custom config file, the name of the individual properties can be set as per source code here NakshaHubConfig.
All properties annotated with @JsonProperty
can be set in custom config file.
Config file is loaded using {config-id}
supplied as CLI argument, as per following precedence on file location (first match wins):
- using env variable
NAKSHA_CONFIG_PATH
(full path will be$NAKSHA_CONFIG_PATH/{config-id}.json
) - as per user's home directory
user.home
(full path will be{user-home}/.config/naksha/{config-id}.json
) - as per config previously loaded in Naksha Admin Storage (PostgreSQL database)
- default config loaded from jar (
here-naksha-lib-hub/src/main/resources/config/default-config.json
)
# Example of env variable NAKSHA_CONFIG_PATH
# First, copy default config to custom location
export NAKSHA_CONFIG_PATH=/my-location/naksha
cp here-naksha-lib-hub/src/main/resources/config/default-config.json $NAKSHA_CONFIG_PATH/
# Modify config as per need
vi $NAKSHA_CONFIG_PATH/default-config.json
# Start application using above config
java -jar naksha.jar default-config
The config also accepts custom RSA256 Private key and multiple Public key files (in PEM format) to support JWT signing/verification operations.
- If custom Private key not provided, default will be loaded from Jar bundled resource here-naksha-app-service/src/main/resources/auth/jwt.key.
- If custom Public key not provided, default will be loaded from Jar bundled resource here-naksha-app-service/src/main/resources/auth/jwt.pub.
Sample commands to generate the custom key files:
# Generate private key
openssl genrsa -out ./custom_rsa256.key 2048
# Generate public key (using above private key)
openssl rsa -in ./custom_rsa256.key -pubout -outform PEM -out ./custom_rsa256.pub
Start using the service by creating a space:
curl -H "content-type:application/json" \
-d '{"id": "test-space", "title": "my first space", "description": "my first geodata repo"}' \
"http://localhost:8080/hub/spaces"
The service will respond with the space definition including the space ID (should you not specify an own id
):
{
"id": "test-space",
"title": "my first space",
"description": "my first geodata repo",
"storage": {
"id": "psql",
"params": null
},
"owner": "ANONYMOUS",
"createdAt": 1576601166681,
"updatedAt": 1576601166681,
"contentUpdatedAt": 1576601166681,
"autoCacheProfile": {
"browserTTL": 0,
"cdnTTL": 0,
"serviceTTL": 0
}
}
You can now add features to your brand new space:
curl -H "content-type:application/geo+json" -d '{"type":"FeatureCollection","features":[{"type":"Feature","geometry":{"type":"Point","coordinates":[-2.960847,53.430828]},"properties":{"name":"Anfield","@ns:com:here:xyz":{"tags":["football","stadium"]},"amenity":"Football Stadium","capacity":54074,"description":"Home of Liverpool Football Club"}}]}' http://localhost:8080/hub/spaces/pvhQepar/features
The service will respond with the inserted geo features:
{
"type": "FeatureCollection",
"etag": "b67016e5dcabbd5f76b0719d75c84424",
"features": [
{
"type": "Feature",
"id": "nf36KMsQAUYoM5kM",
"geometry": {
"type": "Point",
"coordinates": [ -2.960847, 53.430828 ]
},
"properties": {
"@ns:com:here:xyz": {
"space": "pvhQepar",
"createdAt": 1576602412218,
"updatedAt": 1576602412218,
"tags": [ "football", "stadium" ]
},
"amenity": "Football Stadium",
"name": "Anfield",
"description": "Home of Liverpool Football Club",
"capacity": 54074
}
}
],
"inserted": [
"nf36KMsQAUYoM5kM"
]
}
To run tests locally run Gradle test
task:
./gradlew test
Code coverage report is generated with use of jacoco
To generate subproject level coverage, use Gradle task jacocoTestReport
:
./gradlew test jacocoTestReport
Outputs for each subproject will be stored in /[module]/build/reports/jacoco/test/html/index.html
To generate root level aggregated coverage, use additional Gradle task testCodeCoverageReport
:
./gradlew test jacocoTestReport testCodeCoverageReport
Outputs will be stored in /build/reports/jacoco/testCodeCoverageReport/html/index.html
To validate test coverage, run jacocoTestCoverageVerification
Gradle task:
./gradlew test jacocoTestReport jacocoTestCoverageVerification
XYZ Hub uses:
and others.
Your contributions are always welcome! Please have a look at the contribution guidelines first.
Copyright (C) 2017-2022 HERE Europe B.V.
This project is licensed under the Apache License, Version 2.0 - see the LICENSE file for details.