const cds = require('@sap/cds') module.exports = cds.service.impl(async () => { // We are mashing up three services... const bupa = await cds.connect.to('API_BUSINESS_PARTNER') const admin = await cds.connect.to('AdminService') const db = await cds.connect.to('db') // Using reflected definitions from connected services/database const { Addresses: externalAddresses } = bupa.entities // projection on external addresses const { Books, Addresses } = db.entities // entities in local database // Delegate ValueHelp requests to S/4 backend, fetching current user's addresses from there admin.on('READ', 'Addresses', req => { console.log('Delegating to S/4 bupa service...') const UsersAddresses = SELECT.from(externalAddresses) .where({ contact: req.user.id }) .where(req.query.SELECT.where) return bupa.tx(req).run(UsersAddresses) }) // Replicate chosen addresses from S/4 when filling orders. admin.before('PATCH', 'Orders', async req => { const assigned = { ID: req.data.shippingAddress_ID, contact: req.user.id } if (!assigned.ID) return //> something else const local = db.transaction(req) const [replica] = await local.read(Addresses).where(assigned) if (replica) return //> already replicated const [address] = await bupa .tx(req) .run(SELECT.from(externalAddresses).where(assigned)) if (address) return local.create(Addresses).entries(address) }) // Subscribe to S/4 event to update local replicas when sources change in S/4. bupa.on('BusinessPartner/Changed', async msg => { console.log('>> received:', msg.data) const contact = msg.data.KEY[0].BUSINESSPARTNER //> S/4HANA's weird payload format const { SELECT, UPDATE } = cds.ql(msg) //> convenient alternative to .transaction(req).run(SELECT...) // fetch affected entries from local replicas const replicas = await SELECT.from(Addresses).where({ contact }) if (replicas.length === 0) return //> not affected // fetch changed data from S/4 -> might be less than local due to deletes const externals = await SELECT.from(externalAddresses).where({ contact }) // Add a tombstone if remote address was deleted const changed = replicas.map( rep => externals.find(ext => ext.ID === rep.ID) || { ...rep, ...{ tombstone: true } } ) // update local replicas with changes from S/4 const local = db.transaction(msg) //> using that variant to benefit from bulk runs return local.run(changed.map(a => UPDATE(Addresses, a.ID).with(a))) }) // Validate incoming orders and reduce books' stocks. admin.before('CREATE', 'Orders', async req => { const { Items } = req.data // validate input... if (!Items || Items.length === 0) return req.reject('Please order at least one item.') if (!req.data.shippingAddress_ID) return req.reject( 'Please enter a valid shipping address.', 'shippingAddress_ID' ) // reduce stock on ordered books... const all = await db.tx(req).run( Items.map(each => UPDATE(Books) .where('ID =', each.book_ID) .and('stock >=', each.amount) .set('stock -=', each.amount) ) ) all.forEach( (affectedRows, i) => affectedRows > 0 || req.error( 409, `${Items[i].amount} exceeds stock for book #${Items[i].book_ID}` ) ) }) }) require('./utils')