case
This commit is contained in:
@@ -1,2 +0,0 @@
|
|||||||
cds.requires.messaging.kind = file-based-messaging
|
|
||||||
PORT = 4006
|
|
||||||
@@ -1,92 +0,0 @@
|
|||||||
|
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////
|
|
||||||
//
|
|
||||||
// Note: this is designed for the OrdersService being co-located with
|
|
||||||
// bookshop. It does not work if OrdersService is run as a separate
|
|
||||||
// process, and is not intended to do so.
|
|
||||||
//
|
|
||||||
////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
using { OrdersService } from '../srv/orders-service';
|
|
||||||
|
|
||||||
|
|
||||||
@odata.draft.enabled
|
|
||||||
annotate OrdersService.Orders with @(
|
|
||||||
UI: {
|
|
||||||
SelectionFields: [ createdAt, createdBy ],
|
|
||||||
LineItem: [
|
|
||||||
{Value: OrderNo, Label:'OrderNo'},
|
|
||||||
{Value: buyer, Label:'Customer'},
|
|
||||||
{Value: createdAt, Label:'Date'}
|
|
||||||
],
|
|
||||||
HeaderInfo: {
|
|
||||||
TypeName: 'Order', TypeNamePlural: 'Orders',
|
|
||||||
Title: {
|
|
||||||
Label: 'Order number ', //A label is possible but it is not considered on the ObjectPage yet
|
|
||||||
Value: OrderNo
|
|
||||||
},
|
|
||||||
Description: {Value: createdBy}
|
|
||||||
},
|
|
||||||
Identification: [ //Is the main field group
|
|
||||||
{Value: createdBy, Label:'Customer'},
|
|
||||||
{Value: createdAt, Label:'Date'},
|
|
||||||
{Value: OrderNo },
|
|
||||||
],
|
|
||||||
HeaderFacets: [
|
|
||||||
{$Type: 'UI.ReferenceFacet', Label: '{i18n>Created}', Target: '@UI.FieldGroup#Created'},
|
|
||||||
{$Type: 'UI.ReferenceFacet', Label: '{i18n>Modified}', Target: '@UI.FieldGroup#Modified'},
|
|
||||||
],
|
|
||||||
Facets: [
|
|
||||||
{$Type: 'UI.ReferenceFacet', Label: '{i18n>Details}', Target: '@UI.FieldGroup#Details'},
|
|
||||||
{$Type: 'UI.ReferenceFacet', Label: '{i18n>OrderItems}', Target: 'Items/@UI.LineItem'},
|
|
||||||
],
|
|
||||||
FieldGroup#Details: {
|
|
||||||
Data: [
|
|
||||||
{Value: currency.code, Label:'Currency'}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
FieldGroup#Created: {
|
|
||||||
Data: [
|
|
||||||
{Value: createdBy},
|
|
||||||
{Value: createdAt},
|
|
||||||
]
|
|
||||||
},
|
|
||||||
FieldGroup#Modified: {
|
|
||||||
Data: [
|
|
||||||
{Value: modifiedBy},
|
|
||||||
{Value: modifiedAt},
|
|
||||||
]
|
|
||||||
},
|
|
||||||
},
|
|
||||||
) {
|
|
||||||
createdAt @UI.HiddenFilter:false;
|
|
||||||
createdBy @UI.HiddenFilter:false;
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
annotate OrdersService.Orders.Items with @(
|
|
||||||
UI: {
|
|
||||||
LineItem: [
|
|
||||||
{Value: product_ID, Label:'Product ID'},
|
|
||||||
{Value: title, Label:'Product Title'},
|
|
||||||
{Value: price, Label:'Unit Price'},
|
|
||||||
{Value: quantity, Label:'Quantity'},
|
|
||||||
],
|
|
||||||
Identification: [ //Is the main field group
|
|
||||||
{Value: quantity, Label:'Quantity'},
|
|
||||||
{Value: title, Label:'Product'},
|
|
||||||
{Value: price, Label:'Unit Price'},
|
|
||||||
],
|
|
||||||
Facets: [
|
|
||||||
{$Type: 'UI.ReferenceFacet', Label: '{i18n>OrderItems}', Target: '@UI.Identification'},
|
|
||||||
],
|
|
||||||
},
|
|
||||||
) {
|
|
||||||
quantity @(
|
|
||||||
Common.FieldControl: #Mandatory
|
|
||||||
);
|
|
||||||
};
|
|
||||||
@@ -1,39 +0,0 @@
|
|||||||
<!DOCTYPE html>
|
|
||||||
<html>
|
|
||||||
<head>
|
|
||||||
|
|
||||||
<meta http-equiv="X-UA-Compatible" content="IE=edge" />
|
|
||||||
<meta http-equiv="Content-Type" content="text/html;charset=UTF-8" />
|
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
|
||||||
<title>Bookshop</title>
|
|
||||||
|
|
||||||
<script>
|
|
||||||
window["sap-ushell-config"] = {
|
|
||||||
defaultRenderer: "fiori2",
|
|
||||||
applications: {
|
|
||||||
"manage-orders": {
|
|
||||||
title: "Manage Orders",
|
|
||||||
description: "... testing FE v42",
|
|
||||||
additionalInformation: "SAPUI5.Component=orders",
|
|
||||||
applicationType : "URL",
|
|
||||||
url: "/orders/webapp",
|
|
||||||
navigationMode: "embedded"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
</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://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-theme="sap_horizon"
|
|
||||||
data-sap-ui-frameOptions="allow"
|
|
||||||
></script>
|
|
||||||
<script>
|
|
||||||
sap.ui.getCore().attachInit(()=> sap.ushell.Container.createRenderer().placeAt("content"))
|
|
||||||
</script>
|
|
||||||
|
|
||||||
</head>
|
|
||||||
<body class="sapUiBody" id="content"></body>
|
|
||||||
</html>
|
|
||||||
@@ -1,8 +0,0 @@
|
|||||||
sap.ui.define(["sap/fe/core/AppComponent"], function(AppComponent) {
|
|
||||||
"use strict";
|
|
||||||
return AppComponent.extend("orders.Component", {
|
|
||||||
metadata: { manifest: "json" }
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
/* eslint no-undef:0 */
|
|
||||||
@@ -1,11 +0,0 @@
|
|||||||
# This is the resource bundle of itelo
|
|
||||||
# __ldi.translation.uuid=c3431418-9caf-11e8-98d0-529269fb1459
|
|
||||||
|
|
||||||
# JCI app descriptor contains lower case TITLE
|
|
||||||
appTitle=Bookshop Sample
|
|
||||||
|
|
||||||
# JCI app descriptor contains lower case DESCRIPTION
|
|
||||||
appSubTitle=CAP Sample Application
|
|
||||||
|
|
||||||
# JCI app descriptor contains lower case DESCRIPTION
|
|
||||||
appDescription=CDS Sample Service
|
|
||||||
@@ -1,170 +0,0 @@
|
|||||||
{
|
|
||||||
"_version": "1.8.0",
|
|
||||||
"sap.app": {
|
|
||||||
"id": "orders",
|
|
||||||
"type": "application",
|
|
||||||
"title": "Order Books",
|
|
||||||
"description": "Sample Application",
|
|
||||||
"i18n": "i18n/i18n.properties",
|
|
||||||
"dataSources": {
|
|
||||||
"OrdersService": {
|
|
||||||
"uri": "/orders/",
|
|
||||||
"type": "OData",
|
|
||||||
"settings": {
|
|
||||||
"odataVersion": "4.0"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"-sourceTemplate": {
|
|
||||||
"id": "ui5template.basicSAPUI5ApplicationProject",
|
|
||||||
"-id": "ui5template.smartTemplate",
|
|
||||||
"-version": "1.40.12"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"sap.ui5": {
|
|
||||||
"dependencies": {
|
|
||||||
"libs": {
|
|
||||||
"sap.fe.templates": {}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"models": {
|
|
||||||
"i18n": {
|
|
||||||
"type": "sap.ui.model.resource.ResourceModel",
|
|
||||||
"uri": "i18n/i18n.properties"
|
|
||||||
},
|
|
||||||
"": {
|
|
||||||
"dataSource": "OrdersService",
|
|
||||||
"settings": {
|
|
||||||
"synchronizationMode": "None",
|
|
||||||
"operationMode": "Server",
|
|
||||||
"autoExpandSelect" : true,
|
|
||||||
"earlyRequests": true,
|
|
||||||
"groupProperties": {
|
|
||||||
"default": {
|
|
||||||
"submit": "Auto"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"routing": {
|
|
||||||
"routes": [
|
|
||||||
{
|
|
||||||
"pattern": ":?query:",
|
|
||||||
"name": "OrdersList",
|
|
||||||
"target": "OrdersList"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pattern": "Orders({key}):?query:",
|
|
||||||
"name": "OrdersDetails",
|
|
||||||
"target": "OrdersDetails"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pattern": "Orders({boo})/Items({boo2}):?query:",
|
|
||||||
"name": "OrderItemsDetails",
|
|
||||||
"target": "OrderItemsDetails"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"pattern": "Books({key}):?query:",
|
|
||||||
"name": "BooksDetails",
|
|
||||||
"target": "BooksDetails"
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"targets": {
|
|
||||||
"OrdersList": {
|
|
||||||
"type": "Component",
|
|
||||||
"id": "OrdersList",
|
|
||||||
"name": "sap.fe.templates.ListReport",
|
|
||||||
"options": {
|
|
||||||
"settings" : {
|
|
||||||
"entitySet" : "Orders",
|
|
||||||
"navigation" : {
|
|
||||||
"Orders" : {
|
|
||||||
"detail" : {
|
|
||||||
"route" : "OrdersDetails"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"OrdersDetails": {
|
|
||||||
"type": "Component",
|
|
||||||
"id": "OrdersDetails",
|
|
||||||
"name": "sap.fe.templates.ObjectPage",
|
|
||||||
"options": {
|
|
||||||
"settings" : {
|
|
||||||
"entitySet": "Orders",
|
|
||||||
"navigation" : {
|
|
||||||
"Items": {
|
|
||||||
"detail": {
|
|
||||||
"route": "OrderItemsDetails"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"book": {
|
|
||||||
"detail": {
|
|
||||||
"route": "BooksDetails"
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"dummy": {
|
|
||||||
"detail": {
|
|
||||||
"route": "BooksDetails"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"OrderItemsDetails": {
|
|
||||||
"type": "Component",
|
|
||||||
"id": "OrderItemsDetails",
|
|
||||||
"name": "sap.fe.templates.ObjectPage",
|
|
||||||
"options": {
|
|
||||||
"settings" : {
|
|
||||||
"entitySet": "Orders_Items"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"BooksDetails": {
|
|
||||||
"type": "Component",
|
|
||||||
"id": "BooksDetails",
|
|
||||||
"name": "sap.fe.templates.ObjectPage",
|
|
||||||
"options": {
|
|
||||||
"settings" : {
|
|
||||||
"entitySet": "Books",
|
|
||||||
"navigation": {
|
|
||||||
"author": {
|
|
||||||
"detail": {
|
|
||||||
"route": "AuthorsDetails"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"AuthorsDetails": {
|
|
||||||
"type": "Component",
|
|
||||||
"id": "AuthorsDetails",
|
|
||||||
"name": "sap.fe.templates.ObjectPage",
|
|
||||||
"options": {
|
|
||||||
"settings" : {
|
|
||||||
"entitySet": "Authors"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"contentDensities": {
|
|
||||||
"compact": true,
|
|
||||||
"cozy": true
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"sap.ui": {
|
|
||||||
"technology": "UI5",
|
|
||||||
"fullWidth": false
|
|
||||||
},
|
|
||||||
"sap.fiori": {
|
|
||||||
"registrationIds": [],
|
|
||||||
"archeType": "transactional"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,24 +0,0 @@
|
|||||||
using { Currency, User, managed, cuid } from '@sap/cds/common';
|
|
||||||
namespace sap.capire.orders;
|
|
||||||
|
|
||||||
entity Orders : cuid, managed {
|
|
||||||
OrderNo : String @title:'Order Number'; //> readable key
|
|
||||||
Items : Composition of many {
|
|
||||||
key ID : UUID;
|
|
||||||
product : Association to Products;
|
|
||||||
quantity : Integer;
|
|
||||||
title : String; //> intentionally replicated as snapshot from product.title
|
|
||||||
price : Double; //> materialized calculated field
|
|
||||||
};
|
|
||||||
buyer : User;
|
|
||||||
currency : Currency;
|
|
||||||
}
|
|
||||||
|
|
||||||
/** This is a stand-in for arbitrary ordered Products */
|
|
||||||
entity Products @(cds.persistence.skip:'always') {
|
|
||||||
key ID : String;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
// this is to ensure we have filled-in currencies
|
|
||||||
using from '@capire/common';
|
|
||||||
@@ -1,3 +0,0 @@
|
|||||||
ID;createdAt;createdBy;buyer;OrderNo;currency_code
|
|
||||||
7e2f2640-6866-4dcf-8f4d-3027aa831cad;2019-01-31;john.doe@test.com;john.doe@test.com;1;EUR
|
|
||||||
64e718c9-ff99-47f1-8ca3-950c850777d4;2019-01-30;jane.doe@test.com;jane.doe@test.com;2;EUR
|
|
||||||
|
@@ -1,4 +0,0 @@
|
|||||||
ID;Header_ID;quantity;product_ID;title;price
|
|
||||||
58040e66-1dcd-4ffb-ab10-fdce32028b79;7e2f2640-6866-4dcf-8f4d-3027aa831cad;10;201;Wuthering Heights;11.11
|
|
||||||
64e718c9-ff99-47f1-8ca3-950c850777d4;7e2f2640-6866-4dcf-8f4d-3027aa831cad;501;271;Catweazle;15
|
|
||||||
e9641166-e050-4261-bfee-d1e797e6cb7f;64e718c9-ff99-47f1-8ca3-950c850777d4;499;252;Eleonora;28
|
|
||||||
|
@@ -1,32 +0,0 @@
|
|||||||
using { Currency, User, managed, cuid } from '@sap/cds/common';
|
|
||||||
// using {Orders, OrderItems} from './schema';
|
|
||||||
|
|
||||||
namespace sap.capire.orders;
|
|
||||||
|
|
||||||
entity Products {
|
|
||||||
key ID : String;
|
|
||||||
}
|
|
||||||
|
|
||||||
entity OrdersHeaders : managed {
|
|
||||||
key ID : UUID;
|
|
||||||
OrderNo : String @title:'Order Number'; //> readable key
|
|
||||||
buyer : User;
|
|
||||||
currency : Currency;
|
|
||||||
Items : Composition of many OrdersItems on Items.Header = $self;
|
|
||||||
}
|
|
||||||
|
|
||||||
entity OrdersItems {
|
|
||||||
key ID : UUID;
|
|
||||||
product : Association to Products;
|
|
||||||
quantity : Integer;
|
|
||||||
title : String; //> intentionally replicated as snapshot from product.title
|
|
||||||
price : Double; //> materialized calculated field
|
|
||||||
Header : Association to OrdersHeaders;
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
/*
|
|
||||||
This model controls what gets exposed
|
|
||||||
*/
|
|
||||||
namespace sap.capire.orders;
|
|
||||||
using from './srv/orders-service';
|
|
||||||
using from './db/schema';
|
|
||||||
@@ -1,8 +0,0 @@
|
|||||||
{
|
|
||||||
"name": "@capire/orders",
|
|
||||||
"version": "1.0.0",
|
|
||||||
"dependencies": {
|
|
||||||
"@capire/common": "*",
|
|
||||||
"@sap/cds": ">=5"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,5 +0,0 @@
|
|||||||
using { sap.capire.orders as my } from '../db/schema';
|
|
||||||
|
|
||||||
service OrdersService {
|
|
||||||
entity Orders as projection on my.Orders;
|
|
||||||
}
|
|
||||||
@@ -1,87 +0,0 @@
|
|||||||
using { sap.capire.orders as my } from '../db/schema';
|
|
||||||
|
|
||||||
service OrdersService {
|
|
||||||
entity OrdersHeaders as projection on my.OrdersHeaders;
|
|
||||||
entity OrdersItems as projection on my.OrdersItems;
|
|
||||||
entity Products as projection on my.Products;
|
|
||||||
|
|
||||||
|
|
||||||
// static
|
|
||||||
view OrdersItemsViewJoin as select
|
|
||||||
|
|
||||||
OrdersHeaders.ID as Header_ID,
|
|
||||||
OrdersHeaders.OrderNo as OrderNo,
|
|
||||||
OrdersHeaders.buyer as buyer,
|
|
||||||
OrdersHeaders.currency as currency,
|
|
||||||
key OrdersItems.ID as Item_ID,
|
|
||||||
OrdersItems.product as product,
|
|
||||||
OrdersItems.quantity as quantity,
|
|
||||||
OrdersItems.title as title,
|
|
||||||
OrdersItems.price as price
|
|
||||||
|
|
||||||
from OrdersHeaders JOIN OrdersItems on OrdersHeaders.ID = OrdersItems.Header.ID;
|
|
||||||
|
|
||||||
// dynamic entity
|
|
||||||
entity OrderItemsViewAssoc as projection on OrdersHeaders;
|
|
||||||
|
|
||||||
// sort on right table
|
|
||||||
view SortedOrdersJoin as select
|
|
||||||
OrdersHeaders.ID as Header_ID,
|
|
||||||
OrdersHeaders.OrderNo as OrderNo,
|
|
||||||
OrdersHeaders.buyer as buyer,
|
|
||||||
OrdersHeaders.currency as currency,
|
|
||||||
key OrdersItems.ID as Item_ID,
|
|
||||||
OrdersItems.product as product,
|
|
||||||
OrdersItems.quantity as quantity,
|
|
||||||
OrdersItems.title as title,
|
|
||||||
OrdersItems.price as price
|
|
||||||
from OrdersHeaders JOIN OrdersItems on OrdersHeaders.ID = OrdersItems.Header.ID
|
|
||||||
order by title;
|
|
||||||
|
|
||||||
// sort on items and join back to header via assoc
|
|
||||||
|
|
||||||
view SortedOrdersAssoc as select
|
|
||||||
from OrdersItems {*, Header.OrderNo, Header.buyer, Header.currency }
|
|
||||||
order by OrdersItems.title;
|
|
||||||
|
|
||||||
// filter on right table
|
|
||||||
|
|
||||||
view FilteredOrdersJoin as select
|
|
||||||
OrdersHeaders.ID as Header_ID,
|
|
||||||
OrdersHeaders.OrderNo as OrderNo,
|
|
||||||
OrdersHeaders.buyer as buyer,
|
|
||||||
OrdersHeaders.currency as currency,
|
|
||||||
key OrdersItems.ID as Item_ID,
|
|
||||||
OrdersItems.product as product,
|
|
||||||
OrdersItems.quantity as quantity,
|
|
||||||
OrdersItems.title as title,
|
|
||||||
OrdersItems.price as price
|
|
||||||
from OrdersHeaders JOIN OrdersItems on OrdersHeaders.ID = OrdersItems.Header.ID
|
|
||||||
where price > 100;
|
|
||||||
|
|
||||||
// filter on items and join back to header via assoc
|
|
||||||
|
|
||||||
view FilteredOrdersAssoc as select
|
|
||||||
from OrdersItems {*, Header.OrderNo, Header.buyer, Header.currency }
|
|
||||||
where OrdersItems.price > 100;
|
|
||||||
|
|
||||||
|
|
||||||
// TODO avoid CASE and/or JOIN: Denormalization of expensive complex structures,
|
|
||||||
// calculate on write instead of read
|
|
||||||
|
|
||||||
// CASE -> try to remodel to avoid CASE, if re-modelling is not possible,
|
|
||||||
// fill redundant fields at write
|
|
||||||
|
|
||||||
entity OrdersItemsCaseView as projection on OrdersItems {
|
|
||||||
*,
|
|
||||||
case
|
|
||||||
when quantity > 500 then 'Large'
|
|
||||||
when quantity > 100 then 'Medium'
|
|
||||||
else 'Small'
|
|
||||||
end as category : String
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
@@ -1,37 +0,0 @@
|
|||||||
const cds = require ('@sap/cds')
|
|
||||||
class OrdersService extends cds.ApplicationService {
|
|
||||||
|
|
||||||
/** register custom handlers */
|
|
||||||
init(){
|
|
||||||
const { 'Orders.Items':OrderItems } = this.entities
|
|
||||||
|
|
||||||
this.before ('UPDATE', 'Orders', async function(req) {
|
|
||||||
const { ID, Items } = req.data
|
|
||||||
if (Items) for (let { product_ID, quantity } of Items) {
|
|
||||||
const { quantity:before } = await cds.tx(req).run (
|
|
||||||
SELECT.one.from (OrderItems, oi => oi.quantity) .where ({up__ID:ID, product_ID})
|
|
||||||
)
|
|
||||||
if (quantity != before) await this.orderChanged (product_ID, quantity-before)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
this.before ('DELETE', 'Orders', async function(req) {
|
|
||||||
const { ID } = req.data
|
|
||||||
const Items = await cds.tx(req).run (
|
|
||||||
SELECT.from (OrderItems, oi => { oi.product_ID, oi.quantity }) .where ({up__ID:ID})
|
|
||||||
)
|
|
||||||
if (Items) await Promise.all (Items.map(it => this.orderChanged (it.product_ID, -it.quantity)))
|
|
||||||
})
|
|
||||||
|
|
||||||
return super.init()
|
|
||||||
}
|
|
||||||
|
|
||||||
/** order changed -> broadcast event */
|
|
||||||
orderChanged (product, deltaQuantity) {
|
|
||||||
// Emit events to inform subscribers about changes in orders
|
|
||||||
console.log ('> emitting:', 'OrderChanged', { product, deltaQuantity })
|
|
||||||
return this.emit ('OrderChanged', { product, deltaQuantity })
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
module.exports = OrdersService
|
|
||||||
@@ -3,8 +3,8 @@ using { sap.capire.performance as my } from '../db/schema';
|
|||||||
service PerformanceService {
|
service PerformanceService {
|
||||||
entity OrdersHeaders as projection on my.OrdersHeaders;
|
entity OrdersHeaders as projection on my.OrdersHeaders;
|
||||||
entity OrdersItems as projection on my.OrdersItems;
|
entity OrdersItems as projection on my.OrdersItems;
|
||||||
entity Books as projection on my.Books;
|
entity Books as projection on my.Books;
|
||||||
entity Authors as projection on my.Authors;
|
entity Authors as projection on my.Authors;
|
||||||
|
|
||||||
// static
|
// static
|
||||||
view OrdersItemsViewJoin as select
|
view OrdersItemsViewJoin as select
|
||||||
@@ -66,7 +66,7 @@ from OrdersItems {*, Header.OrderNo, Header.buyer, Header.currency }
|
|||||||
where OrdersItems.price > 100;
|
where OrdersItems.price > 100;
|
||||||
|
|
||||||
|
|
||||||
// TODO avoid CASE and/or JOIN: Denormalization of expensive complex structures,
|
// TODO avoid CASE -- Denormalization of expensive complex structures,
|
||||||
// calculate on write instead of read
|
// calculate on write instead of read
|
||||||
|
|
||||||
// CASE -> try to remodel to avoid CASE, if re-modelling is not possible,
|
// CASE -> try to remodel to avoid CASE, if re-modelling is not possible,
|
||||||
@@ -82,6 +82,14 @@ entity OrdersItemsCaseView as projection on OrdersItems {
|
|||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
extend my.OrdersItems with {
|
||||||
|
itemCategory: String enum{ Small; Medium; Large;};
|
||||||
|
// fill itemCategory at runtime in performance-service.js
|
||||||
|
}
|
||||||
|
|
||||||
|
entity OrdersItemsNoCaseView as projection on OrdersItems {
|
||||||
|
*,
|
||||||
|
itemCategory as category
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -5,6 +5,14 @@ class OrdersService extends cds.ApplicationService {
|
|||||||
init(){
|
init(){
|
||||||
const { 'Orders.Items':OrderItems } = this.entities
|
const { 'Orders.Items':OrderItems } = this.entities
|
||||||
|
|
||||||
|
// fill itemCategory at runtime
|
||||||
|
this.before (['CREATE', 'UPDATE'], async req =>{
|
||||||
|
if(req.data.quantity > 500) {req.data.itemCategory = 'Large'}
|
||||||
|
else if (req.data.quantity > 100) {req.data.itemCategory = 'Medium'}
|
||||||
|
else {req.data.itemCategory = 'Small'}
|
||||||
|
})
|
||||||
|
//
|
||||||
|
|
||||||
this.before ('UPDATE', 'Orders', async function(req) {
|
this.before ('UPDATE', 'Orders', async function(req) {
|
||||||
const { ID, Items } = req.data
|
const { ID, Items } = req.data
|
||||||
if (Items) for (let { product_ID, quantity } of Items) {
|
if (Items) for (let { product_ID, quantity } of Items) {
|
||||||
|
|||||||
Reference in New Issue
Block a user