mirror of
https://github.com/overleaf/overleaf.git
synced 2024-11-21 20:47:08 -05:00
Merge pull request #19480 from overleaf/jpa-fast-path-fetch-for-transform
[document-updater] avoid fetching updates to transform when up-to-date GitOrigin-RevId: 7962d8903a7bc9b572d7c6adfd8f33ad36f30459
This commit is contained in:
parent
8e79d72cc2
commit
8c0a78c7e7
2 changed files with 53 additions and 26 deletions
|
@ -17,6 +17,12 @@ const Keys = require('./UpdateKeys')
|
||||||
const RedisManager = require('./RedisManager')
|
const RedisManager = require('./RedisManager')
|
||||||
const Errors = require('./Errors')
|
const Errors = require('./Errors')
|
||||||
|
|
||||||
|
const TRANSFORM_UPDATES_COUNT_BUCKETS = [
|
||||||
|
0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 15, 20, 25, 50, 75, 100,
|
||||||
|
// prepare buckets for full-project history/larger buffer experiments
|
||||||
|
150, 200, 300, 400,
|
||||||
|
]
|
||||||
|
|
||||||
module.exports = ShareJsDB = class ShareJsDB {
|
module.exports = ShareJsDB = class ShareJsDB {
|
||||||
constructor(projectId, docId, lines, version) {
|
constructor(projectId, docId, lines, version) {
|
||||||
this.project_id = projectId
|
this.project_id = projectId
|
||||||
|
@ -31,11 +37,18 @@ module.exports = ShareJsDB = class ShareJsDB {
|
||||||
}
|
}
|
||||||
|
|
||||||
getOps(docKey, start, end, callback) {
|
getOps(docKey, start, end, callback) {
|
||||||
if (start === end) {
|
if (start === end || (start === this.version && end === null)) {
|
||||||
|
const status = 'is-up-to-date'
|
||||||
Metrics.inc('transform-updates', 1, {
|
Metrics.inc('transform-updates', 1, {
|
||||||
status: 'is-up-to-date',
|
status,
|
||||||
path: 'sharejs',
|
path: 'sharejs',
|
||||||
})
|
})
|
||||||
|
Metrics.histogram(
|
||||||
|
'transform-updates.count',
|
||||||
|
0,
|
||||||
|
TRANSFORM_UPDATES_COUNT_BUCKETS,
|
||||||
|
{ path: 'sharejs', status }
|
||||||
|
)
|
||||||
return callback(null, [])
|
return callback(null, [])
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -64,35 +77,27 @@ module.exports = ShareJsDB = class ShareJsDB {
|
||||||
} else {
|
} else {
|
||||||
if (ops.length === 0) {
|
if (ops.length === 0) {
|
||||||
status = 'fetched-zero'
|
status = 'fetched-zero'
|
||||||
|
|
||||||
|
// The sharejs processing is happening under a lock.
|
||||||
|
// In case there are no other ops available, something bypassed the lock (or we overran it).
|
||||||
|
logger.warn(
|
||||||
|
{
|
||||||
|
projectId,
|
||||||
|
docId,
|
||||||
|
start,
|
||||||
|
end,
|
||||||
|
timeSinceShareJsDBInit:
|
||||||
|
performance.now() - this.startTimeShareJsDB,
|
||||||
|
},
|
||||||
|
'found zero docOps while transforming update'
|
||||||
|
)
|
||||||
} else {
|
} else {
|
||||||
status = 'fetched'
|
status = 'fetched'
|
||||||
|
|
||||||
if (start === this.version && end === -1) {
|
|
||||||
// The sharejs processing is happening under a lock.
|
|
||||||
// this.version is the version as read from redis under lock.
|
|
||||||
// In case there are any new ops available, something bypassed the lock (or we overran it).
|
|
||||||
logger.warn(
|
|
||||||
{
|
|
||||||
projectId,
|
|
||||||
docId,
|
|
||||||
start,
|
|
||||||
nOps: ops.length,
|
|
||||||
timeSinceShareJsDBInit:
|
|
||||||
performance.now() - this.startTimeShareJsDB,
|
|
||||||
},
|
|
||||||
'concurrent update of docOps while transforming update'
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
Metrics.histogram(
|
Metrics.histogram(
|
||||||
'transform-updates.count',
|
'transform-updates.count',
|
||||||
ops.length,
|
ops.length,
|
||||||
[
|
TRANSFORM_UPDATES_COUNT_BUCKETS,
|
||||||
0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 15, 20, 25, 50, 75, 100,
|
|
||||||
// prepare buckets for full-project history/larger buffer experiments
|
|
||||||
150,
|
|
||||||
200, 300, 400,
|
|
||||||
],
|
|
||||||
{ path: 'sharejs', status }
|
{ path: 'sharejs', status }
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
|
@ -24,7 +24,9 @@ describe('ShareJsDB', function () {
|
||||||
this.callback = sinon.stub()
|
this.callback = sinon.stub()
|
||||||
this.ShareJsDB = SandboxedModule.require(modulePath, {
|
this.ShareJsDB = SandboxedModule.require(modulePath, {
|
||||||
requires: {
|
requires: {
|
||||||
'./RedisManager': (this.RedisManager = {}),
|
'./RedisManager': (this.RedisManager = {
|
||||||
|
getPreviousDocOps: sinon.stub(),
|
||||||
|
}),
|
||||||
'./Errors': Errors,
|
'./Errors': Errors,
|
||||||
'@overleaf/metrics': {
|
'@overleaf/metrics': {
|
||||||
inc: sinon.stub(),
|
inc: sinon.stub(),
|
||||||
|
@ -87,11 +89,31 @@ describe('ShareJsDB', function () {
|
||||||
return this.db.getOps(this.doc_key, this.start, this.end, this.callback)
|
return this.db.getOps(this.doc_key, this.start, this.end, this.callback)
|
||||||
})
|
})
|
||||||
|
|
||||||
|
it('should not talk to redis', function () {
|
||||||
|
this.RedisManager.getPreviousDocOps.should.not.have.been.called
|
||||||
|
})
|
||||||
|
|
||||||
return it('should return an empty array', function () {
|
return it('should return an empty array', function () {
|
||||||
return this.callback.calledWith(null, []).should.equal(true)
|
return this.callback.calledWith(null, []).should.equal(true)
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
describe('with start == redis-version and end unset', function () {
|
||||||
|
beforeEach(function () {
|
||||||
|
const start = this.version
|
||||||
|
const end = null
|
||||||
|
this.db.getOps(this.doc_key, start, end, this.callback)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should not talk to redis', function () {
|
||||||
|
this.RedisManager.getPreviousDocOps.should.not.have.been.called
|
||||||
|
})
|
||||||
|
|
||||||
|
it('should return an empty array', function () {
|
||||||
|
this.callback.should.have.been.calledWith(null, [])
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
describe('with a non empty range', function () {
|
describe('with a non empty range', function () {
|
||||||
beforeEach(function () {
|
beforeEach(function () {
|
||||||
this.start = 35
|
this.start = 35
|
||||||
|
|
Loading…
Reference in a new issue