npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2024 – Pkg Stats / Ryan Hefner

@esdl/service

v0.1.7

Published

The main ESDL (Energy System Description Language) service entry point, offering services to query and retrieve ESDL data.

Downloads

2

Readme

@esdl/service

The @esdl/service is a micro-service that can serve ESDL (Energy System Description Language) Energy Systems from a database (using the ``@esdl/ormmodule, where ORM stands for Object Relational Mapping) or CSV file (using the@esdl/csvmodule). Using configuration over convention, a savvy non-developer should also be able to prepare his data so it can be served in a Docker container. The micro-service allows you to query the data via a REST (OpenAPI or Swagger) interface, available viahttp://HOST:PORT/swagger.json. Currently, there are three endpoints at http://HOST:PORT/api`:

  1. Query the ESDL in a certain region, e.g. /district/:id
  2. Query the ESDL in a certain region, and aggregate the data over a sub-region, e.g. /district/neighbourhood/:id.
  3. Get the capabilities of the service /capabilities

This also requires the exposed data to contain a geographic reference, for example a coordinate or the name of the area. In the following, you will find a description of how to make your energy data CSV available as ESDL.

Installing prerequisites

The easiest way is to fork and clone this repository, install all dependencies and build the code.

From the directory where you cloned the repository, run the commands from the build instructions listed here (repeated here for convenience)

npm i -g yarn # If yarn isn't installed already
npm i # On linux, you may run into permission issues when registering the cli package.

Then run the following command twice (!!!), to solve some dependency issues:

npm run dev

Exposing a CSV file as ESDL service

Assuming you have a CSV file that contains energy related information and you want to expose it as ESDL, you would need to:

  • Configure the ESDL service
  • Optionally, reformat the CSV a bit so it can be more easily consumed
  • Hosting the ESDL service

Configuring the ESDL service

In the project's config folder, you can find many examples of existing sources that can serve as a baseline for your new service. As a first example, I'll use zonnestroom.json, and explain what you need to do to re-use it and expose your solar power CSV.

Some example data is presented in the table below. Please note that the source represents geographical location in the naam column.

|naam|type|aantal_installaties|vermogen|woningen|naam_w|bedrijven|naam_b| |-------|-------|-------|-------|-------|-------|-------|-------| |Nederland|country|549505|2807377000|1602312|PV op woningen|1205066|PV op bedrijven| |Drenthe|province|31273|160683000|97130|PV op woningen|63553|PV op bedrijven|

  1. In the config folder, clone the zonnestroom.json to a new file, e.g. solarpower.json
  2. In solarpower.json, replace the serviceName property with solarpower.
  3. As we are dealing with a CSV file, the pluginService is @esdl/csv. Alternatively, there is also a @esdl/orm plugin for exposing data from a database.
  4. The regionsBaseUrl should point to a micro-service that exposes regional data, containing the boundaries of areas (municipalities, districts, etc.).
  5. In the rest property, you specify the name of your service, and the port that you wish to expose.
  6. In the csv section, you first specify the
  7. filename: path to the file
  8. delimiter: e.g. "," or ";"
  9. spatialType: can be 'point' (default) or 'name'. In the former case, the data source should have a point location, otherwise an area's name is used. In the latter case, you also need to specify the following two properties. Internally, the following check is performed when filtering the data: item[spatialColumn] === name && item[spatialTypeColumn] === regionType. This is needed in order to separate, for example, the city of Groningen from the province of Groningen.
  10. spatialTypeColumn: column name that contains the spatial type, e.g. province, municipality, district, or neighbourhood. The values need to be represented in the regions micro-service mentioned above.
  11. spatialColumn: column name that contains the name of the area.
  12. Next, you define the ESDL entities that you want to expose. Note that you can have more than one entity for one CSV file.
  13. name: The ESDL entity it represents, such as a WindTurbine, AggregatedBuilding or, in this case, PVPanel.
  14. description: A description
  15. columns: For each entry, the key represents the name of the ESDL attribute, so columns.name: { "name": "naam_w", "type": "string"} indicates that the CSV column naam_w will be exposed in ESDL as name. E.g. <PVPanel name="..."></PVPanel>. In case the name attribute is missing, it is supposed that the column name is the same.

Object and column transformers

|VID|Turbine|Gemeente|Fabrikant|Provincie|Type|Latitude|Ashoogte|Longitude|Diameter|Startjaar|Vermogen (kW)|Windpark| |----|----|----|----|----|----|----|----|----|----|----|----|----| |36241|Loppersum solitair|Loppersum|Lagerwey|Groningen|Onbekend|53.331546|24|6.635759|11|1982|15| |36242|Visafslag|De Marne|Lagerwey|Groningen|LW 15/75|53.407216|25|6.203135|16|1988|75|

As another example, consider windstats.json: this source contains the location of wind turbines as a latitude|longitude pair. For that reason, the entities property contains an optional property, transformer, referring to a LatLonPointTransformer which is an object transformer that transforms the latitude lat and longitude lon columns to an ESDL point.

In addition, there are also column transformers present, such as the StringTransformer, which transforms an integer to a string, which is required for ESDL id attributes.

"id": {
  "type": "int",
  "name": "VID",
  "transformer": "StringTransformer"
},

Available transformers:

  • StringTransformer (column transformer): transforms a number to a string.
  • NumberTransformer (column transformer): transforms a string to a number.
  • GeoJSONTransformer (column transformer): transforms a GeoJSON object to a string using JSON.stringify.
  • LatLonPointTransformer (object transformer): transforms a lat, lon property to a GeoJSON Point. It is assumed that lat and lon are in WGS84. The original lat and lon property are removed from the output.

Starting a new service

Locally

The commands below should be run from the ./packages/esdl-service directory.

To test whether your service is working, run it locally:

ESDL_CONFIG=./config/YOUR_SERVICE_NAME.json npm run start:prod

Or on Windows:

set ESDL_CONFIG=config\YOUR_SERVICE_NAME.json
npm run start:prod
node dist\main.js

This will transpile the TypeScript code into JavaScript, and run node dist/main.js. Next, you should be able to access your data from http://localhost:PORT/api, where PORT is the port that you configured in the JSON file.

Docker

When everything is working as expected, you can create a Docker image and, optionally, host it. In the below example, replace YOUR_NAME, YOUR_SERVICE_NAME and YOUR_PORT with the configured values in the config file. In case your CSV file is not stored in the data folder, you need to adjust the source location as well. See the ../docker folder for other examples.

FROM timbru31/java-node as builder
COPY . .
RUN npm install typescript json-schema-to-typescript jest -g
RUN yarn
RUN yarn build

FROM node:8-slim as production
LABEL Author="YOUR_NAME"

WORKDIR .

# Copy ALL node_modules
COPY --from=builder node_modules node_modules
# Copy linked node_modules
COPY --from=builder ./packages/esdl-core/dist node_modules/@esdl/core/dist
COPY --from=builder ./packages/esdl-core/mappings node_modules/@esdl/core/mappings
COPY --from=builder ./packages/esdl-core/package.json node_modules/@esdl/core/package.json
COPY --from=builder ./packages/esdl-csv/dist node_modules/@esdl/csv/dist
COPY --from=builder ./packages/esdl-csv/package.json node_modules/@esdl/csv/package.json

COPY --from=builder ./packages/esdl-service/data data
COPY --from=builder ./packages/esdl-service/package.json ./
COPY --from=builder ./packages/esdl-service/config/YOUR_SERVICE_NAME.json ./config/YOUR_SERVICE_NAME.json
COPY --from=builder ./packages/esdl-service/dist dist

ENV NODE_ENV production
ENV ESDL_CONFIG ./config/YOUR_SERVICE_NAME.json

# Expose the REST interface (uses the same port as in the config file's REST section.)
EXPOSE YOUR_PORT
CMD [ "node", "dist/main.js" ]

Now you are ready to build a new Docker container image:

$ docker build -f ./docker/YOUR_SERVICE_NAME/Dockerfile -t YOUR_SERVICE_NAME .

$ docker run -it -p YOUR_PORT:YOUR_PORT YOUR_SERVICE_NAME bash

In case YOUR_PORT is not available on your machine, you need to map it to a different port, e.g. 1234, as in -p 1234:YOUR_PORT.

Docker Swarm

There is also an npm script available for running your image in Docker Swarm (you need to run it from the root folder of your cloned project). For example, if you want to expose YOUR_SERVICE_NAME, running on port 4567, on port 1234 in the Swarm, you need to run:

> SVC=YOUR_SERVICE_NAME EXTPORT=1234 INTPORT=4567 npm run docker
# For example
> SVC=zonnestroom EXTPORT=4004 INTPORT=4004 npm run docker
> SVC=windstats EXTPORT=4003 INTPORT=4003 npm run docker

NOTE: The first time, call npm run docker1, which will not try to kill an existing service (and fail, as it isn't there).