diff --git a/services/web/.gitignore b/services/web/.gitignore
index 83d4d28b65..d725b0b837 100644
--- a/services/web/.gitignore
+++ b/services/web/.gitignore
@@ -39,6 +39,7 @@ data/*
app.js
app/js/*
test/unit/js/*
+test/unit_frontend/js/*
test/smoke/js/*
test/acceptance/js/*
cookies.txt
diff --git a/services/web/Jenkinsfile b/services/web/Jenkinsfile
index 6421296330..152d980787 100644
--- a/services/web/Jenkinsfile
+++ b/services/web/Jenkinsfile
@@ -60,7 +60,7 @@ pipeline {
sh 'git config --global core.logallrefupdates false'
sh 'mv app/views/external/robots.txt public/robots.txt'
sh 'mv app/views/external/googlebdb0f8f7f4a17241.html public/googlebdb0f8f7f4a17241.html'
- sh 'npm install'
+ sh 'npm --quiet install'
sh 'npm rebuild'
// It's too easy to end up shrinkwrapping to an outdated version of translations.
// Ensure translations are always latest, regardless of shrinkwrap
@@ -71,16 +71,9 @@ pipeline {
}
}
- stage('Unit Tests') {
+ stage('Test') {
steps {
- sh 'make clean install' // Removes js files, so do before compile
- sh 'make test_unit MOCHA_ARGS="--reporter=tap"'
- }
- }
-
- stage('Acceptance Tests') {
- steps {
- sh 'make test_acceptance MOCHA_ARGS="--reporter=tap"'
+ sh 'make ci'
}
}
@@ -155,6 +148,10 @@ pipeline {
}
post {
+ always {
+ sh 'make ci_clean'
+ }
+
failure {
mail(from: "${EMAIL_ALERT_FROM}",
to: "${EMAIL_ALERT_TO}",
diff --git a/services/web/Makefile b/services/web/Makefile
index 98695a8c1f..5db188e2b7 100644
--- a/services/web/Makefile
+++ b/services/web/Makefile
@@ -16,9 +16,10 @@ add_dev: docker-shared.yml
$(NPM) install --save-dev ${P}
install: docker-shared.yml
+ bin/generate_volumes_file
$(NPM) install
-clean:
+clean: ci_clean
rm -f app.js
rm -rf app/js
rm -rf test/unit/js
@@ -30,9 +31,8 @@ clean:
rm -rf $$dir/test/unit/js; \
rm -rf $$dir/test/acceptance/js; \
done
- # Regenerate docker-shared.yml - not stictly a 'clean',
- # but lets `make clean install` work nicely
- bin/generate_volumes_file
+
+ci_clean:
# Deletes node_modules volume
docker-compose down --volumes
@@ -40,11 +40,14 @@ clean:
docker-shared.yml:
bin/generate_volumes_file
-test: test_unit test_acceptance
+test: test_unit test_frontend test_acceptance
test_unit: docker-shared.yml
docker-compose ${DOCKER_COMPOSE_FLAGS} run --rm test_unit npm -q run test:unit -- ${MOCHA_ARGS}
+test_frontend: docker-shared.yml
+ docker-compose ${DOCKER_COMPOSE_FLAGS} run --rm test_unit npm -q run test:frontend -- ${MOCHA_ARGS}
+
test_acceptance: test_acceptance_app test_acceptance_modules
test_acceptance_app: test_acceptance_app_start_service test_acceptance_app_run test_acceptance_app_stop_service
@@ -71,7 +74,11 @@ test_acceptance_modules: docker-shared.yml
test_acceptance_module: docker-shared.yml
cd $(MODULE) && make test_acceptance
+ci:
+ MOCHA_ARGS="--reporter tap" \
+ $(MAKE) install test
+
.PHONY:
- all add install update test test_unit test_acceptance \
+ all add install update test test_unit test_frontend test_acceptance \
test_acceptance_start_service test_acceptance_stop_service \
- test_acceptance_run
+ test_acceptance_run ci ci_clean
diff --git a/services/web/app/coffee/Features/DocumentUpdater/DocumentUpdaterHandler.coffee b/services/web/app/coffee/Features/DocumentUpdater/DocumentUpdaterHandler.coffee
index bbd7ca4919..e18b8c8123 100644
--- a/services/web/app/coffee/Features/DocumentUpdater/DocumentUpdaterHandler.coffee
+++ b/services/web/app/coffee/Features/DocumentUpdater/DocumentUpdaterHandler.coffee
@@ -205,7 +205,7 @@ module.exports = DocumentUpdaterHandler =
callback new Error("doc updater returned a non-success status code: #{res.statusCode}")
updateProjectStructure : (project_id, userId, changes, callback = (error) ->)->
- return callback() if !settings.apis.project_history?.enabled
+ return callback() if !settings.apis.project_history?.sendProjectStructureOps
docUpdates = DocumentUpdaterHandler._getUpdates('doc', changes.oldDocs, changes.newDocs)
fileUpdates = DocumentUpdaterHandler._getUpdates('file', changes.oldFiles, changes.newFiles)
diff --git a/services/web/app/coffee/Features/History/HistoryController.coffee b/services/web/app/coffee/Features/History/HistoryController.coffee
index 70e0828160..0935118a7c 100644
--- a/services/web/app/coffee/Features/History/HistoryController.coffee
+++ b/services/web/app/coffee/Features/History/HistoryController.coffee
@@ -6,7 +6,7 @@ ProjectDetailsHandler = require "../Project/ProjectDetailsHandler"
module.exports = HistoryController =
initializeProject: (callback = (error, history_id) ->) ->
- return callback() if !settings.apis.project_history?.enabled
+ return callback() if !settings.apis.project_history?.initializeHistoryForNewProjects
request.post {
url: "#{settings.apis.project_history.url}/project"
}, (error, res, body)->
@@ -33,7 +33,8 @@ module.exports = HistoryController =
# find out which type of history service this project uses
ProjectDetailsHandler.getDetails project_id, (err, project) ->
return next(err) if err?
- if project?.overleaf?.history?.display
+ history = project.overleaf?.history
+ if history?.id? and history?.display
req.useProjectHistory = true
else
req.useProjectHistory = false
@@ -58,7 +59,7 @@ module.exports = HistoryController =
buildHistoryServiceUrl: (useProjectHistory) ->
# choose a history service, either document-level (trackchanges)
# or project-level (project_history)
- if settings.apis.project_history?.enabled && useProjectHistory
+ if useProjectHistory
return settings.apis.project_history.url
else
return settings.apis.trackchanges.url
diff --git a/services/web/app/coffee/Features/Project/ProjectController.coffee b/services/web/app/coffee/Features/Project/ProjectController.coffee
index da22373870..fb27611bca 100644
--- a/services/web/app/coffee/Features/Project/ProjectController.coffee
+++ b/services/web/app/coffee/Features/Project/ProjectController.coffee
@@ -216,7 +216,7 @@ module.exports = ProjectController =
project: (cb)->
ProjectGetter.getProject(
project_id,
- { name: 1, lastUpdated: 1, track_changes: 1, owner_ref: 1 },
+ { name: 1, lastUpdated: 1, track_changes: 1, owner_ref: 1, 'overleaf.history.display': 1 },
cb
)
user: (cb)->
@@ -351,6 +351,7 @@ module.exports = ProjectController =
themes: THEME_LIST
maxDocLength: Settings.max_doc_length
showLinkSharingOnboarding: !!results.couldShowLinkSharingOnboarding
+ useV2History: !!project.overleaf?.history?.display
timer.done()
_buildProjectList: (allProjects, v1Projects = [])->
diff --git a/services/web/app/coffee/router.coffee b/services/web/app/coffee/router.coffee
index c89d9cdf5e..2199bdd08e 100644
--- a/services/web/app/coffee/router.coffee
+++ b/services/web/app/coffee/router.coffee
@@ -197,6 +197,7 @@ module.exports = class Router
webRouter.get "/project/:Project_id/updates", AuthorizationMiddlewear.ensureUserCanReadProject, HistoryController.selectHistoryApi, HistoryController.proxyToHistoryApi
webRouter.get "/project/:Project_id/doc/:doc_id/diff", AuthorizationMiddlewear.ensureUserCanReadProject, HistoryController.selectHistoryApi, HistoryController.proxyToHistoryApi
+ webRouter.get "/project/:Project_id/diff", AuthorizationMiddlewear.ensureUserCanReadProject, HistoryController.selectHistoryApi, HistoryController.proxyToHistoryApi
webRouter.post "/project/:Project_id/doc/:doc_id/version/:version_id/restore", AuthorizationMiddlewear.ensureUserCanReadProject, HistoryController.selectHistoryApi, HistoryController.proxyToHistoryApi
webRouter.get '/Project/:Project_id/download/zip', AuthorizationMiddlewear.ensureUserCanReadProject, ProjectDownloadsController.downloadProject
diff --git a/services/web/app/views/project/editor.pug b/services/web/app/views/project/editor.pug
index d1d0d94cf3..4a35b930ff 100644
--- a/services/web/app/views/project/editor.pug
+++ b/services/web/app/views/project/editor.pug
@@ -105,7 +105,7 @@ block requirejs
//- We need to do .replace(/\//g, '\\/') do that '' -> '<\/script>'
//- and doesn't prematurely end the script tag.
script#data(type="application/json").
- !{JSON.stringify({userSettings: userSettings, user: user, trackChangesState: trackChangesState}).replace(/\//g, '\\/')}
+ !{JSON.stringify({userSettings: userSettings, user: user, trackChangesState: trackChangesState, useV2History: useV2History}).replace(/\//g, '\\/')}
script(type="text/javascript").
window.data = JSON.parse($("#data").text());
diff --git a/services/web/app/views/project/editor/history.pug b/services/web/app/views/project/editor/history.pug
index 4806b0a9b8..6621cdb2d2 100644
--- a/services/web/app/views/project/editor/history.pug
+++ b/services/web/app/views/project/editor/history.pug
@@ -134,8 +134,17 @@ div#history(ng-show="ui.view == 'history'")
div.description(ng-click="select()")
div.time {{ update.meta.end_ts | formatDate:'h:mm a' }}
- div.docs(ng-repeat="(doc_id, doc) in update.docs")
- span.doc {{ doc.entity.name }}
+ div.action.action-edited(ng-if="history.isV2 && update.pathnames.length > 0")
+ | Edited
+ div.docs(ng-repeat="pathname in update.pathnames")
+ .doc {{ pathname }}
+ div.docs(ng-repeat="project_op in update.project_ops")
+ div(ng-if="project_op.rename")
+ .action Renamed
+ .doc {{ project_op.rename.pathname }} → {{ project_op.rename.newPathname }}
+ div(ng-if="project_op.add")
+ .action Created
+ .doc {{ project_op.add.pathname }}
div.users
div.user(ng-repeat="update_user in update.meta.users")
.color-square(ng-if="update_user != null", ng-style="{'background-color': 'hsl({{ update_user.hue }}, 70%, 50%)'}")
@@ -165,8 +174,8 @@ div#history(ng-show="ui.view == 'history'")
'other': 'changes'\
}"
)
- | in {{history.diff.doc.name}}
- .toolbar-right
+ | in {{history.diff.pathname}}
+ .toolbar-right(ng-if="!history.isV2")
a.btn.btn-danger.btn-sm(
href,
ng-click="openRestoreDiffModal()"
diff --git a/services/web/bin/compile_app b/services/web/bin/compile_backend
similarity index 100%
rename from services/web/bin/compile_app
rename to services/web/bin/compile_backend
diff --git a/services/web/bin/compile_frontend b/services/web/bin/compile_frontend
new file mode 100755
index 0000000000..bb1dde7dbb
--- /dev/null
+++ b/services/web/bin/compile_frontend
@@ -0,0 +1,5 @@
+#!/bin/bash
+set -e;
+COFFEE=node_modules/.bin/coffee
+echo Compiling public/coffee;
+$COFFEE -o public/js -c public/coffee;
diff --git a/services/web/bin/compile_frontend_tests b/services/web/bin/compile_frontend_tests
new file mode 100755
index 0000000000..0351ad70cd
--- /dev/null
+++ b/services/web/bin/compile_frontend_tests
@@ -0,0 +1,5 @@
+#!/bin/bash
+set -e;
+COFFEE=node_modules/.bin/coffee
+echo Compiling test/unit_frontend/coffee;
+$COFFEE -o test/unit_frontend/js -c test/unit_frontend/coffee;
diff --git a/services/web/bin/frontend_test b/services/web/bin/frontend_test
new file mode 100755
index 0000000000..599055803a
--- /dev/null
+++ b/services/web/bin/frontend_test
@@ -0,0 +1,5 @@
+#!/bin/bash
+set -e;
+MOCHA="node_modules/.bin/mocha --recursive --reporter spec"
+$MOCHA "$@" test/unit_frontend/js
+
diff --git a/services/web/config/settings.defaults.coffee b/services/web/config/settings.defaults.coffee
index 65a8bf1e91..844bffae3c 100644
--- a/services/web/config/settings.defaults.coffee
+++ b/services/web/config/settings.defaults.coffee
@@ -111,7 +111,8 @@ module.exports = settings =
trackchanges:
url : "http://localhost:3015"
project_history:
- enabled: process.env.PROJECT_HISTORY_ENABLED == 'true' or false
+ sendProjectStructureOps: process.env.PROJECT_HISTORY_ENABLED == 'true' or false
+ initializeHistoryForNewProjects: process.env.PROJECT_HISTORY_ENABLED == 'true' or false
url : "http://localhost:3054"
docstore:
url : "http://#{process.env['DOCSTORE_HOST'] or 'localhost'}:3016"
diff --git a/services/web/docker-shared.template.yml b/services/web/docker-shared.template.yml
index d697e59e96..8acc63ef81 100644
--- a/services/web/docker-shared.template.yml
+++ b/services/web/docker-shared.template.yml
@@ -13,17 +13,15 @@ services:
- ./npm-shrinkwrap.json:/app/npm-shrinkwrap.json
- node_modules:/app/node_modules
- ./bin:/app/bin
- # Copying the whole public dir is fine for now, and needed for
- # some unit tests to pass, but we will want to isolate the coffee
- # and vendor js files, so that the compiled js files are not written
- # back to the local filesystem.
- - ./public:/app/public
+ - ./public/coffee:/app/public/coffee:ro
+ - ./public/js/ace-1.2.5:/app/public/js/ace-1.2.5
- ./app.coffee:/app/app.coffee:ro
- ./app/coffee:/app/app/coffee:ro
- ./app/templates:/app/app/templates:ro
- ./app/views:/app/app/views:ro
- ./config:/app/config
- ./test/unit/coffee:/app/test/unit/coffee:ro
+ - ./test/unit_frontend/coffee:/app/test/unit_frontend/coffee:ro
- ./test/acceptance/coffee:/app/test/acceptance/coffee:ro
- ./test/acceptance/files:/app/test/acceptance/files:ro
- ./test/smoke/coffee:/app/test/smoke/coffee:ro
diff --git a/services/web/package.json b/services/web/package.json
index a1cf0fd6e0..398ce3dc78 100644
--- a/services/web/package.json
+++ b/services/web/package.json
@@ -14,11 +14,15 @@
"test:acceptance:run": "bin/acceptance_test $@",
"test:acceptance:dir": "npm -q run compile:acceptance_tests && npm -q run test:acceptance:wait_for_app && npm -q run test:acceptance:run -- $@",
"test:acceptance": "npm -q run test:acceptance:dir -- $@ test/acceptance/js",
- "test:unit": "npm -q run compile:app && npm -q run compile:unit_tests && bin/unit_test $@",
+ "test:unit": "npm -q run compile:backend && npm -q run compile:unit_tests && bin/unit_test $@",
+ "test:frontend": "npm -q run compile:frontend && npm -q run compile:frontend_tests && bin/frontend_test $@",
"compile:unit_tests": "bin/compile_unit_tests",
+ "compile:frontend_tests": "bin/compile_frontend_tests",
"compile:acceptance_tests": "bin/compile_acceptance_tests",
- "compile:app": "bin/compile_app",
- "start": "npm -q run compile:app && node app.js"
+ "compile:frontend": "bin/compile_frontend",
+ "compile:backend": "bin/compile_backend",
+ "compile": "npm -q run compile:backend && npm -q run compile:frontend",
+ "start": "npm -q run compile && node app.js"
},
"dependencies": {
"archiver": "0.9.0",
diff --git a/services/web/public/coffee/ide.coffee b/services/web/public/coffee/ide.coffee
index dbf6a2724e..3d514d093c 100644
--- a/services/web/public/coffee/ide.coffee
+++ b/services/web/public/coffee/ide.coffee
@@ -5,6 +5,7 @@ define [
"ide/editor/EditorManager"
"ide/online-users/OnlineUsersManager"
"ide/history/HistoryManager"
+ "ide/history/HistoryV2Manager"
"ide/permissions/PermissionsManager"
"ide/pdf/PdfManager"
"ide/binary-files/BinaryFilesManager"
@@ -44,6 +45,7 @@ define [
EditorManager
OnlineUsersManager
HistoryManager
+ HistoryV2Manager
PermissionsManager
PdfManager
BinaryFilesManager
@@ -137,7 +139,10 @@ define [
ide.fileTreeManager = new FileTreeManager(ide, $scope)
ide.editorManager = new EditorManager(ide, $scope)
ide.onlineUsersManager = new OnlineUsersManager(ide, $scope)
- ide.historyManager = new HistoryManager(ide, $scope)
+ if window.data.useV2History
+ ide.historyManager = new HistoryV2Manager(ide, $scope)
+ else
+ ide.historyManager = new HistoryManager(ide, $scope)
ide.pdfManager = new PdfManager(ide, $scope)
ide.permissionsManager = new PermissionsManager(ide, $scope)
ide.binaryFilesManager = new BinaryFilesManager(ide, $scope)
diff --git a/services/web/public/coffee/ide/history/HistoryManager.coffee b/services/web/public/coffee/ide/history/HistoryManager.coffee
index 6b42714e79..f896fbb8b4 100644
--- a/services/web/public/coffee/ide/history/HistoryManager.coffee
+++ b/services/web/public/coffee/ide/history/HistoryManager.coffee
@@ -100,6 +100,7 @@ define [
end_ts: end_ts
doc: doc
error: false
+ pathname: doc.name
}
if !doc.deleted
@@ -190,8 +191,10 @@ define [
previousUpdate = @$scope.history.updates[@$scope.history.updates.length - 1]
for update in updates
+ update.pathnames = [] # Used for display
for doc_id, doc of update.docs or {}
doc.entity = @ide.fileTreeManager.findEntityById(doc_id, includeDeleted: true)
+ update.pathnames.push doc.entity.name
for user in update.meta.users or []
if user?
diff --git a/services/web/public/coffee/ide/history/HistoryV2Manager.coffee b/services/web/public/coffee/ide/history/HistoryV2Manager.coffee
new file mode 100644
index 0000000000..8198738e16
--- /dev/null
+++ b/services/web/public/coffee/ide/history/HistoryV2Manager.coffee
@@ -0,0 +1,280 @@
+define [
+ "moment"
+ "ide/colors/ColorManager"
+ "ide/history/controllers/HistoryListController"
+ "ide/history/controllers/HistoryDiffController"
+ "ide/history/directives/infiniteScroll"
+], (moment, ColorManager) ->
+ class HistoryManager
+ constructor: (@ide, @$scope) ->
+ @reset()
+
+ @$scope.toggleHistory = () =>
+ if @$scope.ui.view == "history"
+ @hide()
+ else
+ @show()
+
+ @$scope.$watch "history.selection.updates", (updates) =>
+ if updates? and updates.length > 0
+ @_selectDocFromUpdates()
+ @reloadDiff()
+
+ @$scope.$on "entity:selected", (event, entity) =>
+ if (@$scope.ui.view == "history") and (entity.type == "doc")
+ @$scope.history.selection.pathname = _ide.fileTreeManager.getEntityPath(entity)
+ @reloadDiff()
+
+ show: () ->
+ @$scope.ui.view = "history"
+ @reset()
+
+ hide: () ->
+ @$scope.ui.view = "editor"
+ # Make sure we run the 'open' logic for whatever is currently selected
+ @$scope.$emit "entity:selected", @ide.fileTreeManager.findSelectedEntity()
+
+ reset: () ->
+ @$scope.history = {
+ isV2: true
+ updates: []
+ nextBeforeTimestamp: null
+ atEnd: false
+ selection: {
+ updates: []
+ pathname: null
+ range: {
+ fromV: null
+ toV: null
+ }
+ }
+ diff: null
+ }
+
+ MAX_RECENT_UPDATES_TO_SELECT: 2
+ autoSelectRecentUpdates: () ->
+ return if @$scope.history.updates.length == 0
+
+ @$scope.history.updates[0].selectedTo = true
+
+ indexOfLastUpdateNotByMe = 0
+ for update, i in @$scope.history.updates
+ if @_updateContainsUserId(update, @$scope.user.id) or i > @MAX_RECENT_UPDATES_TO_SELECT
+ break
+ indexOfLastUpdateNotByMe = i
+
+ @$scope.history.updates[indexOfLastUpdateNotByMe].selectedFrom = true
+
+ BATCH_SIZE: 10
+ fetchNextBatchOfUpdates: () ->
+ url = "/project/#{@ide.project_id}/updates?min_count=#{@BATCH_SIZE}"
+ if @$scope.history.nextBeforeTimestamp?
+ url += "&before=#{@$scope.history.nextBeforeTimestamp}"
+ @$scope.history.loading = true
+ @ide.$http
+ .get(url)
+ .then (response) =>
+ { data } = response
+ @_loadUpdates(data.updates)
+ @$scope.history.nextBeforeTimestamp = data.nextBeforeTimestamp
+ if !data.nextBeforeTimestamp?
+ @$scope.history.atEnd = true
+ @$scope.history.loading = false
+
+ reloadDiff: () ->
+ diff = @$scope.history.diff
+ {updates} = @$scope.history.selection
+ {fromV, toV, pathname} = @_calculateDiffDataFromSelection()
+
+ if !pathname?
+ @$scope.history.diff = null
+ return
+
+ return if diff? and
+ diff.pathname == pathname and
+ diff.fromV == fromV and
+ diff.toV == toV
+
+ @$scope.history.diff = diff = {
+ fromV: fromV
+ toV: toV
+ pathname: pathname
+ error: false
+ }
+
+ diff.loading = true
+ url = "/project/#{@$scope.project_id}/diff"
+ query = ["pathname=#{encodeURIComponent(pathname)}"]
+ if diff.fromV? and diff.toV?
+ query.push "from=#{diff.fromV}", "to=#{diff.toV}"
+ url += "?" + query.join("&")
+
+ @ide.$http
+ .get(url)
+ .then (response) =>
+ { data } = response
+ diff.loading = false
+ {text, highlights} = @_parseDiff(data)
+ diff.text = text
+ diff.highlights = highlights
+ .catch () ->
+ diff.loading = false
+ diff.error = true
+
+ _parseDiff: (diff) ->
+ row = 0
+ column = 0
+ highlights = []
+ text = ""
+ for entry, i in diff.diff or []
+ content = entry.u or entry.i or entry.d
+ content ||= ""
+ text += content
+ lines = content.split("\n")
+ startRow = row
+ startColumn = column
+ if lines.length > 1
+ endRow = startRow + lines.length - 1
+ endColumn = lines[lines.length - 1].length
+ else
+ endRow = startRow
+ endColumn = startColumn + lines[0].length
+ row = endRow
+ column = endColumn
+
+ range = {
+ start:
+ row: startRow
+ column: startColumn
+ end:
+ row: endRow
+ column: endColumn
+ }
+
+ if entry.i? or entry.d?
+ if entry.meta.user?
+ name = "#{entry.meta.user.first_name} #{entry.meta.user.last_name}"
+ else
+ name = "Anonymous"
+ if entry.meta.user?.id == @$scope.user.id
+ name = "you"
+ date = moment(entry.meta.end_ts).format("Do MMM YYYY, h:mm a")
+ if entry.i?
+ highlights.push {
+ label: "Added by #{name} on #{date}"
+ highlight: range
+ hue: ColorManager.getHueForUserId(entry.meta.user?.id)
+ }
+ else if entry.d?
+ highlights.push {
+ label: "Deleted by #{name} on #{date}"
+ strikeThrough: range
+ hue: ColorManager.getHueForUserId(entry.meta.user?.id)
+ }
+
+ return {text, highlights}
+
+ _loadUpdates: (updates = []) ->
+ previousUpdate = @$scope.history.updates[@$scope.history.updates.length - 1]
+
+ for update in updates or []
+ for user in update.meta.users or []
+ if user?
+ user.hue = ColorManager.getHueForUserId(user.id)
+
+ if !previousUpdate? or !moment(previousUpdate.meta.end_ts).isSame(update.meta.end_ts, "day")
+ update.meta.first_in_day = true
+
+ update.selectedFrom = false
+ update.selectedTo = false
+ update.inSelection = false
+
+ previousUpdate = update
+
+ firstLoad = @$scope.history.updates.length == 0
+
+ @$scope.history.updates =
+ @$scope.history.updates.concat(updates)
+
+ @autoSelectRecentUpdates() if firstLoad
+
+ _perDocSummaryOfUpdates: (updates) ->
+ # Track current_pathname -> original_pathname
+ original_pathnames = {}
+
+ # Map of original pathname -> doc summary
+ docs_summary = {}
+
+ updatePathnameWithUpdateVersions = (pathname, update) ->
+ # docs_summary is indexed by the original pathname the doc
+ # had at the start, so we have to look this up from the current
+ # pathname via original_pathname first
+ if !original_pathnames[pathname]?
+ original_pathnames[pathname] = pathname
+ original_pathname = original_pathnames[pathname]
+ doc_summary = docs_summary[original_pathname] ?= {
+ fromV: update.fromV, toV: update.toV,
+ }
+ doc_summary.fromV = Math.min(
+ doc_summary.fromV,
+ update.fromV
+ )
+ doc_summary.toV = Math.max(
+ doc_summary.toV,
+ update.toV
+ )
+
+ # Put updates in ascending chronological order
+ updates = updates.slice().reverse()
+ for update in updates
+ for pathname in update.pathnames or []
+ updatePathnameWithUpdateVersions(pathname, update)
+ for project_op in update.project_ops or []
+ if project_op.rename?
+ rename = project_op.rename
+ updatePathnameWithUpdateVersions(rename.pathname, update)
+ original_pathnames[rename.newPathname] = original_pathnames[rename.pathname]
+ delete original_pathnames[rename.pathname]
+ if project_op.add?
+ add = project_op.add
+ updatePathnameWithUpdateVersions(add.pathname, update)
+
+ return docs_summary
+
+ _calculateDiffDataFromSelection: () ->
+ fromV = toV = pathname = null
+
+ selected_pathname = @$scope.history.selection.pathname
+
+ for pathname, doc of @_perDocSummaryOfUpdates(@$scope.history.selection.updates)
+ if pathname == selected_pathname
+ {fromV, toV} = doc
+ return {fromV, toV, pathname}
+
+ return {}
+
+ # Set the track changes selected doc to one of the docs in the range
+ # of currently selected updates. If we already have a selected doc
+ # then prefer this one if present.
+ _selectDocFromUpdates: () ->
+ affected_docs = @_perDocSummaryOfUpdates(@$scope.history.selection.updates)
+
+ selected_pathname = @$scope.history.selection.pathname
+ if selected_pathname? and affected_docs[selected_pathname]
+ # Selected doc is already open
+ else
+ # Set to first possible candidate
+ for pathname, doc of affected_docs
+ selected_pathname = pathname
+ break
+
+ @$scope.history.selection.pathname = selected_pathname
+ if selected_pathname?
+ entity = @ide.fileTreeManager.findEntityByPath(selected_pathname)
+ if entity?
+ @ide.fileTreeManager.selectEntity(entity)
+
+ _updateContainsUserId: (update, user_id) ->
+ for user in update.meta.users
+ return true if user?.id == user_id
+ return false
diff --git a/services/web/public/stylesheets/app/editor/history.less b/services/web/public/stylesheets/app/editor/history.less
index b6dca4b7cc..2824fd2e32 100644
--- a/services/web/public/stylesheets/app/editor/history.less
+++ b/services/web/public/stylesheets/app/editor/history.less
@@ -169,9 +169,19 @@
font-size: 0.8rem;
line-height: @line-height-computed;
}
- .docs {
- font-weight: bold;
+ .doc {
font-size: 0.9rem;
+ font-weight: bold;
+ }
+ .action {
+ color: @gray;
+ text-transform: uppercase;
+ font-size: 0.7em;
+ margin-bottom: -2px;
+ margin-top: 2px;
+ &-edited {
+ margin-top: 0;
+ }
}
}
li.loading-changes, li.empty-message {
diff --git a/services/web/test/unit/coffee/DocumentUpdater/DocumentUpdaterHandlerTests.coffee b/services/web/test/unit/coffee/DocumentUpdater/DocumentUpdaterHandlerTests.coffee
index 111d2ace68..a4e0a4dc53 100644
--- a/services/web/test/unit/coffee/DocumentUpdater/DocumentUpdaterHandlerTests.coffee
+++ b/services/web/test/unit/coffee/DocumentUpdater/DocumentUpdaterHandlerTests.coffee
@@ -393,7 +393,7 @@ describe 'DocumentUpdaterHandler', ->
describe "with project history disabled", ->
beforeEach ->
- @settings.apis.project_history.enabled = false
+ @settings.apis.project_history.sendProjectStructureOps = false
@request.post = sinon.stub()
@handler.updateProjectStructure @project_id, @user_id, {}, @callback
@@ -406,7 +406,7 @@ describe 'DocumentUpdaterHandler', ->
describe "with project history enabled", ->
beforeEach ->
- @settings.apis.project_history.enabled = true
+ @settings.apis.project_history.sendProjectStructureOps = true
@url = "#{@settings.apis.documentupdater.url}/project/#{@project_id}"
@request.post = sinon.stub().callsArgWith(1, null, {statusCode: 204}, "")
diff --git a/services/web/test/unit/coffee/History/HistoryControllerTests.coffee b/services/web/test/unit/coffee/History/HistoryControllerTests.coffee
index e03189526a..3e0464254e 100644
--- a/services/web/test/unit/coffee/History/HistoryControllerTests.coffee
+++ b/services/web/test/unit/coffee/History/HistoryControllerTests.coffee
@@ -31,7 +31,7 @@ describe "HistoryController", ->
describe "for a project with project history", ->
beforeEach ->
- @ProjectDetailsHandler.getDetails = sinon.stub().callsArgWith(1, null, {overleaf:{history:{display:true}}})
+ @ProjectDetailsHandler.getDetails = sinon.stub().callsArgWith(1, null, {overleaf:{history:{id: 42, display:true}}})
@HistoryController.selectHistoryApi @req, @res, @next
it "should set the flag for project history to true", ->
@@ -57,93 +57,55 @@ describe "HistoryController", ->
on: (event, handler) -> @events[event] = handler
@request.returns @proxy
- describe "with project history enabled", ->
+ describe "for a project with the project history flag", ->
beforeEach ->
- @settings.apis.project_history.enabled = true
+ @req.useProjectHistory = true
+ @HistoryController.proxyToHistoryApi @req, @res, @next
- describe "for a project with the project history flag", ->
- beforeEach ->
- @req.useProjectHistory = true
- @HistoryController.proxyToHistoryApi @req, @res, @next
+ it "should get the user id", ->
+ @AuthenticationController.getLoggedInUserId
+ .calledWith(@req)
+ .should.equal true
- it "should get the user id", ->
- @AuthenticationController.getLoggedInUserId
- .calledWith(@req)
- .should.equal true
+ it "should call the project history api", ->
+ @request
+ .calledWith({
+ url: "#{@settings.apis.project_history.url}#{@req.url}"
+ method: @req.method
+ headers:
+ "X-User-Id": @user_id
+ })
+ .should.equal true
- it "should call the project history api", ->
- @request
- .calledWith({
- url: "#{@settings.apis.project_history.url}#{@req.url}"
- method: @req.method
- headers:
- "X-User-Id": @user_id
- })
- .should.equal true
+ it "should pipe the response to the client", ->
+ @proxy.pipe
+ .calledWith(@res)
+ .should.equal true
- it "should pipe the response to the client", ->
- @proxy.pipe
- .calledWith(@res)
- .should.equal true
-
- describe "for a project without the project history flag", ->
- beforeEach ->
- @req.useProjectHistory = false
- @HistoryController.proxyToHistoryApi @req, @res, @next
-
- it "should get the user id", ->
- @AuthenticationController.getLoggedInUserId
- .calledWith(@req)
- .should.equal true
-
- it "should call the track changes api", ->
- @request
- .calledWith({
- url: "#{@settings.apis.trackchanges.url}#{@req.url}"
- method: @req.method
- headers:
- "X-User-Id": @user_id
- })
- .should.equal true
-
- it "should pipe the response to the client", ->
- @proxy.pipe
- .calledWith(@res)
- .should.equal true
-
- describe "with project history disabled", ->
+ describe "for a project without the project history flag", ->
beforeEach ->
- @settings.apis.project_history.enabled = false
+ @req.useProjectHistory = false
+ @HistoryController.proxyToHistoryApi @req, @res, @next
- describe "for a project with the project history flag", ->
- beforeEach ->
- @req.useProjectHistory = true
- @HistoryController.proxyToHistoryApi @req, @res, @next
+ it "should get the user id", ->
+ @AuthenticationController.getLoggedInUserId
+ .calledWith(@req)
+ .should.equal true
- it "should call the track changes api", ->
- @request
- .calledWith({
- url: "#{@settings.apis.trackchanges.url}#{@req.url}"
- method: @req.method
- headers:
- "X-User-Id": @user_id
- })
- .should.equal true
+ it "should call the track changes api", ->
+ @request
+ .calledWith({
+ url: "#{@settings.apis.trackchanges.url}#{@req.url}"
+ method: @req.method
+ headers:
+ "X-User-Id": @user_id
+ })
+ .should.equal true
- describe "for a project without the project history flag", ->
- beforeEach ->
- @req.useProjectHistory = false
- @HistoryController.proxyToHistoryApi @req, @res, @next
-
- it "should call the track changes api", ->
- @request
- .calledWith({
- url: "#{@settings.apis.trackchanges.url}#{@req.url}"
- method: @req.method
- headers:
- "X-User-Id": @user_id
- })
- .should.equal true
+ it "should pipe the response to the client", ->
+ @proxy.pipe
+ .calledWith(@res)
+ .should.equal true
describe "with an error", ->
beforeEach ->
@@ -156,7 +118,7 @@ describe "HistoryController", ->
describe "initializeProject", ->
describe "with project history enabled", ->
beforeEach ->
- @settings.apis.project_history.enabled = true
+ @settings.apis.project_history.initializeHistoryForNewProjects = true
describe "project history returns a successful response", ->
beforeEach ->
@@ -212,7 +174,7 @@ describe "HistoryController", ->
describe "with project history disabled", ->
beforeEach ->
- @settings.apis.project_history.enabled = false
+ @settings.apis.project_history.initializeHistoryForNewProjects = false
@HistoryController.initializeProject @callback
it "should return the callback", ->
diff --git a/services/web/test/unit_frontend/coffee/HistoryManagerV2Tests.coffee b/services/web/test/unit_frontend/coffee/HistoryManagerV2Tests.coffee
new file mode 100644
index 0000000000..9ad7ba9a2e
--- /dev/null
+++ b/services/web/test/unit_frontend/coffee/HistoryManagerV2Tests.coffee
@@ -0,0 +1,134 @@
+Path = require 'path'
+SandboxedModule = require "sandboxed-module"
+modulePath = Path.join __dirname, '../../../public/js/ide/history/HistoryV2Manager'
+sinon = require("sinon")
+expect = require("chai").expect
+
+describe "HistoryV2Manager", ->
+ beforeEach ->
+ @moment = {}
+ @ColorManager = {}
+ SandboxedModule.require modulePath, globals:
+ "define": (dependencies, builder) =>
+ @HistoryV2Manager = builder(@moment, @ColorManager)
+
+ @scope =
+ $watch: sinon.stub()
+ $on: sinon.stub()
+ @ide = {}
+
+ @historyManager = new @HistoryV2Manager(@ide, @scope)
+
+ it "should setup the history scope on intialization", ->
+ expect(@scope.history).to.deep.equal({
+ isV2: true
+ updates: []
+ nextBeforeTimestamp: null
+ atEnd: false
+ selection: {
+ updates: []
+ pathname: null
+ range: {
+ fromV: null
+ toV: null
+ }
+ }
+ diff: null
+ })
+
+ describe "_perDocSummaryOfUpdates", ->
+ it "should return the range of updates for the docs", ->
+ result = @historyManager._perDocSummaryOfUpdates([{
+ pathnames: ["main.tex"]
+ fromV: 7, toV: 9
+ },{
+ pathnames: ["main.tex", "foo.tex"]
+ fromV: 4, toV: 6
+ },{
+ pathnames: ["main.tex"]
+ fromV: 3, toV: 3
+ },{
+ pathnames: ["foo.tex"]
+ fromV: 0, toV: 2
+ }])
+
+ expect(result).to.deep.equal({
+ "main.tex": { fromV: 3, toV: 9 },
+ "foo.tex": { fromV: 0, toV: 6 }
+ })
+
+ it "should track renames", ->
+ result = @historyManager._perDocSummaryOfUpdates([{
+ pathnames: ["main2.tex"]
+ fromV: 5, toV: 9
+ },{
+ project_ops: [{
+ rename: {
+ pathname: "main1.tex",
+ newPathname: "main2.tex"
+ }
+ }],
+ fromV: 4, toV: 4
+ },{
+ pathnames: ["main1.tex"]
+ fromV: 3, toV: 3
+ },{
+ project_ops: [{
+ rename: {
+ pathname: "main0.tex",
+ newPathname: "main1.tex"
+ }
+ }],
+ fromV: 2, toV: 2
+ },{
+ pathnames: ["main0.tex"]
+ fromV: 0, toV: 1
+ }])
+
+ expect(result).to.deep.equal({
+ "main0.tex": { fromV: 0, toV: 9 }
+ })
+
+ it "should track single renames", ->
+ result = @historyManager._perDocSummaryOfUpdates([{
+ project_ops: [{
+ rename: {
+ pathname: "main1.tex",
+ newPathname: "main2.tex"
+ }
+ }],
+ fromV: 4, toV: 5
+ }])
+
+ expect(result).to.deep.equal({
+ "main1.tex": { fromV: 4, toV: 5 }
+ })
+
+ it "should track additions", ->
+ result = @historyManager._perDocSummaryOfUpdates([{
+ project_ops: [{
+ add:
+ pathname: "main.tex"
+ }]
+ fromV: 0, toV: 1
+ }, {
+ pathnames: ["main.tex"]
+ fromV: 1, toV: 4
+ }])
+
+ expect(result).to.deep.equal({
+ "main.tex": { fromV: 0, toV: 4 }
+ })
+
+ it "should track single additions", ->
+ result = @historyManager._perDocSummaryOfUpdates([{
+ project_ops: [{
+ add:
+ pathname: "main.tex"
+ }]
+ fromV: 0, toV: 1
+ }])
+
+ expect(result).to.deep.equal({
+ "main.tex": { fromV: 0, toV: 1 }
+ })