diff --git a/docs/working-with-linked-data.md b/docs/working-with-linked-data.md index 18b7010..f677115 100644 --- a/docs/working-with-linked-data.md +++ b/docs/working-with-linked-data.md @@ -42,14 +42,14 @@ referencing a context file, and this programmatic definition should guarantee th Three basic data access scenarios for the supermarket are defined below: -- Reading Data - e.g. Give me all the data for the **Building** entity `urn:ngsi-ld:Building:store001`. +- Reading Data - e.g. Give me all the data for the **Building** entity `urn:ngsi-ld:Building:store001` - Aggregation - e.g. Combine the **Products** entities sold in **Building** `urn:ngsi-ld:Building:store001` and - display the goods for sale. + display the goods for sale - Altering context within the system - e.g. Make a sale of a product: - - Update the daily sales records by the price of the **Product**. - - decrement the `numberOfItems` of the **Shelf** entity. - - Create a new Transaction Log record showing the sale has occurred. - - Raise an alert in the warehouse if less than 10 objects remain on sale. + - Update the daily sales records by the price of the **Product** + - decrement the `numberOfItems` of the **Shelf** entity + - Create a new Transaction Log record showing the sale has occurred + - Raise an alert in the warehouse if less than 10 objects remain on sale - etc. Further, advanced scenarios will be covered in later tutorials. @@ -106,14 +106,15 @@ proxy has also been added. To visualize and interact with the Context we will ad Therefore, the architecture will consist of three elements: -- The [Orion Context Broker](https://fiware-orion.readthedocs.io/en/latest/) which will receive requests using - [NGSI-LD](https://forge.etsi.org/swagger/ui/?url=https://forge.etsi.org/rep/NGSI-LD/NGSI-LD/raw/master/spec/updated/generated/full_api.json). +- The [Context Broker](https://fiware-orion.readthedocs.io/en/latest/) which will receive requests using + [NGSI-LD](https://forge.etsi.org/swagger/ui/?url=https://forge.etsi.org/rep/NGSI-LD/NGSI-LD/raw/master/spec/updated/generated/full_api.json) +- An HTTP **Web-Server** which offers static `@context` files defining the context entities within the system for both English and Japanese Users. - The underlying [MongoDB](https://www.mongodb.com/) database : - Used by the Orion Context Broker to hold context data information such as data entities, subscriptions and - registrations. + registrations - The **Stock Management Frontend** which will: - - Display store information. - - Show which products can be bought at each store. + - Display store information + - Show which products can be bought at each store - Allow users to "buy" products and reduce the stock count. Since all interactions between the elements are initiated by HTTP requests, the entities can be containerized and run @@ -161,13 +162,14 @@ The configuration information for MongoDB and the Orion Context Broker has been ## Start Up All services can be initialised from the command-line by running the -[services](https://github.com/FIWARE/tutorials.Relationships-Linked-Data/blob/master/services) Bash script provided +[services](https://github.com/FIWARE/tutorials.Relationships-Linked-Data/blob/NGSI-v2/services) Bash script provided within the repository. Please clone the repository and create the necessary images by running the commands as shown: ```bash #!/bin/bash git clone https://github.com/FIWARE/tutorials.Working-with-Linked-Data.git cd tutorials.Working-with-Linked-Data +git checkout NGSI-v2 ./services orion ``` @@ -197,14 +199,14 @@ The code under discussion can be found within the `ngsi-ld/store` controller in As usual, the code for HTTP access can be split out from the business logic of the Supermarket application itself. The lower level calls have been placed into a library file, which simplifies the codebase. This needs to be included in the header of the file as shown. Some constants are also required - for the Supermarket data, the `LinkHeader` is used to -define location of the data models JSON-LD context as -`https://fiware.github.io/tutorials.Step-by-Step/tutorials-context.jsonld`. +define location of the data models JSON-LD context (which) as +`http://context/ngsi-context.jsonld`. ```javascript const ngsiLD = require('../../lib/ngsi-ld'); const LinkHeader = - '; rel="http://www.w3.org/ns/json-ld#context"; type="application/ld+json">'; + '; rel="http://www.w3.org/ns/json-ld#context"; type="application/ld+json">'; ``` ### Retrieve a known Store @@ -270,8 +272,8 @@ The equivalent cUrl statement can be seen below: ```bash curl -G -X GET 'http://localhost:1026/ngsi-ld/v1/entities/urn:ngsi-ld:Building:store001/' \ --H 'Link: ; rel="http://www.w3.org/ns/json-ld#context"; type="application/ld+json"' \ --H 'Accept: application/ld+json' \ +-H 'Link: ; rel="http://www.w3.org/ns/json-ld#context"; type="application/ld+json"' \ +-H 'Content-Type: application/json' \ -d 'options=keyValues' ``` @@ -279,11 +281,13 @@ And the response from the broker is: ```json { - "@context": "https://fiware.github.io/tutorials.Step-by-Step/tutorials-context.jsonld", + "@context": "http://context/ngsi-context.jsonld", "id": "urn:ngsi-ld:Building:store001", "type": "Building", "furniture": ["urn:ngsi-ld:Shelf:unit001", "urn:ngsi-ld:Shelf:unit002", "urn:ngsi-ld:Shelf:unit003"], - "category": "commercial", + "category": { + "vocab": "commercial" + }, "address": { "streetAddress": "Bornholmer Straße 65", "addressRegion": "Berlin", @@ -340,8 +344,8 @@ The equivalent cUrl statement can be seen below: ```bash curl -G -X GET 'http://localhost:1026/ngsi-ld/v1/entities/urn:ngsi-ld:Building:store001/' \ --H 'Link: ; rel="http://www.w3.org/ns/json-ld#context"; type="application/ld+json"' \ --H 'Accept: application/ld+json' \ +-H 'Link: ; rel="http://www.w3.org/ns/json-ld#context"; type="application/ld+json"' \ +-H 'Content-Type: application/json' \ -d 'options=keyValues' \ -d 'attrs=furniture' ``` @@ -350,7 +354,7 @@ And the response from the broker is: ```json { - "@context": "https://fiware.github.io/tutorials.Step-by-Step/tutorials-context.jsonld", + "@context": "https://context/ngsi-context.jsonld", "id": "urn:ngsi-ld:Building:store001", "type": "Building", "furniture": ["urn:ngsi-ld:Shelf:unit001", "urn:ngsi-ld:Shelf:unit002", "urn:ngsi-ld:Shelf:unit003"] @@ -394,8 +398,9 @@ The equivalent cUrl statement can be seen below: ```bash curl -G -X GET 'http://localhost:1026/ngsi-ld/v1/entities/' \ --H 'Link: ; rel="http://www.w3.org/ns/json-ld#context"; type="application/ld+json"' \ --H 'Accept: application/ld+json' \ +-H 'Link: ; rel="http://www.w3.org/ns/json-ld#context"; type="application/ld+json"' \ +-H 'Content-Type: application/json' \ +-H 'Accept: application/json' \ -d 'type=Shelf' \ -d 'options=keyValues' \ -d 'attrs=stocks,numberOfItems' \ @@ -408,21 +413,21 @@ further. ```json [ { - "@context": "https://fiware.github.io/tutorials.Step-by-Step/tutorials-context.jsonld", + "@context": "http://context/ngsi-context.jsonld", "id": "urn:ngsi-ld:Shelf:unit001", "type": "Shelf", "numberOfItems": 15, "stocks": "urn:ngsi-ld:Product:001" }, { - "@context": "https://fiware.github.io/tutorials.Step-by-Step/tutorials-context.jsonld", + "@context": "http://context/ngsi-context.jsonld", "id": "urn:ngsi-ld:Shelf:unit002", "type": "Shelf", "numberOfItems": 15, "stocks": "urn:ngsi-ld:Product:003" }, { - "@context": "https://fiware.github.io/tutorials.Step-by-Step/tutorials-context.jsonld", + "@context": "http://context/ngsi-context.jsonld", "id": "urn:ngsi-ld:Shelf:unit003", "type": "Shelf", "numberOfItems": 15, @@ -465,7 +470,8 @@ The equivalent cUrl statement can be seen below: ```bash curl -G -X GET 'http://localhost:1026/ngsi-ld/v1/entities/' \ --H 'Link: ; rel="http://www.w3.org/ns/json-ld#context"; type="application/ld+json"' \ +-H 'Link: ; rel="http://www.w3.org/ns/json-ld#context"; type="application/ld+json"' \ +-H 'Content-Type: application/json' \ -H 'Accept: application/json' \ -d 'type=Product' \ -d 'options=keyValues' \ @@ -524,7 +530,8 @@ The equivalent cUrl statement can be seen below: ```bash curl -G -X GET 'http://localhost:1026/ngsi-ld/v1/entities/' \ --H 'Link: ; rel="http://www.w3.org/ns/json-ld#context"; type="application/ld+json"' \ +-H 'Link: ; rel="http://www.w3.org/ns/json-ld#context"; type="application/ld+json"' \ +-H 'Content-Type: application/json' \ -H 'Accept: application/json' \ -d 'type=Shelf' \ -d 'options=keyValues' \ @@ -604,7 +611,7 @@ The equivalent cUrl statement can be seen below: ```bash curl -X PATCH 'http://localhost:1026/ngsi-ld/v1/entities/urn:ngsi-ld:Shelf:unit001/attrs' \ --H 'Link: ; rel="http://www.w3.org/ns/json-ld#context"; type="application/ld+json"' \ +-H 'Link: ; rel="http://www.w3.org/ns/json-ld#context"; type="application/ld+json"' \ -H 'Content-Type: application/json' \ -d '{ "numberOfItems": { "type": "Property", "value": 10 } }' ``` @@ -621,16 +628,16 @@ As a demonstration of this, imagine we which to incorporate context data entitie using a different schema. Rather than using `name`, `category`, `location` etc, our Japanese context provider is using data attributes based on Kanji characters. -The core NGSI-LD `@context` defines that `name` = `https://uri.etsi.org/ngsi-ld/name`, similarly we can define `名前` = -`https://uri.etsi.org/ngsi-ld/name` and introduce alternate mappings for attribute names and enumerated values. +The English user NGSI-LD `@context` defines that `name` = `https://schema.org/name`, similarly for a Japanese user we can define `名前` = +`https://schema.org/name` and introduce alternate mappings for attribute names and enumerated values. Provided that two systems can agree upon a **common** system of unique URIs for data interchange, they are free to locally re-interpret those values within their own domain. ### Creating an Entity using an Alternate Schema -An alternative Japanese JSON-LD `@context` file has been created and published to an external server. The file can be -found here: `https://fiware.github.io/tutorials.Step-by-Step/japanese-context.jsonld`. Alternate data mappings can be +An alternative Japanese JSON-LD `@context` file has been created and published to a web server. Within the docker network, the file can be +found here: [`http://context/japanese-user-context.jsonld`](./data-models/japanese-user-context.jsonld). Alternate data mappings can be found for all attribute names used within the tutorials. > **Note**: For comparision the standard tutorial JSON-LD `@context` file can be found here: @@ -655,7 +662,10 @@ curl -L -X POST 'http://localhost:1026/ngsi-ld/v1/entities/' \ --data-raw '{ "id": "urn:ngsi-ld:Building:store005", "type": "ビル", - "カテゴリー": {"type": "Property", "value": ["コマーシャル"]}, + "カテゴリー": { + "type": "VocabularyProperty", + "vocab": ["コマーシャル"] + }, "住所": { "type": "Property", "value": { @@ -667,17 +677,23 @@ curl -L -X POST 'http://localhost:1026/ngsi-ld/v1/entities/' \ }, "場所": { "type": "GeoProperty", - "value": {"type": "Point","coordinates": [13.5646, 52.5435]} + "value": { + "type": "Point", + "coordinates": [13.5646, 52.5435] + } }, - "名前": {"type": "Property","value": "Yuusui-en"}, - "@context":"https://fiware.github.io/tutorials.Step-by-Step/japanese-context.jsonld" + "名前": { + "type": "Property", + "value": "Yuusui-en" + }, + "@context":"http://context/japanese-user-context.jsonld" }' ``` And the response is an HTTP 201 status code with no data. -Note that in this example the name and address have been supplied as simple strings - JSON-LD does support an `@lang` -definition to allow for internationalization, but this is an advanced topic which will not be discussed here. +Note that in this example the name and address have been supplied as simple strings - JSON-LD does support a **LanguageProperty** `@lang` +definition to allow for internationalization, but this is an advanced topic which will not be discussed here. Note that the `category`/ `カテゴリー` has been defined using a **VocabularyProperty**, so that the enumerated value `commercial`/ `"コマーシャル` can also be amended via the `@context`. ### Reading an Entity using the default schema @@ -685,16 +701,16 @@ Within the context broker the full URIs are used to refer to the attributes and different attribute short names, the Japanese JSON-LD `@context` file agrees with the standard tutorial context about the full URIs used for a **Building** entity - effectively it is using the same data model. -Therefore, it is possible to request the new **Building** (created using the Japanese data model) and have it return -using the short names specified in the standard tutorial JSON-LD `@context`, this is done by supplying the `Link` header +Therefore it is possible to request the new **Building** (created using the Japanese data model) and have it return +using the short names specified in the standard English User JSON-LD `@context`, this is done by supplying the `Link` header is pointing to the tutorial JSON-LD `@context` file. -#### 2 Request: +#### 2️⃣ Request: -```bash +```console curl -L -X GET 'http://localhost:1026/ngsi-ld/v1/entities/urn:ngsi-ld:Building:store005' \ --H 'Accept: application/ld+json' \ --H 'Link: ; rel="http://www.w3.org/ns/json-ld#context"; type="application/ld+json"' +-H 'Content-Type: application/json' \ +-H 'Link: ; rel="http://www.w3.org/ns/json-ld#context"; type="application/ld+json"' ``` #### Response: @@ -711,7 +727,7 @@ also returns the standard enumeration for **Building** `category`. ```json { - "@context": "https://fiware.github.io/tutorials.Step-by-Step/tutorials-context.jsonld", + "@context": "http://context/ngsi-context.jsonld", "id": "urn:ngsi-ld:Building:store005", "type": "Building", "address": { @@ -728,7 +744,7 @@ also returns the standard enumeration for **Building** `category`. "value": { "type": "Point", "coordinates": [13.5646, 52.5435] } }, "name": { "type": "Property", "value": "Yuusui-en" }, - "category": { "type": "Property", "value": "commercial" } + "category": { "type": "VocabularyProperty", "vocab": "commercial" } } ``` @@ -745,29 +761,30 @@ With one exception, there is no hierarchy defined within NGSI-LD `@context` file possible to read any of the existing **Building** entities and apply the Japanese `@context`. The `@context` to used is supplied in the `Link` header. -#### 3 Request: +#### 3️⃣ Request: -```bash +```console curl -L -X GET 'http://localhost:1026/ngsi-ld/v1/entities/urn:ngsi-ld:Building:store003' \ --H 'Accept: application/ld+json' \ --H 'Link: ; rel="http://www.w3.org/ns/json-ld#context"; type="application/ld+json"' +-H 'Content-Type: application/json' \ +-H 'Link: ; rel="http://www.w3.org/ns/json-ld#context"; type="application/ld+json"' ``` #### Response: The response is mixed - it uses attribute names and enumerations defined in `japanese-context.jsonld` with some exceptions. NGSI-LD **is not** JSON-LD, in that the core context is always applied after the contexts received in the -`Link` header. Since `name` and `location` are reserved attribute names, they are supplied using the default core +`Link` header. Since `location` is a reserved attribute name, it is always supplied using the default core context. ```json { - "@context": "https://fiware.github.io/tutorials.Step-by-Step/japanese-context.jsonld", "id": "urn:ngsi-ld:Building:store003", "type": "ビル", - "家具": { - "type": "Relationship", - "object": ["urn:ngsi-ld:Shelf:unit006", "urn:ngsi-ld:Shelf:unit007", "urn:ngsi-ld:Shelf:unit008"] + "カテゴリー": { + "type": "VocabularyProperty", + "vocab": [ + "コマーシャル" + ] }, "住所": { "type": "Property", @@ -777,13 +794,32 @@ context. "addressLocality": "Friedrichshain", "postalCode": "10243" }, - "検証済み": { "type": "Property", "value": false } + "検証済み": { + "type": "Property", + "value": false + } }, - "name": { "type": "Property", "value": "East Side Galleria" }, - "カテゴリー": { "type": "Property", "value": "コマーシャル" }, "location": { "type": "GeoProperty", - "value": { "type": "Point", "coordinates": [13.4447, 52.5031] } + "value": { + "type": "Point", + "coordinates": [ + 13.4447, + 52.5031 + ] + } + }, + "名前": { + "type": "Property", + "value": "East Side Galleria" + }, + "家具": { + "type": "Relationship", + "object": [ + "urn:ngsi-ld:Shelf:unit006", + "urn:ngsi-ld:Shelf:unit007", + "urn:ngsi-ld:Shelf:unit008" + ] } } ``` @@ -824,21 +860,20 @@ function translateRequest(req, res) { } ``` -#### 4 Request: +#### 4️⃣ Request: A `/japanese` endpoint has been created which forwards a request to the context broker and then applies an expansion/compaction operation. -```bash +```console curl -L -X GET 'http://localhost:3000/japanese/ngsi-ld/v1/entities/urn:ngsi-ld:Building:store005' \ --H 'Accept: application/json' \ --H 'Link: ; rel="http://www.w3.org/ns/json-ld#context"; type="application/ld+json"' +-H 'Accept: application/ld+json' ``` #### Response: -The response after the expansion/compaction operation is data which now uses all the preferred attribute names - this is -**no longer** valid NGSI-LD, but would be of use if the receiving system requests data in this format. +The response after the expansion/compaction operation is data which now uses all of the preferred attribute names - this +is **no longer** valid NGSI-LD, but would be of use if the receiving system requests data in this format. Note that the reverse expansion/compaction operation could be used to convert this JSON back into a valid NGSI-LD payload before sending data to the context broker. @@ -847,24 +882,56 @@ payload before sending data to the context broker. { "識別子": "urn:ngsi-ld:Building:store005", "タイプ": "ビル", - "カテゴリー": { "タイプ": "プロパティ", "値": "コマーシャル" }, "住所": { - "タイプ": "プロパティ", - "値": { - "addressLocality": "Marzahn", - "addressRegion": "Berlin", - "postalCode": "12685", - "streetAddress": "Eisenacher Straße 98" - } + "addressLocality": "Marzahn", + "addressRegion": "Berlin", + "postalCode": "12685", + "streetAddress": "Eisenacher Straße 98" }, + "名前": "Yuusui-en", "場所": { - "タイプ": "ジオプロパティ", - "値": { "タイプ": "Point", "座標": [13.5646, 52.5435] } + "タイプ": "場", + "座標": [ + 13.5646, + 52.5435 + ] }, - "名前": { "タイプ": "プロパティ", "値": "Yuusui-en" } + "カテゴリー": { + "語彙": "コマーシャル" + } } ``` + +It is worth noting that the subattributes of `address` / `住所"` were left undefined by the `@context` files and therefore have remained unconverted. Both `address` and `住所"` are mapped to the same URI `http://schema.org/address`, it would have been possible to +include additional mappings for the subattributes using the schema.org ontology as shown: + +```json +{ + "@context": { + "addressLocality": "http://schema.org/addressLocality", + "addressRegion": "http://schema.org/addressRegion", + "postalCode": "http://schema.org/postalCode", + "streetAddress": "http://schema.org/streetAddress" + } +} + +``` + +and their equivalents in Japanese: + +```json +{ + "@context": { + "地方": "http://schema.org/addressLocality", + "地域": "http://schema.org/addressRegion", + "郵便番号": "http://schema.org/postalCode", + "住所": "http://schema.org/streetAddress" + } +} + +``` + #### Video: JSON-LD Compaction & Expansion [![](https://fiware.github.io/tutorials.Step-by-Step/img/video-logo.png)](https://www.youtube.com/watch?v=Tm3fD89dqRE 'JSON-LD Compaction & Expansion') diff --git a/tutorials.Working-with-Linked-Data b/tutorials.Working-with-Linked-Data index cecb1f9..c9a8a1a 160000 --- a/tutorials.Working-with-Linked-Data +++ b/tutorials.Working-with-Linked-Data @@ -1 +1 @@ -Subproject commit cecb1f920f14eee76dc42185069ecccaf6c37f7f +Subproject commit c9a8a1af583776ded623baabd701379d0cd51b7c