Compare commits
2 Commits
shared-db-
...
refactor/s
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
55e4c92a60 | ||
|
|
7738f85036 |
3
.github/dependabot.yml
vendored
3
.github/dependabot.yml
vendored
@@ -15,6 +15,3 @@ updates:
|
|||||||
- dependency-name: "chai"
|
- dependency-name: "chai"
|
||||||
# chai 5 doesn't work atm w/ cds.test, TODO fix that in cds.test
|
# chai 5 doesn't work atm w/ cds.test, TODO fix that in cds.test
|
||||||
versions: ["5.x"]
|
versions: ["5.x"]
|
||||||
- dependency-name: "chai-as-promised"
|
|
||||||
# chai-as-promised 8 doesn't work atm w/ cds.test, TODO fix that in cds.test
|
|
||||||
versions: ["8.x"]
|
|
||||||
|
|||||||
4
.github/workflows/node.js.yml
vendored
4
.github/workflows/node.js.yml
vendored
@@ -15,7 +15,7 @@ jobs:
|
|||||||
|
|
||||||
strategy:
|
strategy:
|
||||||
matrix:
|
matrix:
|
||||||
node-version: [22.x, 20.x, 18.x]
|
node-version: [20.x, 18.x]
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v4
|
||||||
@@ -33,6 +33,6 @@ jobs:
|
|||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v4
|
||||||
- uses: actions/setup-node@v4
|
- uses: actions/setup-node@v4
|
||||||
with:
|
with:
|
||||||
node-version: 22.x
|
node-version: 20.x
|
||||||
- run: npm ci
|
- run: npm ci
|
||||||
- run: npm run lint
|
- run: npm run lint
|
||||||
|
|||||||
@@ -24,6 +24,6 @@ Disclaimer: The code in this project may include calls to APIs (“API Calls”)
|
|||||||
you any rights to use or access any SAP External Product, or provide any third
|
you any rights to use or access any SAP External Product, or provide any third
|
||||||
parties the right to use of access any SAP External Product, through API Calls.
|
parties the right to use of access any SAP External Product, through API Calls.
|
||||||
|
|
||||||
Files: *.*
|
Files: *
|
||||||
Copyright: 2019-2025 SAP SE or an SAP affiliate company and cap-cloud-samples
|
Copyright: 2019-2020 SAP SE or an SAP affiliate company and cap-cloud-samples
|
||||||
License: Apache-2.0
|
License: Apache-2.0
|
||||||
|
|||||||
10
.vscode/settings.json
vendored
10
.vscode/settings.json
vendored
@@ -13,5 +13,13 @@
|
|||||||
"**/cds/lib/req/cds-context.js",
|
"**/cds/lib/req/cds-context.js",
|
||||||
"**/odata-v4/okra/**"
|
"**/odata-v4/okra/**"
|
||||||
]
|
]
|
||||||
}
|
},
|
||||||
|
"eslint.probe": [
|
||||||
|
"cds",
|
||||||
|
"csn",
|
||||||
|
"csv",
|
||||||
|
"csv (semicolon)",
|
||||||
|
"tsv",
|
||||||
|
"tab"
|
||||||
|
]
|
||||||
}
|
}
|
||||||
|
|||||||
14
README.md
14
README.md
@@ -1,5 +1,3 @@
|
|||||||
|
|
||||||
|
|
||||||
# Welcome to cap/samples
|
# Welcome to cap/samples
|
||||||
|
|
||||||
Find here a collection of samples for the [SAP Cloud Application Programming Model](https://cap.cloud.sap) organized in a simplistic [monorepo setup](samples.md#all-in-one-monorepo).
|
Find here a collection of samples for the [SAP Cloud Application Programming Model](https://cap.cloud.sap) organized in a simplistic [monorepo setup](samples.md#all-in-one-monorepo).
|
||||||
@@ -9,12 +7,20 @@ Find here a collection of samples for the [SAP Cloud Application Programming Mod
|
|||||||

|

|
||||||
|
|
||||||

|

|
||||||
|
[](https://api.reuse.software/info/github.com/SAP-samples/cloud-cap-samples)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
### Preliminaries
|
### Preliminaries
|
||||||
|
|
||||||
Ensure you have the latest LTS version of Node.js, [`@sap/cds-dk`](https://www.npmjs.com/package/@sap/cds-dk) installed globally, `git` and your IDE ready (see [Initial Setup](https://cap.cloud.sap/docs/get-started/#setup))
|
1. Ensure you have the latest LTS version of Node.js installed (see [Getting Started](https://cap.cloud.sap/docs/get-started/))
|
||||||
|
2. Install [**@sap/cds-dk**](https://cap.cloud.sap/docs/get-started/) globally:
|
||||||
|
|
||||||
|
```sh
|
||||||
|
npm i -g @sap/cds-dk
|
||||||
|
```
|
||||||
|
|
||||||
|
3. _Optional:_ [Use Visual Studio Code](https://cap.cloud.sap/docs/get-started/tools#vscode)
|
||||||
|
|
||||||
### Download
|
### Download
|
||||||
|
|
||||||
@@ -30,7 +36,7 @@ cd samples
|
|||||||
In the samples folder run:
|
In the samples folder run:
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
npm ci
|
npm install
|
||||||
```
|
```
|
||||||
|
|
||||||
### Run
|
### Run
|
||||||
|
|||||||
@@ -1 +0,0 @@
|
|||||||
../../bookshop/app/vue
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
../../orders/app/orders
|
|
||||||
2427
app/router/package-lock.json
generated
2427
app/router/package-lock.json
generated
File diff suppressed because it is too large
Load Diff
@@ -1,12 +0,0 @@
|
|||||||
{
|
|
||||||
"name": "approuter",
|
|
||||||
"dependencies": {
|
|
||||||
"@sap/approuter": "^19.0.0"
|
|
||||||
},
|
|
||||||
"engines": {
|
|
||||||
"node": "^20"
|
|
||||||
},
|
|
||||||
"scripts": {
|
|
||||||
"start": "node node_modules/@sap/approuter/approuter.js"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
../../reviews/app/vue
|
|
||||||
@@ -1,52 +0,0 @@
|
|||||||
{
|
|
||||||
"welcomeFile": "app/bookshop/index.html",
|
|
||||||
"routes": [
|
|
||||||
{
|
|
||||||
"source": "^/app/(.*)$",
|
|
||||||
"target": "$1",
|
|
||||||
"localDir": ".",
|
|
||||||
"cacheControl": "no-cache, no-store, must-revalidate"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"source": "^/appconfig/",
|
|
||||||
"localDir": ".",
|
|
||||||
"cacheControl": "no-cache, no-store, must-revalidate"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"source": "^/admin/(.*)$",
|
|
||||||
"target": "/admin/$1",
|
|
||||||
"destination": "bookstore-api",
|
|
||||||
"csrfProtection": true
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"source": "^/browse/(.*)$",
|
|
||||||
"target": "/browse/$1",
|
|
||||||
"destination": "bookstore-api",
|
|
||||||
"csrfProtection": true
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"source": "^/user/(.*)$",
|
|
||||||
"target": "/user/$1",
|
|
||||||
"destination": "bookstore-api",
|
|
||||||
"csrfProtection": true
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"source": "^/odata/v4/orders/(.*)$",
|
|
||||||
"target": "/odata/v4/orders/$1",
|
|
||||||
"destination": "orders-api",
|
|
||||||
"csrfProtection": true
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"source": "^/reviews/(.*)$",
|
|
||||||
"target": "/reviews/$1",
|
|
||||||
"destination": "reviews-api",
|
|
||||||
"csrfProtection": true
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"source": "^(.*)$",
|
|
||||||
"target": "$1",
|
|
||||||
"localDir": ".",
|
|
||||||
"cacheControl": "no-cache, no-store, must-revalidate"
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
@@ -19,7 +19,7 @@ const books = Vue.createApp ({
|
|||||||
search: ({target:{value:v}}) => books.fetch(v && '&$search='+v),
|
search: ({target:{value:v}}) => books.fetch(v && '&$search='+v),
|
||||||
|
|
||||||
async fetch (etc='') {
|
async fetch (etc='') {
|
||||||
const {data} = await GET(`/ListOfBooks?$expand=genre($select=name),currency($select=symbol)${etc}`)
|
const {data} = await GET(`/ListOfBooks?$expand=genre,currency${etc}`)
|
||||||
books.list = data.value
|
books.list = data.value
|
||||||
},
|
},
|
||||||
|
|
||||||
@@ -77,7 +77,7 @@ function csrfToken (request) {
|
|||||||
document.csrfToken = token
|
document.csrfToken = token
|
||||||
request.headers['x-csrf-token'] = document.csrfToken
|
request.headers['x-csrf-token'] = document.csrfToken
|
||||||
return request
|
return request
|
||||||
}).catch(() => {
|
}).catch(_ => {
|
||||||
document.csrfToken = null // set mark to not try again
|
document.csrfToken = null // set mark to not try again
|
||||||
return request
|
return request
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -1,5 +1,3 @@
|
|||||||
const cds = require('@sap/cds')
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* In order to keep basic bookshop sample as simple as possible, we don't add
|
* In order to keep basic bookshop sample as simple as possible, we don't add
|
||||||
* reuse dependencies. This db/init.js ensures we still have a minimum set of
|
* reuse dependencies. This db/init.js ensures we still have a minimum set of
|
||||||
@@ -8,7 +6,7 @@ const cds = require('@sap/cds')
|
|||||||
|
|
||||||
// NOTE: We use cds.on('served') to delay the UPSERTs after the db init
|
// NOTE: We use cds.on('served') to delay the UPSERTs after the db init
|
||||||
// to run after all INSERTs from .csv files happened.
|
// to run after all INSERTs from .csv files happened.
|
||||||
module.exports = cds.on('served', ()=>
|
module.exports = cds.on('served', ()=> cds.run(
|
||||||
UPSERT.into ('sap.common.Currencies') .columns (
|
UPSERT.into ('sap.common.Currencies') .columns (
|
||||||
[ 'code', 'symbol', 'name' ]
|
[ 'code', 'symbol', 'name' ]
|
||||||
) .rows (
|
) .rows (
|
||||||
@@ -18,4 +16,4 @@ module.exports = cds.on('served', ()=>
|
|||||||
[ 'ILS', '₪', 'Shekel' ],
|
[ 'ILS', '₪', 'Shekel' ],
|
||||||
[ 'JPY', '¥', 'Yen' ],
|
[ 'JPY', '¥', 'Yen' ],
|
||||||
)
|
)
|
||||||
)
|
))
|
||||||
|
|||||||
2
bookshop/index.js
Normal file
2
bookshop/index.js
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
const { CatalogService } = require('./srv/cat-service')
|
||||||
|
module.exports = { CatalogService }
|
||||||
@@ -1,14 +1,13 @@
|
|||||||
const cds = require('@sap/cds')
|
const cds = require('@sap/cds/lib')
|
||||||
|
|
||||||
module.exports = class AdminService extends cds.ApplicationService { init(){
|
module.exports = class AdminService extends cds.ApplicationService { init(){
|
||||||
this.before (['NEW','CREATE'],'Authors', genid)
|
this.before ('NEW','Authors', genid)
|
||||||
this.before (['NEW','CREATE'],'Books', genid)
|
this.before ('NEW','Books', genid)
|
||||||
return super.init()
|
return super.init()
|
||||||
}}
|
}}
|
||||||
|
|
||||||
/** Generate primary keys for target entity in request */
|
/** Generate primary keys for target entity in request */
|
||||||
async function genid (req) {
|
async function genid (req) {
|
||||||
if (req.data.ID) return
|
|
||||||
const {id} = await SELECT.one.from(req.target).columns('max(ID) as id')
|
const {id} = await SELECT.one.from(req.target).columns('max(ID) as id')
|
||||||
req.data.ID = id + 4 // Note: that is not safe! ok for this sample only.
|
req.data.ID = id - id % 100 + 100 + 1
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,4 @@
|
|||||||
const cds = require('@sap/cds')
|
module.exports = class CatalogService extends cds.ApplicationService { init() {
|
||||||
|
|
||||||
class CatalogService extends cds.ApplicationService { init() {
|
|
||||||
|
|
||||||
const { Books } = cds.entities('sap.capire.bookshop')
|
const { Books } = cds.entities('sap.capire.bookshop')
|
||||||
const { ListOfBooks } = this.entities
|
const { ListOfBooks } = this.entities
|
||||||
@@ -34,5 +32,3 @@ class CatalogService extends cds.ApplicationService { init() {
|
|||||||
// Delegate requests to the underlying generic service
|
// Delegate requests to the underlying generic service
|
||||||
return super.init()
|
return super.init()
|
||||||
}}
|
}}
|
||||||
|
|
||||||
module.exports = CatalogService
|
|
||||||
|
|||||||
@@ -3,15 +3,15 @@
|
|||||||
# Genres
|
# Genres
|
||||||
#
|
#
|
||||||
|
|
||||||
GET http://localhost:4004/odata/v4/test/Genres?
|
GET http://localhost:4004/test/Genres?
|
||||||
###
|
###
|
||||||
|
|
||||||
GET http://localhost:4004/odata/v4/test/Genres?
|
GET http://localhost:4004/test/Genres?
|
||||||
&$filter=parent_ID eq null&$select=name
|
&$filter=parent_ID eq null&$select=name
|
||||||
&$expand=children($select=name)
|
&$expand=children($select=name)
|
||||||
###
|
###
|
||||||
|
|
||||||
POST http://localhost:4004/odata/v4/test/Genres?
|
POST http://localhost:4004/test/Genres?
|
||||||
Content-Type: application/json
|
Content-Type: application/json
|
||||||
|
|
||||||
{ "ID":100, "name":"Some Sample Genres...", "children":[
|
{ "ID":100, "name":"Some Sample Genres...", "children":[
|
||||||
@@ -26,13 +26,13 @@ Content-Type: application/json
|
|||||||
]}
|
]}
|
||||||
###
|
###
|
||||||
|
|
||||||
GET http://localhost:4004/odata/v4/test/Genres(100)?
|
GET http://localhost:4004/test/Genres(100)?
|
||||||
# &$expand=children
|
# &$expand=children
|
||||||
# &$expand=children($expand=children($expand=children($expand=children)))
|
# &$expand=children($expand=children($expand=children($expand=children)))
|
||||||
###
|
###
|
||||||
|
|
||||||
DELETE http://localhost:4004/odata/v4/test/Genres(103)
|
DELETE http://localhost:4004/test/Genres(103)
|
||||||
###
|
###
|
||||||
|
|
||||||
DELETE http://localhost:4004/odata/v4/test/Genres(100)
|
DELETE http://localhost:4004/test/Genres(100)
|
||||||
###
|
###
|
||||||
|
|||||||
@@ -1,12 +0,0 @@
|
|||||||
Books = Bücher
|
|
||||||
Book = Buch
|
|
||||||
Title = Titel
|
|
||||||
Description = Beschreibung
|
|
||||||
Stock = Bestand
|
|
||||||
Image = Bild
|
|
||||||
Price = Preis
|
|
||||||
Currency = Währung
|
|
||||||
|
|
||||||
Authors = Autoren
|
|
||||||
Author = Autor
|
|
||||||
AuthorID = ID des Autors
|
|
||||||
@@ -1,21 +0,0 @@
|
|||||||
Books = Books
|
|
||||||
Book = Book
|
|
||||||
ID = ID
|
|
||||||
Title = Title
|
|
||||||
Description = Description
|
|
||||||
Stock = Stock
|
|
||||||
Image = Image
|
|
||||||
Price = Price
|
|
||||||
Currency = Currency
|
|
||||||
|
|
||||||
Authors = Authors
|
|
||||||
Author = Author
|
|
||||||
AuthorID = Author''s ID
|
|
||||||
Name = Name
|
|
||||||
DateOfBirth = Date of Birth
|
|
||||||
DateOfDeath = Date of Death
|
|
||||||
PlaceOfBirth = Place of Birth
|
|
||||||
PlaceOfDeath = Place of Death
|
|
||||||
|
|
||||||
Genres = Genres
|
|
||||||
Genre = Genre
|
|
||||||
@@ -1,20 +0,0 @@
|
|||||||
Books = Livres
|
|
||||||
Book = Livre
|
|
||||||
Title = Titre
|
|
||||||
Description = Description
|
|
||||||
Stock = Action
|
|
||||||
Image = Image
|
|
||||||
Price = Prix
|
|
||||||
Currency = Devise
|
|
||||||
|
|
||||||
Authors = Auteurs
|
|
||||||
Author = Auteur
|
|
||||||
AuthorID = ID de l''auteur
|
|
||||||
Name = Nom
|
|
||||||
DateOfBirth = Date de naissance
|
|
||||||
DateOfDeath = Date de décès
|
|
||||||
PlaceOfBirth = Lieu de naissance
|
|
||||||
PlaceOfDeath = Lieu de décès
|
|
||||||
|
|
||||||
Genres = Genre
|
|
||||||
Genre = Genre
|
|
||||||
@@ -1,34 +0,0 @@
|
|||||||
400 = Ungültige Anfrage
|
|
||||||
401 = Nicht autorisiert
|
|
||||||
403 = Verboten
|
|
||||||
404 = Nicht gefunden
|
|
||||||
405 = Methode nicht zulässig
|
|
||||||
406 = Nicht akzeptabel
|
|
||||||
407 = Proxy-Authentifizierung erforderlich
|
|
||||||
408 = Anfrage-Timeout
|
|
||||||
409 = Konflikt
|
|
||||||
410 = Weg
|
|
||||||
411 = Erforderliche Länge
|
|
||||||
412 = Vorbedingung fehlgeschlagen
|
|
||||||
413 = Nutzlast zu groß
|
|
||||||
414 = URI zu lang
|
|
||||||
415 = Nicht unterstützter Medientyp
|
|
||||||
416 = Bereich nicht erfüllbar
|
|
||||||
417 = Erwartung fehlgeschlagen
|
|
||||||
422 = Nicht verarbeitbarer Inhalt
|
|
||||||
424 = Fehlgeschlagene Abhängigkeit
|
|
||||||
428 = Vorbedingung erforderlich
|
|
||||||
429 = Zu viele Anfragen
|
|
||||||
431 = Anfrage-Headerfelder zu groß
|
|
||||||
451 = Aus rechtlichen Gründen nicht verfügbar
|
|
||||||
500 = Interner Serverfehler
|
|
||||||
501 = Der Server unterstützt die zur Erfüllung der Anfrage erforderliche Funktionalität nicht
|
|
||||||
502 = Ungültiges Gateway
|
|
||||||
503 = Dienst nicht verfügbar
|
|
||||||
504 = Gateway-Timeout
|
|
||||||
|
|
||||||
ASSERT_RANGE = Wert {0} liegt nicht im angegebenen Bereich [{1}, {2}]
|
|
||||||
ASSERT_FORMAT = Wert "{0}" liegt nicht im angegebenen Format "{1}"
|
|
||||||
ASSERT_ARRAY = Wert muss ein Array sein
|
|
||||||
ASSERT_ENUM = Wert {0} ist gemäß Enumerationsdeklaration {{1}} ungültig
|
|
||||||
ASSERT_NOT_NULL = Wert ist erforderlich
|
|
||||||
@@ -1,34 +0,0 @@
|
|||||||
400 = Bad Request
|
|
||||||
401 = Unauthorized
|
|
||||||
403 = Forbidden
|
|
||||||
404 = Not Found
|
|
||||||
405 = Method Not Allowed
|
|
||||||
406 = Not Acceptable
|
|
||||||
407 = Proxy Authentication Required
|
|
||||||
408 = Request Timeout
|
|
||||||
409 = Conflict
|
|
||||||
410 = Gone
|
|
||||||
411 = Length Required
|
|
||||||
412 = Precondition Failed
|
|
||||||
413 = Payload Too Large
|
|
||||||
414 = URI Too Long
|
|
||||||
415 = Unsupported Media Type
|
|
||||||
416 = Range Not Satisfiable
|
|
||||||
417 = Expectation Failed
|
|
||||||
422 = Unprocessable Content
|
|
||||||
424 = Failed Dependency
|
|
||||||
428 = Precondition Required
|
|
||||||
429 = Too Many Requests
|
|
||||||
431 = Request Header Fields Too Large
|
|
||||||
451 = Unavailable For Legal Reasons
|
|
||||||
500 = Internal Server Error
|
|
||||||
501 = The server does not support the functionality required to fulfill the request
|
|
||||||
502 = Bad Gateway
|
|
||||||
503 = Service Unavailable
|
|
||||||
504 = Gateway Timeout
|
|
||||||
|
|
||||||
ASSERT_RANGE = Value {0} is not in specified range [{1}, {2}]
|
|
||||||
ASSERT_FORMAT = Value "{0}" is not in specified format "{1}"
|
|
||||||
ASSERT_ARRAY = Value must be an array
|
|
||||||
ASSERT_ENUM = Value {0} is invalid according to enum declaration {{1}}
|
|
||||||
ASSERT_NOT_NULL = Value is required
|
|
||||||
@@ -1,34 +0,0 @@
|
|||||||
400 = Requête incorrecte
|
|
||||||
401 = Non autorisée
|
|
||||||
403 = Interdite
|
|
||||||
404 = Introuvable
|
|
||||||
405 = Méthode non autorisée
|
|
||||||
406 = Non acceptable
|
|
||||||
407 = Authentification proxy requise
|
|
||||||
408 = Délai d''expiration de la requête
|
|
||||||
409 = Conflit
|
|
||||||
410 = Disparu
|
|
||||||
411 = Longueur requise
|
|
||||||
412 = Échec de la condition préalable
|
|
||||||
413 = Charge utile trop importante
|
|
||||||
414 = URI trop longue
|
|
||||||
415 = Type de média non pris en charge
|
|
||||||
416 = Plage non satisfaisante
|
|
||||||
417 = Échec de l''attente
|
|
||||||
422 = Contenu non traitable
|
|
||||||
424 = Dépendance échouée
|
|
||||||
428 = Condition préalable requise
|
|
||||||
429 = Trop de requêtes
|
|
||||||
431 = Champs d''en-tête de requête trop importants
|
|
||||||
451 = Indisponible pour des raisons juridiques
|
|
||||||
500 = Erreur interne du serveur
|
|
||||||
501 = Le serveur ne prend pas en charge la fonctionnalité requise pour répondre à la requête
|
|
||||||
502 = Passerelle incorrecte
|
|
||||||
503 = Service indisponible
|
|
||||||
504 = Délai d''attente de la passerelle
|
|
||||||
|
|
||||||
ASSERT_RANGE = La valeur {0} n''est pas dans la plage spécifiée [{1}, {2}]
|
|
||||||
ASSERT_FORMAT = La valeur "{0}" n''est pas au format spécifié "{1}"
|
|
||||||
ASSERT_ARRAY = La valeur doit être un tableau
|
|
||||||
ASSERT_ENUM = La valeur {0} n''est pas valide selon la déclaration d''énumération {{1}}
|
|
||||||
ASSERT_NOT_NULL = La valeur est obligatoire
|
|
||||||
@@ -2,21 +2,14 @@
|
|||||||
"name": "@capire/bookstore",
|
"name": "@capire/bookstore",
|
||||||
"version": "1.0.0",
|
"version": "1.0.0",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@cap-js/hana": "^1.7.0",
|
|
||||||
"@capire/bookshop": "*",
|
"@capire/bookshop": "*",
|
||||||
|
"@capire/reviews": "*",
|
||||||
|
"@capire/orders": "*",
|
||||||
"@capire/common": "*",
|
"@capire/common": "*",
|
||||||
"@capire/data-viewer": "*",
|
"@capire/data-viewer": "*",
|
||||||
"@capire/orders": "*",
|
|
||||||
"@capire/reviews": "*",
|
|
||||||
"@sap-cloud-sdk/http-client": "^3.26.4",
|
|
||||||
"@sap-cloud-sdk/resilience": "^3.26.4",
|
|
||||||
"@sap/cds": ">=5",
|
"@sap/cds": ">=5",
|
||||||
"@sap/xssec": "^4.4.0",
|
|
||||||
"express": "^4.17.1"
|
"express": "^4.17.1"
|
||||||
},
|
},
|
||||||
"scripts": {
|
|
||||||
"start": "cds-serve"
|
|
||||||
},
|
|
||||||
"cds": {
|
"cds": {
|
||||||
"requires": {
|
"requires": {
|
||||||
"ReviewsService": {
|
"ReviewsService": {
|
||||||
@@ -27,13 +20,15 @@
|
|||||||
"kind": "odata",
|
"kind": "odata",
|
||||||
"model": "@capire/orders"
|
"model": "@capire/orders"
|
||||||
},
|
},
|
||||||
"messaging": true,
|
"messaging": {
|
||||||
|
"[development]": { "kind": "file-based-messaging" },
|
||||||
|
"[hybrid]": { "kind": "enterprise-messaging-shared" },
|
||||||
|
"[production]": { "kind": "enterprise-messaging" }
|
||||||
|
},
|
||||||
"db": {
|
"db": {
|
||||||
"kind": "sql"
|
"kind": "sql"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"log": {
|
"log": { "service": true }
|
||||||
"service": true
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -5,15 +5,10 @@ cds.once('served', require('./srv/mashup'))
|
|||||||
|
|
||||||
// Add routes to UIs from imported packages
|
// Add routes to UIs from imported packages
|
||||||
cds.once('bootstrap',(app)=>{
|
cds.once('bootstrap',(app)=>{
|
||||||
try {
|
|
||||||
app.serve ('/bookshop') .from ('@capire/bookshop','app/vue')
|
app.serve ('/bookshop') .from ('@capire/bookshop','app/vue')
|
||||||
app.serve ('/reviews') .from ('@capire/reviews','app/vue')
|
app.serve ('/reviews') .from ('@capire/reviews','app/vue')
|
||||||
app.serve ('/orders') .from('@capire/orders','app/orders')
|
app.serve ('/orders') .from('@capire/orders','app/orders')
|
||||||
app.serve ('/data') .from('@capire/data-viewer','app/viewer')
|
app.serve ('/data') .from('@capire/data-viewer','app/viewer')
|
||||||
} catch (err) {
|
|
||||||
if (err.code === 'MODULE_NOT_FOUND') throw new Error('Run "npm ci" to install the required dependencies', { cause: err })
|
|
||||||
throw err
|
|
||||||
}
|
|
||||||
})
|
})
|
||||||
|
|
||||||
// Add Swagger UI
|
// Add Swagger UI
|
||||||
|
|||||||
@@ -18,7 +18,7 @@ module.exports = async()=>{ // called by server.js
|
|||||||
// Note: prepend is neccessary to intercept generic default handler
|
// Note: prepend is neccessary to intercept generic default handler
|
||||||
//
|
//
|
||||||
CatalogService.prepend (srv => srv.on ('READ', 'Books/reviews', (req) => {
|
CatalogService.prepend (srv => srv.on ('READ', 'Books/reviews', (req) => {
|
||||||
console.debug ('> delegating request to ReviewsService') // eslint-disable-line no-console
|
console.debug ('> delegating request to ReviewsService')
|
||||||
const [id] = req.params, { columns, limit } = req.query.SELECT
|
const [id] = req.params, { columns, limit } = req.query.SELECT
|
||||||
return ReviewsService.read ('Reviews',columns).limit(limit).where({subject:String(id)})
|
return ReviewsService.read ('Reviews',columns).limit(limit).where({subject:String(id)})
|
||||||
}))
|
}))
|
||||||
@@ -28,8 +28,8 @@ module.exports = async()=>{ // called by server.js
|
|||||||
//
|
//
|
||||||
CatalogService.on ('OrderedBook', async (msg) => {
|
CatalogService.on ('OrderedBook', async (msg) => {
|
||||||
const { book, quantity, buyer } = msg.data
|
const { book, quantity, buyer } = msg.data
|
||||||
const { title, price } = await db.read (Books, book, b => { b.title, b.price })
|
const { title, price } = await db.tx(msg).read (Books, book, b => { b.title, b.price })
|
||||||
return OrdersService.create ('OrdersNoDraft').entries({
|
return OrdersService.tx(msg).create ('Orders').entries({
|
||||||
OrderNo: 'Order at '+ (new Date).toLocaleString(),
|
OrderNo: 'Order at '+ (new Date).toLocaleString(),
|
||||||
Items: [{ product:{ID:`${book}`}, title, price, quantity }],
|
Items: [{ product:{ID:`${book}`}, title, price, quantity }],
|
||||||
buyer, createdBy: buyer
|
buyer, createdBy: buyer
|
||||||
@@ -40,7 +40,7 @@ module.exports = async()=>{ // called by server.js
|
|||||||
// Update Books' average ratings when ReviewsService signals updated reviews
|
// Update Books' average ratings when ReviewsService signals updated reviews
|
||||||
//
|
//
|
||||||
ReviewsService.on ('reviewed', (msg) => {
|
ReviewsService.on ('reviewed', (msg) => {
|
||||||
console.debug ('> received:', msg.event, msg.data) // eslint-disable-line no-console
|
console.debug ('> received:', msg.event, msg.data)
|
||||||
const { subject, count, rating } = msg.data
|
const { subject, count, rating } = msg.data
|
||||||
return UPDATE(Books,subject).with({ numberOfReviews:count, rating })
|
return UPDATE(Books,subject).with({ numberOfReviews:count, rating })
|
||||||
})
|
})
|
||||||
@@ -49,7 +49,7 @@ module.exports = async()=>{ // called by server.js
|
|||||||
// Reduce stock of ordered books for orders are created from Orders admin UI
|
// Reduce stock of ordered books for orders are created from Orders admin UI
|
||||||
//
|
//
|
||||||
OrdersService.on ('OrderChanged', (msg) => {
|
OrdersService.on ('OrderChanged', (msg) => {
|
||||||
console.debug ('> received:', msg.event, msg.data) // eslint-disable-line no-console
|
console.debug ('> received:', msg.event, msg.data)
|
||||||
const { product, deltaQuantity } = msg.data
|
const { product, deltaQuantity } = msg.data
|
||||||
return UPDATE (Books) .where ('ID =', product)
|
return UPDATE (Books) .where ('ID =', product)
|
||||||
.and ('stock >=', deltaQuantity)
|
.and ('stock >=', deltaQuantity)
|
||||||
|
|||||||
@@ -12,12 +12,11 @@
|
|||||||
#
|
#
|
||||||
|
|
||||||
GET {{reviews-service}}/Reviews
|
GET {{reviews-service}}/Reviews
|
||||||
Authorization: Basic me:
|
|
||||||
|
|
||||||
###
|
###
|
||||||
|
|
||||||
POST {{reviews-service}}/Reviews
|
POST {{reviews-service}}/Reviews
|
||||||
Authorization: Basic me:
|
Authorization: Basic {{$processEnv USER}}:
|
||||||
Content-Type: application/json
|
Content-Type: application/json
|
||||||
|
|
||||||
{"subject":"201", "title":"boo", "rating":3 }
|
{"subject":"201", "title":"boo", "rating":3 }
|
||||||
@@ -42,7 +41,7 @@ GET {{bookshop}}/browse/Books(201)?
|
|||||||
###
|
###
|
||||||
|
|
||||||
GET {{bookshop}}/browse/Books?
|
GET {{bookshop}}/browse/Books?
|
||||||
&$select=title,author&$expand=currency
|
&$select=title,author&$expand=currency
|
||||||
Accept-Language: de
|
Accept-Language: de
|
||||||
|
|
||||||
#################################################
|
#################################################
|
||||||
@@ -51,23 +50,23 @@ Accept-Language: de
|
|||||||
#
|
#
|
||||||
@newOrderID = e939604c-ab83-4d4f-bdb6-95fe30b3773e
|
@newOrderID = e939604c-ab83-4d4f-bdb6-95fe30b3773e
|
||||||
|
|
||||||
GET {{bookshop}}/odata/v4/orders/Orders
|
GET {{bookshop}}/orders/Orders
|
||||||
|
|
||||||
### Create order, still inactive
|
### Create order, still inactive
|
||||||
POST {{bookshop}}/odata/v4/orders/Orders
|
POST {{bookshop}}/orders/Orders
|
||||||
Content-Type: application/json
|
Content-Type: application/json
|
||||||
|
|
||||||
{"ID": "{{newOrderID}}"}
|
{"ID": "{{newOrderID}}"}
|
||||||
|
|
||||||
### Get inactive order. We have to specify `IsActiveEntity`.
|
### Get inactive order. We have to specify `IsActiveEntity`.
|
||||||
GET {{bookshop}}/odata/v4/orders/Orders(ID={{newOrderID}},IsActiveEntity=false)
|
GET {{bookshop}}/orders/Orders(ID={{newOrderID}},IsActiveEntity=false)
|
||||||
|
|
||||||
### Activate order using `.../<servicename>.draftActivate`
|
### Activate order using `.../<servicename>.draftActivate`
|
||||||
POST {{bookshop}}/odata/v4/orders/Orders(ID={{newOrderID}},IsActiveEntity=false)/OrdersService.draftActivate
|
POST {{bookshop}}/orders/Orders(ID={{newOrderID}},IsActiveEntity=false)/OrdersService.draftActivate
|
||||||
Content-Type: application/json
|
Content-Type: application/json
|
||||||
|
|
||||||
### Get active order
|
### Get active order
|
||||||
GET {{bookshop}}/odata/v4/orders/Orders(ID={{newOrderID}},IsActiveEntity=true)
|
GET {{bookshop}}/orders/Orders(ID={{newOrderID}},IsActiveEntity=true)
|
||||||
|
|
||||||
### Create author
|
### Create author
|
||||||
POST {{bookshop}}/admin/Authors
|
POST {{bookshop}}/admin/Authors
|
||||||
|
|||||||
@@ -1 +0,0 @@
|
|||||||
// dummy to auto-load the plugin
|
|
||||||
@@ -4,12 +4,5 @@
|
|||||||
"version": "1.0.0",
|
"version": "1.0.0",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@sap/cds": "*"
|
"@sap/cds": "*"
|
||||||
},
|
|
||||||
"cds": {
|
|
||||||
"requires": {
|
|
||||||
"@capire/common/data": {
|
|
||||||
"model": "@capire/common"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -18,7 +18,7 @@ class DataService extends cds.ApplicationService { init(){
|
|||||||
.sort((e1, e2) => e1.name.localeCompare(e2.name))
|
.sort((e1, e2) => e1.name.localeCompare(e2.name))
|
||||||
.map(e => {
|
.map(e => {
|
||||||
const columns = Object.entries(e.elements)
|
const columns = Object.entries(e.elements)
|
||||||
.filter(([,el]) => !(el instanceof cds.Association)) // exclude assocs+compositions
|
.filter(([_, el]) => !(el instanceof cds.Association)) // exclude assocs+compositions
|
||||||
.map(([name, el]) => { return { name, type: el.type, isKey:!!el.key }})
|
.map(([name, el]) => { return { name, type: el.type, isKey:!!el.key }})
|
||||||
return { name: e.name, columns }
|
return { name: e.name, columns }
|
||||||
})
|
})
|
||||||
@@ -34,8 +34,7 @@ class DataService extends cds.ApplicationService { init(){
|
|||||||
query.SELECT.limit = req.query.SELECT.limit // forward $skip / $top
|
query.SELECT.limit = req.query.SELECT.limit // forward $skip / $top
|
||||||
|
|
||||||
const dataSource = findDataSource(dataSourceName, entityName)
|
const dataSource = findDataSource(dataSourceName, entityName)
|
||||||
let res = await dataSource.run(query)
|
const res = await dataSource.run(query)
|
||||||
if (!Array.isArray(res)) res = [res] // singleton result
|
|
||||||
return res.map((line) => {
|
return res.map((line) => {
|
||||||
const record = Object.entries(line).map(([column, data]) => ({ column, data }))
|
const record = Object.entries(line).map(([column, data]) => ({ column, data }))
|
||||||
return {
|
return {
|
||||||
@@ -50,7 +49,6 @@ class DataService extends cds.ApplicationService { init(){
|
|||||||
|
|
||||||
module.exports = { DataService }
|
module.exports = { DataService }
|
||||||
|
|
||||||
/** @returns {cds.Service} */
|
|
||||||
function findDataSource(dataSourceName, entityName) {
|
function findDataSource(dataSourceName, entityName) {
|
||||||
for (let srv of Object.values(cds.services)) { // all connected services
|
for (let srv of Object.values(cds.services)) { // all connected services
|
||||||
if (!srv.name) continue // FIXME intermediate/pending in cds.services ?
|
if (!srv.name) continue // FIXME intermediate/pending in cds.services ?
|
||||||
27
eslint.config.js
Normal file
27
eslint.config.js
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
const eslintCds = require('@sap/eslint-plugin-cds')
|
||||||
|
const eslintJs = require('@eslint/js')
|
||||||
|
const globals = require('globals')
|
||||||
|
|
||||||
|
module.exports = [
|
||||||
|
eslintJs.configs.recommended,
|
||||||
|
eslintCds.configs.recommended,
|
||||||
|
{
|
||||||
|
languageOptions: {
|
||||||
|
globals: {
|
||||||
|
sap: true,
|
||||||
|
...globals.es2022,
|
||||||
|
...globals.browser,
|
||||||
|
...globals.node,
|
||||||
|
...globals.jest,
|
||||||
|
...globals.mocha,
|
||||||
|
...eslintCds.configs.recommended.languageOptions.globals
|
||||||
|
}
|
||||||
|
},
|
||||||
|
rules: {
|
||||||
|
'no-console': 'off',
|
||||||
|
'no-unused-vars': ['warn', { argsIgnorePattern: '^_' }],
|
||||||
|
'require-atomic-updates': 'off',
|
||||||
|
'require-await': 'warn'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
@@ -1,20 +0,0 @@
|
|||||||
import cds from '@sap/cds/eslint.config.mjs'
|
|
||||||
export default [ ...cds.recommended ]
|
|
||||||
|
|
||||||
//
|
|
||||||
// The above is the recommended minimalistic eslint config, just using
|
|
||||||
// recommended defaults provided by cds. Alternatively, go for enhanced
|
|
||||||
// project-specific options, but only if really required, like this:
|
|
||||||
//
|
|
||||||
// export default [ ...cds.recommended, {
|
|
||||||
// rules: {
|
|
||||||
// 'complexity': [ 'warn', 66 ],
|
|
||||||
// 'require-await': 'warn',
|
|
||||||
// 'no-await-in-loop': 'warn',
|
|
||||||
// },
|
|
||||||
// }, {
|
|
||||||
// ignores: [
|
|
||||||
// '**/webapp/**'
|
|
||||||
// ]
|
|
||||||
// }]
|
|
||||||
//
|
|
||||||
@@ -1,29 +0,0 @@
|
|||||||
{
|
|
||||||
"version": "1.1.0",
|
|
||||||
"options": {
|
|
||||||
"management": true,
|
|
||||||
"messagingrest": true,
|
|
||||||
"messaging": true
|
|
||||||
},
|
|
||||||
"rules": {
|
|
||||||
"topicRules": {
|
|
||||||
"publishFilter": [
|
|
||||||
"*"
|
|
||||||
],
|
|
||||||
"subscribeFilter": [
|
|
||||||
"*"
|
|
||||||
]
|
|
||||||
},
|
|
||||||
"queueRules": {
|
|
||||||
"publishFilter": [
|
|
||||||
"*"
|
|
||||||
],
|
|
||||||
"subscribeFilter": [
|
|
||||||
"*"
|
|
||||||
]
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"authorities": [
|
|
||||||
"$ACCEPT_GRANTED_AUTHORITIES"
|
|
||||||
]
|
|
||||||
}
|
|
||||||
43
fiori/app/_i18n/i18n.properties
Normal file
43
fiori/app/_i18n/i18n.properties
Normal file
@@ -0,0 +1,43 @@
|
|||||||
|
Books = Books
|
||||||
|
Book = Book
|
||||||
|
ID = ID
|
||||||
|
Title = Title
|
||||||
|
Author = Author
|
||||||
|
AuthorID = Author ID
|
||||||
|
Stock = Stock
|
||||||
|
Name = Name
|
||||||
|
Description = Description
|
||||||
|
Image = Image
|
||||||
|
AuthorName = Author's Name
|
||||||
|
DateOfBirth = Date of Birth
|
||||||
|
DateOfDeath = Date of Death
|
||||||
|
PlaceOfBirth = Place of Birth
|
||||||
|
PlaceOfDeath = Place of Death
|
||||||
|
Age = Age
|
||||||
|
Lifetime = Lifetime
|
||||||
|
Authors = Authors
|
||||||
|
|
||||||
|
Order = Order
|
||||||
|
Orders = Orders
|
||||||
|
OrderNo = Order Number
|
||||||
|
OrderItems = Order Items
|
||||||
|
Customer = Customer
|
||||||
|
Product = Product
|
||||||
|
ProductID = Product ID
|
||||||
|
ProductTitle = Product Title
|
||||||
|
UnitPrice = Unit Price
|
||||||
|
Quantity = Quantity
|
||||||
|
|
||||||
|
Price = Price
|
||||||
|
Currency = Currency
|
||||||
|
Date = Date
|
||||||
|
Rating = Rating
|
||||||
|
NumberOfReviews = Number of Reviews
|
||||||
|
|
||||||
|
Genre = Genre
|
||||||
|
Genres = Genres
|
||||||
|
SubGenres = Sub Genres
|
||||||
|
|
||||||
|
NumCode = Numeric Code
|
||||||
|
MinorUnit = Minor Unit
|
||||||
|
Exponent = Exponent
|
||||||
@@ -1,5 +1,14 @@
|
|||||||
|
Books = Bücher
|
||||||
|
Book = Buch
|
||||||
|
ID = ID
|
||||||
|
Title = Titel
|
||||||
|
Authors = Autoren
|
||||||
|
Author = Autor
|
||||||
|
AuthorID = ID des Autors
|
||||||
AuthorName = Name des Autors
|
AuthorName = Name des Autors
|
||||||
Age = Alter
|
Age = Alter
|
||||||
rder = Bestellung
|
Name = Name
|
||||||
|
Stock = Bestand
|
||||||
|
Order = Bestellung
|
||||||
Orders = Bestellungen
|
Orders = Bestellungen
|
||||||
Price = Preis
|
Price = Preis
|
||||||
|
|||||||
@@ -1,8 +0,0 @@
|
|||||||
Age = Age
|
|
||||||
Lifetime = Lifetime
|
|
||||||
|
|
||||||
SubGenres = Sub Genres
|
|
||||||
|
|
||||||
NumCode = Numeric Code
|
|
||||||
MinorUnit = Minor Unit
|
|
||||||
Exponent = Exponent
|
|
||||||
@@ -6,8 +6,8 @@ using CatalogService from '@capire/bookstore';
|
|||||||
//
|
//
|
||||||
annotate CatalogService.Books with @(UI : {
|
annotate CatalogService.Books with @(UI : {
|
||||||
HeaderInfo : {
|
HeaderInfo : {
|
||||||
TypeName : '{i18n>Book}',
|
TypeName : 'Book',
|
||||||
TypeNamePlural : '{i18n>Books}',
|
TypeNamePlural : 'Books',
|
||||||
Description : {Value : author}
|
Description : {Value : author}
|
||||||
},
|
},
|
||||||
HeaderFacets : [{
|
HeaderFacets : [{
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<!doctype html>
|
<!DOCTYPE html>
|
||||||
<html>
|
<html>
|
||||||
|
|
||||||
<head>
|
<head>
|
||||||
|
|
||||||
<meta http-equiv="X-UA-Compatible" content="IE=edge" />
|
<meta http-equiv="X-UA-Compatible" content="IE=edge" />
|
||||||
<meta http-equiv="Content-Type" content="text/html;charset=UTF-8" />
|
<meta http-equiv="Content-Type" content="text/html;charset=UTF-8" />
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
||||||
@@ -10,22 +10,21 @@
|
|||||||
<script>
|
<script>
|
||||||
window["sap-ushell-config"] = {
|
window["sap-ushell-config"] = {
|
||||||
defaultRenderer: "fiori2",
|
defaultRenderer: "fiori2",
|
||||||
applications: {},
|
applications: {}
|
||||||
};
|
};
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<script id="sap-ushell-bootstrap" src="https://ui5.sap.com/test-resources/sap/ushell/bootstrap/sandbox.js"></script>
|
<script id="sap-ushell-bootstrap" src="https://sapui5.hana.ondemand.com/test-resources/sap/ushell/bootstrap/sandbox.js"></script>
|
||||||
<script id="sap-ui-bootstrap" src="https://ui5.sap.com/resources/sap-ui-core.js"
|
<script id="sap-ui-bootstrap" src="https://sapui5.hana.ondemand.com/resources/sap-ui-core.js"
|
||||||
data-sap-ui-libs="sap.m, sap.ushell, sap.collaboration, sap.ui.layout" data-sap-ui-compatVersion="edge"
|
data-sap-ui-libs="sap.m, sap.ushell, sap.collaboration, sap.ui.layout"
|
||||||
data-sap-ui-theme="sap_horizon"></script>
|
data-sap-ui-compatVersion="edge"
|
||||||
|
data-sap-ui-theme="sap_horizon"
|
||||||
|
data-sap-ui-frameOptions="allow"
|
||||||
|
></script>
|
||||||
<script>
|
<script>
|
||||||
sap.ui.getCore().attachInit(() =>
|
sap.ui.getCore().attachInit(()=> sap.ushell.Container.createRenderer().placeAt("content"))
|
||||||
sap.ushell.Container.createRenderer().placeAt("content")
|
|
||||||
);
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
</head>
|
</head>
|
||||||
|
<body class="sapUiBody" id="content"></body>
|
||||||
<body class="sapUiBody" id="content">
|
|
||||||
</body>
|
|
||||||
|
|
||||||
</html>
|
</html>
|
||||||
8
fiori/server.js
Normal file
8
fiori/server.js
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
// install OData v2 adapter
|
||||||
|
const cds = require("@sap/cds")
|
||||||
|
const proxy = require('@cap-js-community/odata-v2-adapter')
|
||||||
|
const opts = global.it ? { target:'auto' } : {} // for tests, set 'auto' to detect port dynamically
|
||||||
|
cds.on('bootstrap', app => app.use(proxy(opts))) // install proxy
|
||||||
|
// cds.log('cov2ap','silent') // suppress anoying log outpout, e.g. for `npm run mocha -- --reporter nyan`
|
||||||
|
|
||||||
|
module.exports = require('@capire/bookstore/server.js')
|
||||||
15
hello/README.md
Normal file
15
hello/README.md
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
# Hello World Getting Started Sample
|
||||||
|
|
||||||
|
## Next Steps
|
||||||
|
|
||||||
|
- To run the JavaScript implementation, open a new terminal and run `cds watch`.
|
||||||
|
- To run the TypeScript implementation, open a new terminal and run `cds-ts watch`.
|
||||||
|
|
||||||
|
Then call the service at: http://localhost:4004/say/hello(to='world')
|
||||||
|
|
||||||
|
## Learn More
|
||||||
|
|
||||||
|
Learn more about:
|
||||||
|
|
||||||
|
- [Hello World!](https://cap.cloud.sap/docs/get-started/hello-world)
|
||||||
|
- [Using TypeScript](https://cap.cloud.sap/docs/node.js/typescript)
|
||||||
42
hello/package.json
Normal file
42
hello/package.json
Normal file
@@ -0,0 +1,42 @@
|
|||||||
|
{
|
||||||
|
"name": "@capire/hello-world",
|
||||||
|
"version": "1.0.0",
|
||||||
|
"scripts": {
|
||||||
|
"test": "npx jest --silent",
|
||||||
|
"start": "cds serve srv/world.cds",
|
||||||
|
"start:ts": "cds-ts serve srv/world.cds"
|
||||||
|
},
|
||||||
|
"dependencies": {
|
||||||
|
"@sap/cds": ">=5.0.4"
|
||||||
|
},
|
||||||
|
"devDependencies": {
|
||||||
|
"@types/jest": "*",
|
||||||
|
"@types/node": "*",
|
||||||
|
"typescript": ">=4.3.5"
|
||||||
|
},
|
||||||
|
"eslintConfig": {
|
||||||
|
"extends": "eslint:recommended",
|
||||||
|
"env": {
|
||||||
|
"es2020": true,
|
||||||
|
"node": true,
|
||||||
|
"jest": true,
|
||||||
|
"mocha": true
|
||||||
|
},
|
||||||
|
"globals": {
|
||||||
|
"SELECT": true,
|
||||||
|
"INSERT": true,
|
||||||
|
"UPDATE": true,
|
||||||
|
"DELETE": true,
|
||||||
|
"CREATE": true,
|
||||||
|
"DROP": true,
|
||||||
|
"CDL": true,
|
||||||
|
"CQL": true,
|
||||||
|
"CXL": true,
|
||||||
|
"cds": true
|
||||||
|
},
|
||||||
|
"rules": {
|
||||||
|
"no-console": "off",
|
||||||
|
"require-atomic-updates": "off"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
3
hello/srv/world.cds
Normal file
3
hello/srv/world.cds
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
service say @(path: '/say') {
|
||||||
|
function hello (to:String) returns String;
|
||||||
|
}
|
||||||
7
hello/srv/world.js
Normal file
7
hello/srv/world.js
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
module.exports = class say {
|
||||||
|
hello(req) {
|
||||||
|
let {to} = req.data
|
||||||
|
if (to === 'me') to = require('os').userInfo().username
|
||||||
|
return `Hello ${to}!`
|
||||||
|
}
|
||||||
|
}
|
||||||
7
hello/srv/world.ts
Normal file
7
hello/srv/world.ts
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
import { Request } from "@sap/cds"
|
||||||
|
|
||||||
|
module.exports = class say {
|
||||||
|
hello(req: Request) {
|
||||||
|
return `Hello ${req.data.to} from a TypeScript file!`
|
||||||
|
}
|
||||||
|
}
|
||||||
13
hello/test/hello-world-test.js
Normal file
13
hello/test/hello-world-test.js
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
const cds = require ('@sap/cds')
|
||||||
|
describe('Hello world!', () => {
|
||||||
|
|
||||||
|
beforeAll (()=> process.env.CDS_TYPESCRIPT = true)
|
||||||
|
afterAll (()=> delete process.env.CDS_TYPESCRIPT)
|
||||||
|
const {GET} = cds.test.in(__dirname,'../srv').run('serve', 'world.cds')
|
||||||
|
|
||||||
|
it('should say hello with class impl', async () => {
|
||||||
|
const {data} = await GET`/say/hello(to='world')`
|
||||||
|
expect(data.value).toMatch(/Hello world.*typescript.*/i)
|
||||||
|
})
|
||||||
|
|
||||||
|
})
|
||||||
1
hello/test/test.http
Normal file
1
hello/test/test.http
Normal file
@@ -0,0 +1 @@
|
|||||||
|
GET http://localhost:4004/say/hello(to='world')
|
||||||
@@ -13,5 +13,10 @@
|
|||||||
"scripts": {
|
"scripts": {
|
||||||
"start": "cds-serve",
|
"start": "cds-serve",
|
||||||
"watch": "cds watch"
|
"watch": "cds watch"
|
||||||
|
},
|
||||||
|
"cds": {
|
||||||
|
"requires": {
|
||||||
|
"db": "sql"
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
const cds = require ('@sap/cds')
|
const cds = require ('@sap/cds/lib')
|
||||||
const LOG = cds.log('cds.log')
|
const LOG = cds.log('cds.log')
|
||||||
|
|
||||||
module.exports = class LogService extends cds.Service {
|
module.exports = class LogService extends cds.Service {
|
||||||
13
media/db/data-model.cds
Normal file
13
media/db/data-model.cds
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
namespace sap.capire.media;
|
||||||
|
|
||||||
|
entity Media {
|
||||||
|
|
||||||
|
key id:Integer;
|
||||||
|
@Core.MediaType: mediaType
|
||||||
|
content : LargeBinary ;
|
||||||
|
|
||||||
|
@Core.IsMediaType: true
|
||||||
|
mediaType : String;
|
||||||
|
fileName : String;
|
||||||
|
applicationName:String;
|
||||||
|
}
|
||||||
2
media/index.cds
Normal file
2
media/index.cds
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
using from './db/data-model';
|
||||||
|
using from './srv/media-service';
|
||||||
19
media/package.json
Normal file
19
media/package.json
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
{
|
||||||
|
"name": "@capire/media",
|
||||||
|
"version": "1.0.0",
|
||||||
|
"dependencies": {
|
||||||
|
"lokijs": "^1.5.6"
|
||||||
|
},
|
||||||
|
"files": [
|
||||||
|
"db",
|
||||||
|
"srv",
|
||||||
|
"index.cds"
|
||||||
|
],
|
||||||
|
"cds": {
|
||||||
|
"requires": {
|
||||||
|
"db": {
|
||||||
|
"kind": "sql"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
8
media/srv/media-service.cds
Normal file
8
media/srv/media-service.cds
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
using { sap.capire.media as db } from '../db/data-model';
|
||||||
|
namespace sap.capire.media;
|
||||||
|
|
||||||
|
service MediaServer {
|
||||||
|
entity Media as projection on db.Media ;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
68
media/srv/media-service.js
Normal file
68
media/srv/media-service.js
Normal file
@@ -0,0 +1,68 @@
|
|||||||
|
const loki = require('lokijs')
|
||||||
|
const db = new loki('DB')
|
||||||
|
const mediaDB = db.addCollection('Media')
|
||||||
|
const { Readable, PassThrough } = require('stream')
|
||||||
|
|
||||||
|
module.exports = srv => {
|
||||||
|
srv.before('CREATE', 'Media', req => {
|
||||||
|
const obj = mediaDB.insert({ media: '' })
|
||||||
|
req.data.id = obj.$loki
|
||||||
|
})
|
||||||
|
|
||||||
|
srv.on('UPDATE', 'Media', (req, next) => {
|
||||||
|
const url = req._.req.path
|
||||||
|
if (url.includes('content')) {
|
||||||
|
const id = req.data.id
|
||||||
|
const obj = mediaDB.get(id)
|
||||||
|
if (!obj) {
|
||||||
|
req.reject(404, 'No record found for the ID')
|
||||||
|
return
|
||||||
|
}
|
||||||
|
const stream = new PassThrough()
|
||||||
|
const chunks = []
|
||||||
|
stream.on('data', chunk => {
|
||||||
|
chunks.push(chunk)
|
||||||
|
})
|
||||||
|
stream.on('end', () => {
|
||||||
|
obj.media = Buffer.concat(chunks).toString('base64')
|
||||||
|
mediaDB.update(obj)
|
||||||
|
})
|
||||||
|
req.data.content.pipe(stream)
|
||||||
|
} else return next()
|
||||||
|
})
|
||||||
|
|
||||||
|
srv.on('READ', 'Media', (req, next) => {
|
||||||
|
const url = req._.req.path
|
||||||
|
if (url.includes('content')) {
|
||||||
|
const id = req.data.id
|
||||||
|
const mediaObj = mediaDB.get(id)
|
||||||
|
if (!mediaObj) {
|
||||||
|
req.reject(404, 'Media not found for the ID')
|
||||||
|
return
|
||||||
|
}
|
||||||
|
const decodedMedia = Buffer.from(
|
||||||
|
mediaObj.media.split(';base64,').pop(),
|
||||||
|
'base64'
|
||||||
|
)
|
||||||
|
return _formatResult(decodedMedia)
|
||||||
|
} else return next() //> delegate to next/default handlers
|
||||||
|
})
|
||||||
|
|
||||||
|
srv.on('DELETE', 'Media', (req, next) => {
|
||||||
|
const id = req.data.id
|
||||||
|
mediaDB
|
||||||
|
.chain()
|
||||||
|
.find({ $loki: id })
|
||||||
|
.remove()
|
||||||
|
return next() //> delegate to next/default handlers
|
||||||
|
})
|
||||||
|
|
||||||
|
function _formatResult (decodedMedia) {
|
||||||
|
const readable = new Readable()
|
||||||
|
const result = new Array()
|
||||||
|
readable.push(decodedMedia)
|
||||||
|
readable.push(null)
|
||||||
|
result.push({ value: readable })
|
||||||
|
return result
|
||||||
|
}
|
||||||
|
}
|
||||||
BIN
media/test/Test.png
Normal file
BIN
media/test/Test.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 3.0 KiB |
35
media/test/media.http
Normal file
35
media/test/media.http
Normal file
@@ -0,0 +1,35 @@
|
|||||||
|
### Requires REST Client for VS Code
|
||||||
|
### https://marketplace.visualstudio.com/items?itemName=humao.rest-client
|
||||||
|
###
|
||||||
|
@protocol = http
|
||||||
|
@host = localhost
|
||||||
|
@port = 4004
|
||||||
|
### Read Pictures
|
||||||
|
GET {{protocol}}://{{host}}:{{port}}/media-server/Media
|
||||||
|
Authorization: Basic admin:
|
||||||
|
|
||||||
|
### Create Picture with mediatype
|
||||||
|
POST {{protocol}}://{{host}}:{{port}}/media-server/Media
|
||||||
|
Authorization: Basic admin:
|
||||||
|
Accept: application/json
|
||||||
|
Content-Type: application/json
|
||||||
|
|
||||||
|
{
|
||||||
|
"id": 1,
|
||||||
|
"mediaType": "image/png"
|
||||||
|
}
|
||||||
|
|
||||||
|
### Upload Binary PNG
|
||||||
|
PUT {{protocol}}://{{host}}:{{port}}/media-server/Media(1)/content
|
||||||
|
Authorization: Basic admin:
|
||||||
|
Content-Type: image/png
|
||||||
|
|
||||||
|
< ./Test.png
|
||||||
|
|
||||||
|
### Read Binary
|
||||||
|
GET {{protocol}}://{{host}}:{{port}}/media-server/Media(1)/content
|
||||||
|
Authorization: Basic admin:
|
||||||
|
|
||||||
|
### Delete Image
|
||||||
|
DELETE {{protocol}}://{{host}}:{{port}}/media-server/Media(1)
|
||||||
|
Authorization: Basic admin:
|
||||||
177
mta.yaml
177
mta.yaml
@@ -1,177 +0,0 @@
|
|||||||
_schema-version: 3.3.0
|
|
||||||
ID: capire.samples
|
|
||||||
version: 3.0.0
|
|
||||||
description: "A monorepo with several samples for CAP."
|
|
||||||
parameters:
|
|
||||||
enable-parallel-deployments: true
|
|
||||||
build-parameters:
|
|
||||||
before-all:
|
|
||||||
- builder: custom
|
|
||||||
commands:
|
|
||||||
- npm ci
|
|
||||||
- npx cds build ./shared-db --for hana --production
|
|
||||||
- npx cds build ./orders --for nodejs --production --ws-pack
|
|
||||||
- npx cds build ./reviews --for nodejs --production
|
|
||||||
- npx cds build ./bookstore --for nodejs --production --ws-pack
|
|
||||||
modules:
|
|
||||||
- name: bookstore-srv
|
|
||||||
type: nodejs
|
|
||||||
path: bookstore/gen/srv
|
|
||||||
parameters:
|
|
||||||
instances: 1
|
|
||||||
buildpack: nodejs_buildpack
|
|
||||||
build-parameters:
|
|
||||||
builder: npm
|
|
||||||
properties:
|
|
||||||
cds_requires_ReviewsService_credentials: {"destination": "reviews-dest","path": "/reviews"}
|
|
||||||
cds_requires_OrdersService_credentials: {"destination": "orders-dest","path": "/odata/v4/orders"}
|
|
||||||
provides:
|
|
||||||
- name: bookstore-api # required by consumers of CAP services (e.g. approuter)
|
|
||||||
properties:
|
|
||||||
srv-url: ${default-url}
|
|
||||||
requires:
|
|
||||||
- name: samples-db
|
|
||||||
- name: samples-auth
|
|
||||||
- name: samples-messaging
|
|
||||||
- name: samples-destination
|
|
||||||
|
|
||||||
- name: orders-srv
|
|
||||||
type: nodejs
|
|
||||||
path: orders/gen/srv
|
|
||||||
parameters:
|
|
||||||
instances: 1
|
|
||||||
buildpack: nodejs_buildpack
|
|
||||||
build-parameters:
|
|
||||||
builder: npm
|
|
||||||
provides:
|
|
||||||
- name: orders-api # required by consumers of CAP services (e.g. approuter)
|
|
||||||
properties:
|
|
||||||
srv-url: ${default-url}
|
|
||||||
requires:
|
|
||||||
- name: samples-db
|
|
||||||
- name: samples-auth
|
|
||||||
- name: samples-messaging
|
|
||||||
- name: samples-destination
|
|
||||||
|
|
||||||
- name: reviews-srv
|
|
||||||
type: nodejs
|
|
||||||
path: reviews/gen/srv
|
|
||||||
parameters:
|
|
||||||
instances: 1
|
|
||||||
buildpack: nodejs_buildpack
|
|
||||||
build-parameters:
|
|
||||||
builder: npm
|
|
||||||
provides:
|
|
||||||
- name: reviews-api # required by consumers of CAP services (e.g. approuter)
|
|
||||||
properties:
|
|
||||||
srv-url: ${default-url}
|
|
||||||
requires:
|
|
||||||
- name: samples-db
|
|
||||||
- name: samples-auth
|
|
||||||
- name: samples-messaging
|
|
||||||
- name: samples-destination
|
|
||||||
|
|
||||||
- name: samples-db-deployer
|
|
||||||
type: hdb
|
|
||||||
path: shared-db/gen/db
|
|
||||||
parameters:
|
|
||||||
buildpack: nodejs_buildpack
|
|
||||||
requires:
|
|
||||||
- name: samples-db
|
|
||||||
|
|
||||||
- name: samples
|
|
||||||
type: approuter.nodejs
|
|
||||||
path: app/router
|
|
||||||
parameters:
|
|
||||||
keep-existing-routes: true
|
|
||||||
disk-quota: 256M
|
|
||||||
memory: 256M
|
|
||||||
requires:
|
|
||||||
- name: orders-api
|
|
||||||
group: destinations
|
|
||||||
properties:
|
|
||||||
name: orders-api
|
|
||||||
url: ~{srv-url}
|
|
||||||
forwardAuthToken: true
|
|
||||||
- name: reviews-api
|
|
||||||
group: destinations
|
|
||||||
properties:
|
|
||||||
name: reviews-api
|
|
||||||
url: ~{srv-url}
|
|
||||||
forwardAuthToken: true
|
|
||||||
- name: bookstore-api
|
|
||||||
group: destinations
|
|
||||||
properties:
|
|
||||||
name: bookstore-api
|
|
||||||
url: ~{srv-url}
|
|
||||||
forwardAuthToken: true
|
|
||||||
- name: samples-auth
|
|
||||||
- name: samples-destination
|
|
||||||
provides:
|
|
||||||
- name: app-api
|
|
||||||
properties:
|
|
||||||
app-protocol: ${protocol}
|
|
||||||
app-uri: ${default-uri}
|
|
||||||
|
|
||||||
- name: destination-content
|
|
||||||
type: com.sap.application.content
|
|
||||||
requires:
|
|
||||||
- name: orders-api
|
|
||||||
- name: reviews-api
|
|
||||||
- name: bookstore-api
|
|
||||||
- name: samples-auth
|
|
||||||
parameters:
|
|
||||||
service-key:
|
|
||||||
name: xsuaa_service-key
|
|
||||||
- name: samples-destination
|
|
||||||
parameters:
|
|
||||||
content-target: true
|
|
||||||
build-parameters:
|
|
||||||
no-source: true
|
|
||||||
parameters:
|
|
||||||
content:
|
|
||||||
instance:
|
|
||||||
existing_destinations_policy: update
|
|
||||||
destinations:
|
|
||||||
- Name: orders-dest
|
|
||||||
URL: ~{orders-api/srv-url}
|
|
||||||
Authentication: OAuth2ClientCredentials
|
|
||||||
TokenServiceInstanceName: samples-auth
|
|
||||||
TokenServiceKeyName: xsuaa_service-key
|
|
||||||
- Name: reviews-dest
|
|
||||||
URL: ~{reviews-api/srv-url}
|
|
||||||
Authentication: OAuth2ClientCredentials
|
|
||||||
TokenServiceInstanceName: samples-auth
|
|
||||||
TokenServiceKeyName: xsuaa_service-key
|
|
||||||
|
|
||||||
resources:
|
|
||||||
- name: samples-db
|
|
||||||
type: com.sap.xs.hdi-container
|
|
||||||
parameters:
|
|
||||||
service: hana
|
|
||||||
service-plan: hdi-shared
|
|
||||||
- name: samples-auth
|
|
||||||
type: org.cloudfoundry.managed-service
|
|
||||||
processed-after:
|
|
||||||
- samples-messaging
|
|
||||||
parameters:
|
|
||||||
service: xsuaa
|
|
||||||
service-plan: application
|
|
||||||
path: ./xs-security.json
|
|
||||||
config:
|
|
||||||
xsappname: samples-${org}-${space}
|
|
||||||
tenant-mode: dedicated
|
|
||||||
- name: samples-messaging
|
|
||||||
type: org.cloudfoundry.managed-service
|
|
||||||
parameters:
|
|
||||||
service: enterprise-messaging
|
|
||||||
service-plan: default
|
|
||||||
path: ./event-mesh.json
|
|
||||||
config:
|
|
||||||
emname: bookstore-${org}-${space}
|
|
||||||
namespace: cap/samples/${space}
|
|
||||||
- name: samples-destination
|
|
||||||
type: org.cloudfoundry.managed-service
|
|
||||||
parameters:
|
|
||||||
service: destination
|
|
||||||
service-plan: lite
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
Order = Bestellung
|
|
||||||
Orders = Bestellungen
|
|
||||||
Price = Preis
|
|
||||||
@@ -1,10 +0,0 @@
|
|||||||
Order = Order
|
|
||||||
Orders = Orders
|
|
||||||
OrderNo = Order Number
|
|
||||||
OrderItems = Order Items
|
|
||||||
Customer = Customer
|
|
||||||
Product = Product
|
|
||||||
ProductID = Product ID
|
|
||||||
ProductTitle = Product Title
|
|
||||||
UnitPrice = Unit Price
|
|
||||||
Quantity = Quantity
|
|
||||||
@@ -1,10 +0,0 @@
|
|||||||
Order = Order
|
|
||||||
Orders = Orders
|
|
||||||
OrderNo = Numéro de commande
|
|
||||||
OrderItems = Articles de la commande
|
|
||||||
Customer = Client
|
|
||||||
Product = Produit
|
|
||||||
ProductID = ID du produit
|
|
||||||
ProductTitle = Titre du produit
|
|
||||||
UnitPrice = Prix unitaire
|
|
||||||
Quantity = Quantité
|
|
||||||
@@ -2,7 +2,7 @@ using { Currency, User, managed, cuid } from '@sap/cds/common';
|
|||||||
namespace sap.capire.orders;
|
namespace sap.capire.orders;
|
||||||
|
|
||||||
entity Orders : cuid, managed {
|
entity Orders : cuid, managed {
|
||||||
OrderNo : String(44) @title:'Order Number'; //> readable key
|
OrderNo : String(22) @title:'Order Number'; //> readable key
|
||||||
Items : Composition of many {
|
Items : Composition of many {
|
||||||
key ID : UUID;
|
key ID : UUID;
|
||||||
product : Association to Products;
|
product : Association to Products;
|
||||||
|
|||||||
@@ -2,18 +2,7 @@
|
|||||||
"name": "@capire/orders",
|
"name": "@capire/orders",
|
||||||
"version": "1.0.0",
|
"version": "1.0.0",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@cap-js/hana": "^1.7.0",
|
|
||||||
"@capire/common": "*",
|
"@capire/common": "*",
|
||||||
"@sap/cds": ">=5",
|
"@sap/cds": ">=5"
|
||||||
"@sap/xssec": "^4.4.0"
|
|
||||||
},
|
|
||||||
"scripts": {
|
|
||||||
"start": "cds-serve"
|
|
||||||
},
|
|
||||||
"cds": {
|
|
||||||
"requires": {
|
|
||||||
"messaging": true,
|
|
||||||
"db": true
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -2,14 +2,4 @@ using { sap.capire.orders as my } from '../db/schema';
|
|||||||
|
|
||||||
service OrdersService {
|
service OrdersService {
|
||||||
entity Orders as projection on my.Orders;
|
entity Orders as projection on my.Orders;
|
||||||
|
|
||||||
event OrderChanged {
|
|
||||||
product: String;
|
|
||||||
deltaQuantity: Integer;
|
|
||||||
}
|
|
||||||
|
|
||||||
@odata.draft.bypass
|
|
||||||
@(requires: 'system-user')
|
|
||||||
entity OrdersNoDraft as projection on my.Orders;
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -29,7 +29,7 @@ class OrdersService extends cds.ApplicationService {
|
|||||||
/** order changed -> broadcast event */
|
/** order changed -> broadcast event */
|
||||||
orderChanged (product, deltaQuantity) {
|
orderChanged (product, deltaQuantity) {
|
||||||
// Emit events to inform subscribers about changes in orders
|
// Emit events to inform subscribers about changes in orders
|
||||||
console.log ('> emitting:', 'OrderChanged', { product, deltaQuantity }) // eslint-disable-line no-console
|
console.log ('> emitting:', 'OrderChanged', { product, deltaQuantity })
|
||||||
return this.emit ('OrderChanged', { product, deltaQuantity })
|
return this.emit ('OrderChanged', { product, deltaQuantity })
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
2546
package-lock.json
generated
2546
package-lock.json
generated
File diff suppressed because it is too large
Load Diff
58
package.json
58
package.json
@@ -1,42 +1,52 @@
|
|||||||
{
|
{
|
||||||
"name": "@capire/samples",
|
"name": "@capire/samples",
|
||||||
"version": "3.0.0",
|
"version": "2.0.0",
|
||||||
"description": "A monorepo with several samples for CAP.",
|
"description": "A monorepo with several samples for CAP.",
|
||||||
"repository": "https://github.com/sap-samples/cloud-cap-samples.git",
|
"repository": "https://github.com/sap-samples/cloud-cap-samples.git",
|
||||||
"author": "daniel.hutzel@sap.com",
|
"author": "daniel.hutzel@sap.com",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@sap/cds": ">=8",
|
"@sap/cds": ">=7"
|
||||||
"@cap-js/hana": "^1",
|
|
||||||
"@sap/xssec": "^4"
|
|
||||||
},
|
},
|
||||||
"workspaces": [
|
"workspaces": [
|
||||||
"./bookshop",
|
"./bookshop",
|
||||||
"./bookstore",
|
"./bookstore",
|
||||||
"./common",
|
"./common",
|
||||||
|
"./data-viewer",
|
||||||
"./fiori",
|
"./fiori",
|
||||||
|
"./hello",
|
||||||
|
"./media",
|
||||||
"./orders",
|
"./orders",
|
||||||
"./reviews",
|
"./loggers",
|
||||||
"./etc/data-viewer",
|
"./reviews"
|
||||||
"./etc/loggers"
|
|
||||||
],
|
],
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@cap-js/cds-test": "^0",
|
|
||||||
"@cap-js/cds-types": "^0",
|
|
||||||
"@cap-js/sqlite": "^1",
|
"@cap-js/sqlite": "^1",
|
||||||
"@sap/cds-dk": "^8",
|
"@sap/eslint-plugin-cds": "^3",
|
||||||
|
"axios": "^1",
|
||||||
|
"chai": "^4.3.4",
|
||||||
|
"chai-as-promised": "^7.1.1",
|
||||||
|
"chai-subset": "^1.6.0",
|
||||||
"eslint": "^9",
|
"eslint": "^9",
|
||||||
"semver": "^7"
|
"semver": "^7"
|
||||||
},
|
},
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"build": "mbt build -t gen --mtar mta.tar",
|
"cleanup": "rm -rf node_modules && rm -rf */node_modules && rm -rf */*/node_modules",
|
||||||
"deploy": "cf deploy gen/mta.tar",
|
|
||||||
"undeploy": "cf undeploy capire.samples --delete-services --delete-service-keys",
|
|
||||||
"bookshop": "cds watch bookshop",
|
"bookshop": "cds watch bookshop",
|
||||||
"fiori": "cds watch fiori",
|
"fiori": "cds watch fiori",
|
||||||
"lint": "eslint",
|
"hello": "cds watch hello",
|
||||||
"test": "npx jest --silent",
|
"media": "cds watch media",
|
||||||
|
"mocha": "CDS_TEST_SILENT=y npx mocha",
|
||||||
"jest": "npx jest --silent",
|
"jest": "npx jest --silent",
|
||||||
"mocha": "CDS_TEST_SILENT=y npx mocha"
|
"start": "cds watch fiori",
|
||||||
|
"test": "npm run jest -- --silent",
|
||||||
|
"test:hello": "cd hello && npm test",
|
||||||
|
"lint": "eslint ."
|
||||||
|
},
|
||||||
|
"jest": {
|
||||||
|
"testTimeout": 20000,
|
||||||
|
"testMatch": [
|
||||||
|
"**/*.test.js"
|
||||||
|
]
|
||||||
},
|
},
|
||||||
"mocha": {
|
"mocha": {
|
||||||
"recursive": true,
|
"recursive": true,
|
||||||
@@ -44,19 +54,5 @@
|
|||||||
"timeout": 6666
|
"timeout": 6666
|
||||||
},
|
},
|
||||||
"license": "SEE LICENSE IN LICENSE",
|
"license": "SEE LICENSE IN LICENSE",
|
||||||
"private": true,
|
"private": true
|
||||||
"cds": {
|
|
||||||
"sql": {
|
|
||||||
"native_hana_associations": false
|
|
||||||
},
|
|
||||||
"requires": {
|
|
||||||
"[production]": {
|
|
||||||
"auth": "xsuaa"
|
|
||||||
},
|
|
||||||
"messaging": {
|
|
||||||
"kind": "enterprise-messaging"
|
|
||||||
},
|
|
||||||
"destinations": true
|
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -1,3 +0,0 @@
|
|||||||
Date = Datum
|
|
||||||
Rating = Bewertung
|
|
||||||
NumberOfReviews = Anzahl der Reviews
|
|
||||||
@@ -1,5 +0,0 @@
|
|||||||
Reviews = Reviews
|
|
||||||
Review = Review
|
|
||||||
Date = Date
|
|
||||||
Rating = Rating
|
|
||||||
NumberOfReviews = Number of Reviews
|
|
||||||
@@ -1,5 +0,0 @@
|
|||||||
Reviews = Avis
|
|
||||||
Review = Avis
|
|
||||||
Date = Date
|
|
||||||
Rating = Note
|
|
||||||
NumberOfReviews = Nombre d''avis
|
|
||||||
@@ -50,6 +50,7 @@ const reviews = Vue.createApp ({
|
|||||||
const res = await POST(`/Reviews`,review)
|
const res = await POST(`/Reviews`,review)
|
||||||
reviews.ID = res.data.ID
|
reviews.ID = res.data.ID
|
||||||
} else {
|
} else {
|
||||||
|
console.trace()
|
||||||
await PUT(`/Reviews/${review.ID}`,review)
|
await PUT(`/Reviews/${review.ID}`,review)
|
||||||
}
|
}
|
||||||
reviews.message = { succeeded: 'Your review was submitted successfully. Thanks.' }
|
reviews.message = { succeeded: 'Your review was submitted successfully. Thanks.' }
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
ID;subject;rating;reviewer;title;text
|
subject;rating;reviewer;title;text
|
||||||
1689144d-3b10-4849-bcbe-2408a13e161a;201;5;bob;Intriguing;Lorem ipsum dolor sit amet, consetetur sadipscing elitr, sed diam nonumy eirmod tempor invidunt ut labore et dolore magna aliquyam erat, sed diam voluptua. At vero eos et accusam et justo duo dolores et ea rebum. Stet clita kasd gubergren, no sea takimata sanctus est Lorem ipsum dolor sit amet. Lorem ipsum dolor sit amet, consetetur sadipscing elitr, sed diam nonumy eirmod tempor invidunt ut labore et dolore magna aliquyam erat, sed diam voluptua. At vero eos et accusam et justo duo dolores et ea rebum. Stet clita kasd gubergren, no sea takimata sanctus est Lorem ipsum dolor sit amet.
|
201;5;bob;Intriguing;Lorem ipsum dolor sit amet, consetetur sadipscing elitr, sed diam nonumy eirmod tempor invidunt ut labore et dolore magna aliquyam erat, sed diam voluptua. At vero eos et accusam et justo duo dolores et ea rebum. Stet clita kasd gubergren, no sea takimata sanctus est Lorem ipsum dolor sit amet. Lorem ipsum dolor sit amet, consetetur sadipscing elitr, sed diam nonumy eirmod tempor invidunt ut labore et dolore magna aliquyam erat, sed diam voluptua. At vero eos et accusam et justo duo dolores et ea rebum. Stet clita kasd gubergren, no sea takimata sanctus est Lorem ipsum dolor sit amet.
|
||||||
1689144d-3b10-4849-bcbe-2408a13e161b;201;4;bob;Fascinating;Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Id diam maecenas ultricies mi eget mauris pharetra et. Risus at ultrices mi tempus imperdiet nulla malesuada pellentesque. Pulvinar mattis nunc sed blandit libero. Facilisis magna etiam tempor orci eu. Nec sagittis aliquam malesuada bibendum arcu. Eu consequat ac felis donec. Ultricies tristique nulla aliquet enim tortor at auctor urna nunc. Tortor posuere ac ut consequat semper viverra nam libero. Amet nisl suscipit adipiscing bibendum est ultricies integer quis auctor. Scelerisque purus semper eget duis at tellus. Elementum tempus egestas sed sed risus pretium. Arcu dictum varius duis at. Amet luctus venenatis lectus magna fringilla urna. Eget velit aliquet sagittis id consectetur purus ut faucibus. Vitae auctor eu augue ut lectus. Fermentum iaculis eu non diam phasellus vestibulum.
|
201;4;bob;Fascinating;Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Id diam maecenas ultricies mi eget mauris pharetra et. Risus at ultrices mi tempus imperdiet nulla malesuada pellentesque. Pulvinar mattis nunc sed blandit libero. Facilisis magna etiam tempor orci eu. Nec sagittis aliquam malesuada bibendum arcu. Eu consequat ac felis donec. Ultricies tristique nulla aliquet enim tortor at auctor urna nunc. Tortor posuere ac ut consequat semper viverra nam libero. Amet nisl suscipit adipiscing bibendum est ultricies integer quis auctor. Scelerisque purus semper eget duis at tellus. Elementum tempus egestas sed sed risus pretium. Arcu dictum varius duis at. Amet luctus venenatis lectus magna fringilla urna. Eget velit aliquet sagittis id consectetur purus ut faucibus. Vitae auctor eu augue ut lectus. Fermentum iaculis eu non diam phasellus vestibulum.
|
||||||
1689144d-3b10-4849-bcbe-2408a13e161c;207;2;bob;What is this?;Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Libero justo laoreet sit amet cursus sit amet dictum. Nunc faucibus a pellentesque sit. Dis parturient montes nascetur ridiculus mus mauris vitae ultricies. Enim nunc faucibus a pellentesque. Commodo quis imperdiet massa tincidunt nunc pulvinar sapien. Cras ornare arcu dui vivamus. Facilisi etiam dignissim diam quis enim lobortis. Et molestie ac feugiat sed. Urna neque viverra justo nec ultrices dui. Ullamcorper a lacus vestibulum sed arcu non. Volutpat ac tincidunt vitae semper quis. Dignissim sodales ut eu sem. Feugiat in fermentum posuere urna nec. At augue eget arcu dictum varius.
|
207;2;bob;What is this?;Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Libero justo laoreet sit amet cursus sit amet dictum. Nunc faucibus a pellentesque sit. Dis parturient montes nascetur ridiculus mus mauris vitae ultricies. Enim nunc faucibus a pellentesque. Commodo quis imperdiet massa tincidunt nunc pulvinar sapien. Cras ornare arcu dui vivamus. Facilisi etiam dignissim diam quis enim lobortis. Et molestie ac feugiat sed. Urna neque viverra justo nec ultrices dui. Ullamcorper a lacus vestibulum sed arcu non. Volutpat ac tincidunt vitae semper quis. Dignissim sodales ut eu sem. Feugiat in fermentum posuere urna nec. At augue eget arcu dictum varius.
|
||||||
1689144d-3b10-4849-bcbe-2408a13e161d;251;3;bob;It's dark...;Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Suscipit tellus mauris a diam. Velit aliquet sagittis id consectetur purus ut. Viverra adipiscing at in tellus integer. Vitae elementum curabitur vitae nunc. Mattis ullamcorper velit sed ullamcorper morbi. Diam quis enim lobortis scelerisque. Auctor neque vitae tempus quam pellentesque nec nam aliquam. Semper auctor neque vitae tempus. Quis eleifend quam adipiscing vitae proin. Neque convallis a cras semper auctor neque vitae. Imperdiet massa tincidunt nunc pulvinar sapien et ligula. Sit amet consectetur adipiscing elit ut aliquam purus. Pretium quam vulputate dignissim suspendisse.
|
251;3;bob;It's dark...;Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua. Suscipit tellus mauris a diam. Velit aliquet sagittis id consectetur purus ut. Viverra adipiscing at in tellus integer. Vitae elementum curabitur vitae nunc. Mattis ullamcorper velit sed ullamcorper morbi. Diam quis enim lobortis scelerisque. Auctor neque vitae tempus quam pellentesque nec nam aliquam. Semper auctor neque vitae tempus. Quis eleifend quam adipiscing vitae proin. Neque convallis a cras semper auctor neque vitae. Imperdiet massa tincidunt nunc pulvinar sapien et ligula. Sit amet consectetur adipiscing elit ut aliquam purus. Pretium quam vulputate dignissim suspendisse.
|
||||||
|
@@ -7,28 +7,17 @@
|
|||||||
"index.cds"
|
"index.cds"
|
||||||
],
|
],
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@cap-js/hana": "^1.7.0",
|
|
||||||
"@sap/cds": ">=5",
|
"@sap/cds": ">=5",
|
||||||
"@sap/xssec": "^4.4.0",
|
|
||||||
"express": "^4.17.1"
|
"express": "^4.17.1"
|
||||||
},
|
},
|
||||||
"scripts": {
|
|
||||||
"start": "cds-serve"
|
|
||||||
},
|
|
||||||
"cds": {
|
"cds": {
|
||||||
"requires": {
|
"requires": {
|
||||||
"messaging": {
|
"messaging": {
|
||||||
"[development]": {
|
"[development]": { "kind": "file-based-messaging" },
|
||||||
"kind": "file-based-messaging"
|
"[hybrid]": { "kind": "enterprise-messaging-shared" },
|
||||||
|
"[production]": { "kind": "enterprise-messaging" }
|
||||||
},
|
},
|
||||||
"[hybrid]": {
|
"db": { "kind": "sql" }
|
||||||
"kind": "enterprise-messaging-shared"
|
|
||||||
},
|
|
||||||
"[production]": {
|
|
||||||
"kind": "enterprise-messaging"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"db": true
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -15,7 +15,7 @@ module.exports = cds.service.impl (function(){
|
|||||||
const { count, rating } = await cds.tx(req) .run (
|
const { count, rating } = await cds.tx(req) .run (
|
||||||
SELECT.one `round(avg(rating),2) as rating, count(*) as count` .from (Reviews) .where ({subject})
|
SELECT.one `round(avg(rating),2) as rating, count(*) as count` .from (Reviews) .where ({subject})
|
||||||
)
|
)
|
||||||
global.it || console.log ('< emitting:', 'reviewed', { subject, count, rating }) // eslint-disable-line no-console
|
global.it || console.log ('< emitting:', 'reviewed', { subject, count, rating })
|
||||||
await this.emit ('reviewed', { subject, count, rating })
|
await this.emit ('reviewed', { subject, count, rating })
|
||||||
})
|
})
|
||||||
|
|
||||||
|
|||||||
10
samples.md
10
samples.md
@@ -6,6 +6,12 @@ Each sub directory essentially is an individual npm package arranged in an [all-
|
|||||||

|

|
||||||
|
|
||||||
|
|
||||||
|
## [@capire/hello-world](hello)
|
||||||
|
|
||||||
|
- A simplistic [Hello World](https://cap.cloud.sap/docs/get-started/hello-world) service using [CDS](https://cap.cloud.sap/docs/cds/) and [cds.services](https://cap.cloud.sap/docs/node.js/api#services-api).
|
||||||
|
- [Typescript support](https://cap.cloud.sap/docs/node.js/typescript)
|
||||||
|
|
||||||
|
|
||||||
## [@capire/bookshop](bookshop)
|
## [@capire/bookshop](bookshop)
|
||||||
|
|
||||||
- [Getting Started](https://cap.cloud.sap/docs/get-started/in-a-nutshell) with CAP, briefly introducing:
|
- [Getting Started](https://cap.cloud.sap/docs/get-started/in-a-nutshell) with CAP, briefly introducing:
|
||||||
@@ -54,10 +60,10 @@ Each sub directory essentially is an individual npm package arranged in an [all-
|
|||||||
- [@capire/reviews](reviews)
|
- [@capire/reviews](reviews)
|
||||||
- [@capire/orders](orders)
|
- [@capire/orders](orders)
|
||||||
- [@capire/common](common)
|
- [@capire/common](common)
|
||||||
- [@capire/data-viewer](etc/data-viewer)
|
- [@capire/data-viewer](data-viewer)
|
||||||
- [The Vue.js app](bookshop/app/vue) imported from `bookshop` is served as well
|
- [The Vue.js app](bookshop/app/vue) imported from `bookshop` is served as well
|
||||||
- [The Vue.js app](reviews/app/vue) imported from `reviews` is served as well
|
- [The Vue.js app](reviews/app/vue) imported from `reviews` is served as well
|
||||||
- [The Vue.js app](etc/data-viewer/app/data) imported from `data-viewer` is served as well
|
- [The Vue.js app](data-viewer/app/data) imported from `data-viewer` is served as well
|
||||||
- [The Fiori app](orders/app) imported from `orders` is served as well
|
- [The Fiori app](orders/app) imported from `orders` is served as well
|
||||||
- [OpenAPI export + Swagger UI](https://cap.cloud.sap/docs/advanced/openapi)
|
- [OpenAPI export + Swagger UI](https://cap.cloud.sap/docs/advanced/openapi)
|
||||||
|
|
||||||
|
|||||||
@@ -1,3 +0,0 @@
|
|||||||
using from '@capire/bookstore';
|
|
||||||
using from '@capire/reviews';
|
|
||||||
using from '@capire/orders';
|
|
||||||
@@ -1,17 +0,0 @@
|
|||||||
{
|
|
||||||
"name": "shared-db",
|
|
||||||
"version": "1.0.0",
|
|
||||||
"main": "index.js",
|
|
||||||
"scripts": {
|
|
||||||
"test": "echo \"Error: no test specified\" && exit 1"
|
|
||||||
},
|
|
||||||
"keywords": [],
|
|
||||||
"author": "",
|
|
||||||
"license": "ISC",
|
|
||||||
"description": "",
|
|
||||||
"cds": {
|
|
||||||
"sql": {
|
|
||||||
"native_hana_associations": false
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,8 +1,7 @@
|
|||||||
const cds = require('@sap/cds')
|
|
||||||
const { expect } = cds.test
|
|
||||||
|
|
||||||
describe('cds.ql → cqn', () => {
|
describe('cds.ql → cqn', () => {
|
||||||
|
|
||||||
|
const cds = require('@sap/cds/lib')
|
||||||
|
const { expect } = cds.test
|
||||||
const Foo = { name: 'Foo' }
|
const Foo = { name: 'Foo' }
|
||||||
const Books = { name: 'capire.bookshop.Books' }
|
const Books = { name: 'capire.bookshop.Books' }
|
||||||
|
|
||||||
@@ -677,7 +676,7 @@ describe('cds.ql → cqn', () => {
|
|||||||
.to.eql(INSERT.into(Foo).entries(...entries))
|
.to.eql(INSERT.into(Foo).entries(...entries))
|
||||||
.to.eql(INSERT.into(Foo).entries(entries))
|
.to.eql(INSERT.into(Foo).entries(entries))
|
||||||
.to.eql({
|
.to.eql({
|
||||||
INSERT: { into: { ref: ['Foo'] }, entries },
|
INSERT: { into: cds.env.ql.quirks_mode ? 'Foo' : { ref: ['Foo'] }, entries },
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
@@ -693,7 +692,7 @@ describe('cds.ql → cqn', () => {
|
|||||||
.to.eql(INSERT.into(Foo).columns('a', 'b').rows([1, 2], [3, 4]))
|
.to.eql(INSERT.into(Foo).columns('a', 'b').rows([1, 2], [3, 4]))
|
||||||
.to.eql({
|
.to.eql({
|
||||||
INSERT: {
|
INSERT: {
|
||||||
into: { ref: ['Foo'] },
|
into: cds.env.ql.quirks_mode ? 'Foo' : { ref: ['Foo'] },
|
||||||
columns: ['a', 'b'],
|
columns: ['a', 'b'],
|
||||||
rows: [
|
rows: [
|
||||||
[1, 2],
|
[1, 2],
|
||||||
@@ -707,7 +706,7 @@ describe('cds.ql → cqn', () => {
|
|||||||
expect(INSERT.into(Foo).columns('a', 'b').values([1, 2]))
|
expect(INSERT.into(Foo).columns('a', 'b').values([1, 2]))
|
||||||
.to.eql(INSERT.into(Foo).columns('a', 'b').values(1, 2))
|
.to.eql(INSERT.into(Foo).columns('a', 'b').values(1, 2))
|
||||||
.to.eql({
|
.to.eql({
|
||||||
INSERT: { into: { ref: ['Foo'] }, columns: ['a', 'b'], values: [1, 2] },
|
INSERT: { into: cds.env.ql.quirks_mode ? 'Foo' : { ref: ['Foo'] }, columns: ['a', 'b'], values: [1, 2] },
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
@@ -722,7 +721,7 @@ describe('cds.ql → cqn', () => {
|
|||||||
test('entity (..., <key>)', () => {
|
test('entity (..., <key>)', () => {
|
||||||
const cqnWhere = {
|
const cqnWhere = {
|
||||||
UPDATE: {
|
UPDATE: {
|
||||||
entity: { ref: ['capire.bookshop.Books'] },
|
entity: cds.env.ql.quirks_mode ? 'capire.bookshop.Books' : { ref: ['capire.bookshop.Books'] },
|
||||||
where: [{ ref: ['ID'] }, '=', { val: 4711 }],
|
where: [{ ref: ['ID'] }, '=', { val: 4711 }],
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
@@ -766,7 +765,7 @@ describe('cds.ql → cqn', () => {
|
|||||||
.to.eql(UPDATE(Foo).with({ foo: 11, bar: { '-=': 22 } }))
|
.to.eql(UPDATE(Foo).with({ foo: 11, bar: { '-=': 22 } }))
|
||||||
.to.eql({
|
.to.eql({
|
||||||
UPDATE: {
|
UPDATE: {
|
||||||
entity: { ref: ['Foo'] },
|
entity: cds.env.ql.quirks_mode ? 'Foo' : { ref: ['Foo'] },
|
||||||
data: { foo: 11 },
|
data: { foo: 11 },
|
||||||
with: {
|
with: {
|
||||||
bar: { xpr: [{ ref: ['bar'] }, '-', { val: 22 }] },
|
bar: { xpr: [{ ref: ['bar'] }, '-', { val: 22 }] },
|
||||||
@@ -777,7 +776,7 @@ describe('cds.ql → cqn', () => {
|
|||||||
// some more
|
// some more
|
||||||
expect(UPDATE(Foo).with(`bar = coalesce(x,y), car = 'foo''s bar, car'`)).to.eql({
|
expect(UPDATE(Foo).with(`bar = coalesce(x,y), car = 'foo''s bar, car'`)).to.eql({
|
||||||
UPDATE: {
|
UPDATE: {
|
||||||
entity: { ref: ['Foo'] },
|
entity: cds.env.ql.quirks_mode ? 'Foo' : { ref: ['Foo'] },
|
||||||
data: {
|
data: {
|
||||||
car: "foo's bar, car",
|
car: "foo's bar, car",
|
||||||
},
|
},
|
||||||
@@ -797,7 +796,7 @@ describe('cds.ql → cqn', () => {
|
|||||||
test('from (..., <key>)', () => {
|
test('from (..., <key>)', () => {
|
||||||
const cqnWhere = {
|
const cqnWhere = {
|
||||||
DELETE: {
|
DELETE: {
|
||||||
from: { ref: ['capire.bookshop.Books'] },
|
from: cds.env.ql.quirks_mode ? 'capire.bookshop.Books' : { ref: ['capire.bookshop.Books'] },
|
||||||
where: [{ ref: ['ID'] }, '=', { val: 4711 }],
|
where: [{ ref: ['ID'] }, '=', { val: 4711 }],
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,53 +0,0 @@
|
|||||||
const cds = require("@sap/cds");
|
|
||||||
const { expect } = cds.test(
|
|
||||||
"serve",
|
|
||||||
"CatalogService",
|
|
||||||
"--from",
|
|
||||||
"@capire/bookshop,@capire/common",
|
|
||||||
"--in-memory"
|
|
||||||
);
|
|
||||||
|
|
||||||
describe("Consuming actions locally", () => {
|
|
||||||
let cats, CatalogService, Books, stockBefore;
|
|
||||||
const BOOK_ID = 251;
|
|
||||||
const QUANTITY = 1;
|
|
||||||
|
|
||||||
before("bootstrap the database", async () => {
|
|
||||||
CatalogService = cds.services.CatalogService;
|
|
||||||
expect(CatalogService).not.to.be.undefined;
|
|
||||||
|
|
||||||
Books = CatalogService.entities.Books;
|
|
||||||
expect(Books).not.to.be.undefined;
|
|
||||||
|
|
||||||
cats = await cds.connect.to("CatalogService");
|
|
||||||
});
|
|
||||||
|
|
||||||
beforeEach(async () => {
|
|
||||||
// Read the stock before the action is called
|
|
||||||
stockBefore = (await cats.get(Books, BOOK_ID)).stock;
|
|
||||||
});
|
|
||||||
|
|
||||||
it("calls unbound actions - basic variant using srv.send", async () => {
|
|
||||||
// Use a managed transaction to create a continuation with an authenticated user
|
|
||||||
const res1 = await cats.tx({ user: "alice" }, () => {
|
|
||||||
return cats.send("submitOrder", { book: BOOK_ID, quantity: QUANTITY });
|
|
||||||
});
|
|
||||||
expect(res1.stock).to.eql(stockBefore - QUANTITY);
|
|
||||||
});
|
|
||||||
|
|
||||||
it("calls unbound actions - named args variant", async () => {
|
|
||||||
// Use a managed transaction to create a continuation with an authenticated user
|
|
||||||
const res2 = await cats.tx({ user: "alice" }, () => {
|
|
||||||
return cats.submitOrder({ book: BOOK_ID, quantity: QUANTITY });
|
|
||||||
});
|
|
||||||
expect(res2.stock).to.eql(stockBefore - QUANTITY);
|
|
||||||
});
|
|
||||||
|
|
||||||
it("calls unbound actions - positional args variant", async () => {
|
|
||||||
// Use a managed transaction to create a continuation with an authenticated user
|
|
||||||
const res3 = await cats.tx({ user: "alice" }, () => {
|
|
||||||
return cats.submitOrder(BOOK_ID, QUANTITY);
|
|
||||||
});
|
|
||||||
expect(res3.stock).to.eql(stockBefore - QUANTITY);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
const cds = require('@sap/cds')
|
const cds = require('@sap/cds/lib')
|
||||||
|
|
||||||
describe('cap/samples - Consuming Services locally', () => {
|
describe('cap/samples - Consuming Services locally', () => {
|
||||||
|
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
const cds = require('@sap/cds')
|
const cds = require('@sap/cds/lib')
|
||||||
|
|
||||||
describe('cap/samples - Custom Handlers', () => {
|
describe('cap/samples - Custom Handlers', () => {
|
||||||
|
|
||||||
@@ -8,10 +8,9 @@ describe('cap/samples - Custom Handlers', () => {
|
|||||||
})
|
})
|
||||||
|
|
||||||
it('should reject out-of-stock orders', async () => {
|
it('should reject out-of-stock orders', async () => {
|
||||||
await expect(POST `/browse/submitOrder ${{ book: 201, quantity: 5 }}`).to.be.fulfilled
|
await POST `/browse/submitOrder ${{ book: 201, quantity: 5 }}`
|
||||||
await expect(POST `/browse/submitOrder ${{ book: 201, quantity: 5 }}`).to.be.fulfilled
|
await POST `/browse/submitOrder ${{ book: 201, quantity: 5 }}`
|
||||||
await expect(POST `/browse/submitOrder ${{ book: 201, quantity: 5 }}`).to.be.rejectedWith(
|
await expect(POST `/browse/submitOrder ${{ book: 201, quantity: 5 }}`).to.be.rejectedWith(/409 - 5 exceeds stock for book #201/)
|
||||||
/409 - 5 exceeds stock for book #201/)
|
|
||||||
const { data } = await GET`/admin/Books/201/stock/$value`
|
const { data } = await GET`/admin/Books/201/stock/$value`
|
||||||
expect(data).to.equal(2)
|
expect(data).to.equal(2)
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -1,8 +1,4 @@
|
|||||||
// Quick hack: suppress deprecation warnings w/ Node22 caused by http-proxy (used by OData v2 proxy)
|
const cds = require('@sap/cds/lib')
|
||||||
// See also: https://github.com/http-party/node-http-proxy/pull/1666
|
|
||||||
require('util')._extend = Object.assign
|
|
||||||
|
|
||||||
const cds = require('@sap/cds')
|
|
||||||
|
|
||||||
describe('cap/samples - Fiori APIs - v2', function() {
|
describe('cap/samples - Fiori APIs - v2', function() {
|
||||||
|
|
||||||
|
|||||||
22
test/hello-world.test.js
Normal file
22
test/hello-world.test.js
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
const cds = require('@sap/cds/lib')
|
||||||
|
|
||||||
|
describe('cap/samples - Hello world!', () => {
|
||||||
|
|
||||||
|
const { GET, expect } = cds.test (__dirname+'/../hello')
|
||||||
|
|
||||||
|
it('should say hello with class impl', async () => {
|
||||||
|
const {data} = await GET `/say/hello(to='world')`
|
||||||
|
expect(data.value).to.eql('Hello world!')
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should say hello with another impl', async () => {
|
||||||
|
await cds.serve('say').from(cds.model)
|
||||||
|
.at('/say-again').in(cds.app)
|
||||||
|
.with(srv => {
|
||||||
|
srv.on('hello', (req) => `Hello again ${req.data.to}!`)
|
||||||
|
})
|
||||||
|
const {data} = await GET `/say-again/hello(to='world')`
|
||||||
|
expect(data.value).to.eql('Hello again world!')
|
||||||
|
})
|
||||||
|
|
||||||
|
})
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
const cds = require('@sap/cds')
|
const cds = require('@sap/cds/lib')
|
||||||
const { expect } = cds.test.in(__dirname,'..')
|
const { expect } = cds.test.in(__dirname,'..')
|
||||||
|
|
||||||
describe('cap/samples - Hierarchical Data', ()=>{
|
describe('cap/samples - Hierarchical Data', ()=>{
|
||||||
@@ -77,27 +77,29 @@ describe('cap/samples - Hierarchical Data', ()=>{
|
|||||||
)
|
)
|
||||||
})
|
})
|
||||||
|
|
||||||
it ('supports nested reads', ()=> expect (
|
it ('supports nested reads', async()=>{
|
||||||
|
expect (await
|
||||||
SELECT.one.from (Cats, c=>{
|
SELECT.one.from (Cats, c=>{
|
||||||
c.ID, c.name.as('parent'), c.children (c=>{
|
c.ID, c.name.as('parent'), c.children (c=>{
|
||||||
c.name.as('child')
|
c.name.as('child')
|
||||||
})
|
})
|
||||||
}) .where ({name:'Cat'})
|
}) .where ({name:'Cat'})
|
||||||
) .to.eventually.eql (
|
) .to.eql (
|
||||||
{ ID:101, parent:'Cat', children:[
|
{ ID:101, parent:'Cat', children:[
|
||||||
{ child:'Kitty' },
|
{ child:'Kitty' },
|
||||||
{ child:'Catwoman' },
|
{ child:'Catwoman' },
|
||||||
]}
|
]}
|
||||||
))
|
)
|
||||||
|
})
|
||||||
|
|
||||||
it ('supports deeply nested reads', ()=> expect (
|
it ('supports deeply nested reads', async()=>{
|
||||||
SELECT.one.from (Cats, c=>{
|
expect (await SELECT.one.from (Cats, c=>{
|
||||||
c.ID, c.name, c.children (
|
c.ID, c.name, c.children (
|
||||||
c => { c.name },
|
c => { c.name },
|
||||||
{levels:3}
|
{levels:3}
|
||||||
)
|
)
|
||||||
}) .where ({name:'Cat'})
|
}) .where ({name:'Cat'})
|
||||||
) .to.eventually.eql (
|
) .to.eql (
|
||||||
{ ID:101, name:'Cat', children:[
|
{ ID:101, name:'Cat', children:[
|
||||||
{ name:'Kitty', children:[
|
{ name:'Kitty', children:[
|
||||||
{ name:'Kitty Cat', children:[
|
{ name:'Kitty Cat', children:[
|
||||||
@@ -106,16 +108,18 @@ describe('cap/samples - Hierarchical Data', ()=>{
|
|||||||
{ name:'Catwoman', children:[
|
{ name:'Catwoman', children:[
|
||||||
{ name:'Catalina', children:[] } ]},
|
{ name:'Catalina', children:[] } ]},
|
||||||
]}
|
]}
|
||||||
))
|
)
|
||||||
|
})
|
||||||
|
|
||||||
it ('supports cascaded deletes', async()=>{
|
it ('supports cascaded deletes', async()=>{
|
||||||
const affectedRows = await DELETE.from (Cats) .where ({ID:[102,106]})
|
const affectedRows = await DELETE.from (Cats) .where ({ID:[102,106]})
|
||||||
expect (affectedRows) .to.be.greaterThan (0)
|
expect (affectedRows) .to.be.greaterThan (0)
|
||||||
await expect (SELECT`ID,name`.from(Cats) ).to.eventually.eql ([
|
const expected = [
|
||||||
{ ID:100, name:'Some Cats...' },
|
{ ID:100, name:'Some Cats...' },
|
||||||
{ ID:101, name:'Cat' },
|
{ ID:101, name:'Cat' },
|
||||||
{ ID:108, name:'Catweazle' }
|
{ ID:108, name:'Catweazle' }
|
||||||
])
|
]
|
||||||
|
expect ( await SELECT`ID,name`.from(Cats) ).to.eql (expected)
|
||||||
})
|
})
|
||||||
|
|
||||||
})
|
})
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
const cds = require('@sap/cds')
|
const cds = require('@sap/cds/lib')
|
||||||
|
|
||||||
describe('cap/samples - Localized Data', () => {
|
describe('cap/samples - Localized Data', () => {
|
||||||
|
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
const cds = require('@sap/cds')
|
const cds = require('@sap/cds/lib')
|
||||||
|
|
||||||
describe('cap/samples - Messaging', ()=>{
|
describe('cap/samples - Messaging', ()=>{
|
||||||
|
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
const cds = require('@sap/cds')
|
const cds = require('@sap/cds/lib')
|
||||||
|
|
||||||
describe('cap/samples - Bookshop APIs', () => {
|
describe('cap/samples - Bookshop APIs', () => {
|
||||||
const { GET, expect, axios } = cds.test ('@capire/bookshop')
|
const { GET, expect, axios } = cds.test ('@capire/bookshop')
|
||||||
@@ -8,10 +8,9 @@ describe('cap/samples - Bookshop APIs', () => {
|
|||||||
const { headers, status, data } = await GET `/browse/$metadata`
|
const { headers, status, data } = await GET `/browse/$metadata`
|
||||||
expect(status).to.equal(200)
|
expect(status).to.equal(200)
|
||||||
expect(headers).to.contain({
|
expect(headers).to.contain({
|
||||||
// 'content-type': 'application/xml', //> fails with 'application/xml;charset=utf-8', which is set by express
|
'content-type': 'application/xml',
|
||||||
'odata-version': '4.0',
|
'odata-version': '4.0',
|
||||||
})
|
})
|
||||||
expect(headers['content-type']).to.match(/application\/xml/)
|
|
||||||
expect(data).to.contain('<EntitySet Name="Books" EntityType="CatalogService.Books">')
|
expect(data).to.contain('<EntitySet Name="Books" EntityType="CatalogService.Books">')
|
||||||
expect(data).to.contain('<Annotation Term="Common.Label" String="Currency"/>')
|
expect(data).to.contain('<Annotation Term="Common.Label" String="Currency"/>')
|
||||||
})
|
})
|
||||||
@@ -29,8 +28,6 @@ describe('cap/samples - Bookshop APIs', () => {
|
|||||||
])
|
])
|
||||||
})
|
})
|
||||||
|
|
||||||
describe('query options...', () => {
|
|
||||||
|
|
||||||
it('supports $search in multiple fields', async () => {
|
it('supports $search in multiple fields', async () => {
|
||||||
const { data } = await GET `/browse/Books ${{
|
const { data } = await GET `/browse/Books ${{
|
||||||
params: { $search: 'Po', $select: `title,author` },
|
params: { $search: 'Po', $select: `title,author` },
|
||||||
@@ -89,7 +86,6 @@ describe('cap/samples - Bookshop APIs', () => {
|
|||||||
{ ID: 271, title: 'Catweazle' },
|
{ ID: 271, title: 'Catweazle' },
|
||||||
])
|
])
|
||||||
})
|
})
|
||||||
})
|
|
||||||
|
|
||||||
it('serves user info', async () => {
|
it('serves user info', async () => {
|
||||||
const { data: alice } = await GET `/user/me`
|
const { data: alice } = await GET `/user/me`
|
||||||
|
|||||||
@@ -1,34 +0,0 @@
|
|||||||
{
|
|
||||||
"scopes": [
|
|
||||||
{
|
|
||||||
"name": "$XSAPPNAME.admin",
|
|
||||||
"description": "admin"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "$XSAPPNAME.emcallback",
|
|
||||||
"description": "Enterprise-Messaging Callback Access",
|
|
||||||
"grant-as-authority-to-apps": [
|
|
||||||
"$XSSERVICENAME(samples-messaging)"
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"name": "$XSAPPNAME.emmanagement",
|
|
||||||
"description": "Enterprise-Messaging Management Access"
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"attributes": [],
|
|
||||||
"role-templates": [
|
|
||||||
{
|
|
||||||
"name": "admin",
|
|
||||||
"scope-references": [
|
|
||||||
"$XSAPPNAME.admin"
|
|
||||||
],
|
|
||||||
"description": "cap samples multi-service shared-db"
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"authorities-inheritance": false,
|
|
||||||
"authorities": [
|
|
||||||
"$XSAPPNAME.emmanagement",
|
|
||||||
"$XSAPPNAME.mtcallback"
|
|
||||||
]
|
|
||||||
}
|
|
||||||
Reference in New Issue
Block a user