Skip to content

Commit

Permalink
fix(Editor): separate close and disconnect functions
Browse files Browse the repository at this point in the history
* `close` is for closing the editor.
  It tries to save the document and clean everything up.
* `disconnect` is for cleaning up the current collaboration sessions.
  It will not save the document and asumes the editing will be resumed
  after a reconnect.

Move `sendRemainingSteps` out to the sync service.
Also make close in the websocket polyfill sync.
Just clean up the polyfills state.

Signed-off-by: Max <max@nextcloud.com>
  • Loading branch information
max-nextcloud authored and backportbot[bot] committed Jul 3, 2024
1 parent 82dc59c commit 73c4d69
Show file tree
Hide file tree
Showing 3 changed files with 39 additions and 36 deletions.
24 changes: 14 additions & 10 deletions src/components/Editor.vue
Original file line number Diff line number Diff line change
Expand Up @@ -361,6 +361,7 @@ export default {
}
unsubscribe('text:image-node:add', this.onAddImageNode)
unsubscribe('text:image-node:delete', this.onDeleteImageNode)
unsubscribe('text:translate-modal:show', this.showTranslateModal)
if (this.dirty) {
const timeout = new Promise((resolve) => setTimeout(resolve, 2000))
await Promise.any([timeout, this.$syncService.save()])
Expand Down Expand Up @@ -389,8 +390,6 @@ export default {

this.listenSyncServiceEvents()

this.$providers.forEach(p => p?.destroy())
this.$providers = []
const syncServiceProvider = createSyncServiceProvider({
ydoc: this.$ydoc,
syncService: this.$syncService,
Expand Down Expand Up @@ -438,7 +437,7 @@ export default {
reconnect() {
this.contentLoaded = false
this.hasConnectionIssue = false
this.close().then(this.initSession)
this.disconnect().then(this.initSession)
this.idle = false
},

Expand Down Expand Up @@ -668,14 +667,19 @@ export default {
await this.$syncService.save()
},

async disconnect() {
await this.$syncService.close()
this.unlistenSyncServiceEvents()
this.$providers.forEach(p => p?.destroy())
this.$providers = []
this.$syncService = null
// disallow editing while still showing the content
this.readOnly = true
},

async close() {
if (this.currentSession && this.$syncService) {
await this.$syncService.close()
this.unlistenSyncServiceEvents()
this.$syncService = null
// disallow editing while still showing the content
this.readOnly = true
}
await this.$syncService.sendRemainingSteps(this.$queue)
await this.disconnect()
if (this.$editor) {
try {
this.unlistenEditorEvents()
Expand Down
24 changes: 24 additions & 0 deletions src/services/SyncService.js
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,7 @@ import debounce from 'debounce'

import PollingBackend from './PollingBackend.js'
import SessionApi, { Connection } from './SessionApi.js'
import { encodeArrayBuffer } from '../helpers/base64.ts'
import { logger } from '../helpers/logger.js'

/**
Expand Down Expand Up @@ -285,6 +286,29 @@ class SyncService {
})
}

async sendRemainingSteps(queue) {
if (queue.length === 0) {
return
}
let outbox = []
const steps = queue.map(s => encodeArrayBuffer(s))
.filter(s => s < 'AQ')
const awareness = queue.map(s => encodeArrayBuffer(s))
.findLast(s => s > 'AQ') || ''
return this.sendStepsNow(() => {
const data = { steps, awareness, version: this.version }
outbox = [...queue]
logger.debug('sending final steps ', data)
return data
})?.then(() => {
// only keep the steps that were not send yet
queue.splice(0,
queue.length,
...queue.filter(s => !outbox.includes(s)),
)
}, err => logger.error(err))
}

async close() {
// Make sure to leave no pending requests behind.
this.autosave.clear()
Expand Down
27 changes: 1 addition & 26 deletions src/services/WebSocketPolyfill.js
Original file line number Diff line number Diff line change
Expand Up @@ -114,37 +114,12 @@ export default function initWebSocketPolyfill(syncService, fileId, initialSessio
}

async close() {
await this.#sendRemainingSteps()
Object.entries(this.#handlers)
.forEach(([key, value]) => syncService.off(key, value))
this.#handlers = []
syncService.close().then(() => {
this.onclose()
})
this.onclose()
logger.debug('Websocket closed')
}

#sendRemainingSteps() {
if (queue.length) {
let outbox = []
return syncService.sendStepsNow(() => {
const data = {
steps: this.#steps,
awareness: this.#awareness,
version: this.#version,
}
outbox = [...queue]
logger.debug('sending final steps ', data)
return data
})?.then(() => {
// only keep the steps that were not send yet
queue.splice(0,
queue.length,
...queue.filter(s => !outbox.includes(s)),
)
}, err => logger.error(err))
}
}

}
}

0 comments on commit 73c4d69

Please sign in to comment.