Skip to content

Commit 163480b

Browse files
authored
feat(forUpdate): ignore locked (#1074)
Unofficial, needed for outbox/task queue
1 parent 9bd9b0a commit 163480b

File tree

4 files changed

+31
-7
lines changed

4 files changed

+31
-7
lines changed

db-service/lib/cqn2sql.js

+2-1
Original file line numberDiff line numberDiff line change
@@ -449,9 +449,10 @@ class CQN2SQLRenderer {
449449
* @returns {string} SQL
450450
*/
451451
forUpdate(update) {
452-
const { wait, of } = update
452+
const { wait, of, ignoreLocked } = update
453453
let sql = 'FOR UPDATE'
454454
if (!_empty(of)) sql += ` OF ${of.map(x => this.expr(x)).join(', ')}`
455+
if (ignoreLocked) sql += ' IGNORE LOCKED'
455456
if (typeof wait === 'number') sql += ` WAIT ${wait}`
456457
return sql
457458
}

hana/lib/HANAService.js

+8
Original file line numberDiff line numberDiff line change
@@ -157,6 +157,14 @@ class HANAService extends SQLService {
157157
const resultQuery = query.clone()
158158
resultQuery.SELECT.forUpdate = undefined
159159
resultQuery.SELECT.forShareLock = undefined
160+
const keys = Object.keys(req.target.keys || {})
161+
if (keys.length && query.SELECT.forUpdate.ignoreLocked) {
162+
// REVISIT: No support for count
163+
// where [keys] in [values]
164+
const left = { list: keys.map(k => ({ ref: [k] })) }
165+
const right = { list: rows.map(r => ({ list: keys.map(k => ({ val: r[k.toUpperCase()] })) })) }
166+
resultQuery.SELECT.where = [left, 'in', right]
167+
}
160168
return this.onSELECT({ query: resultQuery, __proto__: req })
161169
}
162170

postgres/lib/PostgresService.js

+5-3
Original file line numberDiff line numberDiff line change
@@ -466,9 +466,11 @@ GROUP BY k
466466
// Postgres does not support locking columns only tables which makes of unapplicable
467467
// Postgres does not support "wait n" it only supports "nowait"
468468
forUpdate(update) {
469-
const { wait } = update
470-
if (wait === 0) return 'FOR UPDATE NOWAIT'
471-
return 'FOR UPDATE'
469+
const { wait, ignoreLocked } = update
470+
let sql = 'FOR UPDATE'
471+
if (wait === 0) sql += ' NOWAIT'
472+
if (ignoreLocked) sql += ' SKIP LOCKED'
473+
return sql
472474
}
473475

474476
forShareLock(lock) {

test/compliance/SELECT.test.js

+16-3
Original file line numberDiff line numberDiff line change
@@ -752,7 +752,7 @@ describe('SELECT', () => {
752752
})
753753
})
754754

755-
const generalLockTest = (lock4, shared = false) => {
755+
const generalLockTest = (lock4, { shared = false, ignoreLocked = false } = {}) => {
756756
const isSQLite = () => cds.db.options.impl === '@cap-js/sqlite'
757757

758758
const setMax = max => {
@@ -851,7 +851,7 @@ describe('SELECT', () => {
851851
await tx1.run(lock4(false))
852852

853853
// Lock false
854-
if (shared) {
854+
if (shared || ignoreLocked) {
855855
const ret = await tx2.run(lock4(false))
856856
expect(ret).is.not.undefined
857857
} else {
@@ -885,7 +885,20 @@ describe('SELECT', () => {
885885

886886
generalLockTest(bool => boolLock.clone()
887887
.where([{ ref: ['bool'] }, '=', { val: bool }]),
888-
true
888+
{ shared: true }
889+
)
890+
})
891+
892+
describe('forUpdate ignore locked', () => {
893+
const boolLock = SELECT.from('basic.projection.globals')
894+
.forShareLock({
895+
of: ['bool'],
896+
ignoreLocked: true
897+
})
898+
899+
generalLockTest(bool => boolLock.clone()
900+
.where([{ ref: ['bool'] }, '=', { val: bool }]),
901+
{ ignoreLocked: true }
889902
)
890903
})
891904

0 commit comments

Comments
 (0)