mirror of
https://github.com/overleaf/overleaf.git
synced 2024-11-21 20:47:08 -05:00
moved load project from old proj controller to new one
This commit is contained in:
parent
9398625ac5
commit
88ba45b9dc
6 changed files with 173 additions and 144 deletions
|
@ -6,9 +6,12 @@ projectCreationHandler = require("./ProjectCreationHandler")
|
||||||
metrics = require('../../infrastructure/Metrics')
|
metrics = require('../../infrastructure/Metrics')
|
||||||
sanitize = require('sanitizer')
|
sanitize = require('sanitizer')
|
||||||
Project = require('../../models/Project').Project
|
Project = require('../../models/Project').Project
|
||||||
|
User = require('../../models/User').User
|
||||||
TagsHandler = require("../Tags/TagsHandler")
|
TagsHandler = require("../Tags/TagsHandler")
|
||||||
SubscriptionLocator = require("../Subscription/SubscriptionLocator")
|
SubscriptionLocator = require("../Subscription/SubscriptionLocator")
|
||||||
_ = require("underscore")
|
_ = require("underscore")
|
||||||
|
Settings = require("settings-sharelatex")
|
||||||
|
SecurityManager = require("../../managers/SecurityManager")
|
||||||
|
|
||||||
module.exports =
|
module.exports =
|
||||||
|
|
||||||
|
@ -72,6 +75,88 @@ module.exports =
|
||||||
timer.done()
|
timer.done()
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
loadEditor: (req, res)->
|
||||||
|
timer = new metrics.Timer("load-editor")
|
||||||
|
if !Settings.editorIsOpen
|
||||||
|
res.render("general/closed", {title:"updating site"})
|
||||||
|
else
|
||||||
|
if req.session.user?
|
||||||
|
user_id = req.session.user._id
|
||||||
|
else
|
||||||
|
user_id = 'openUser'
|
||||||
|
project_id = req.params.Project_id
|
||||||
|
Project.findPopulatedById project_id, (err, project)->
|
||||||
|
User.findById user_id, (err, user)->
|
||||||
|
if user_id == 'openUser'
|
||||||
|
anonymous = true
|
||||||
|
user =
|
||||||
|
id : user_id
|
||||||
|
ace:
|
||||||
|
mode:'none'
|
||||||
|
theme:'textmate'
|
||||||
|
fontSize: '12'
|
||||||
|
autoComplete: true
|
||||||
|
spellCheckLanguage: ""
|
||||||
|
pdfViewer: ""
|
||||||
|
subscription:
|
||||||
|
freeTrial:
|
||||||
|
allowed: true
|
||||||
|
featureSwitches:
|
||||||
|
dropbox: false
|
||||||
|
trackChanges: false
|
||||||
|
else
|
||||||
|
anonymous = false
|
||||||
|
SubscriptionLocator.getUsersSubscription user?._id, (err, subscription)->
|
||||||
|
SecurityManager.userCanAccessProject user, project, (canAccess, privilegeLevel)->
|
||||||
|
allowedFreeTrial = true
|
||||||
|
if subscription? and subscription.freeTrial? and subscription.freeTrial.expiresAt?
|
||||||
|
allowedFreeTrial = !!subscription.freeTrial.allowed
|
||||||
|
if canAccess
|
||||||
|
timer.done()
|
||||||
|
res.render 'project/editor',
|
||||||
|
title: project.name
|
||||||
|
priority_title: true
|
||||||
|
bodyClasses: ["editor"]
|
||||||
|
project : project
|
||||||
|
#owner : project.owner_ref
|
||||||
|
userObject : JSON.stringify({
|
||||||
|
id : user.id
|
||||||
|
email : user.email
|
||||||
|
first_name : user.first_name
|
||||||
|
last_name : user.last_name
|
||||||
|
referal_id : user.referal_id
|
||||||
|
subscription :
|
||||||
|
freeTrial: {allowed: allowedFreeTrial}
|
||||||
|
})
|
||||||
|
userSettingsObject: JSON.stringify({
|
||||||
|
mode : user.ace.mode
|
||||||
|
theme : user.ace.theme
|
||||||
|
project_id : project._id
|
||||||
|
fontSize : user.ace.fontSize
|
||||||
|
autoComplete: user.ace.autoComplete
|
||||||
|
spellCheckLanguage: user.ace.spellCheckLanguage
|
||||||
|
pdfViewer : user.ace.pdfViewer
|
||||||
|
docPositions: {}
|
||||||
|
oldHistory: !!user.featureSwitches?.oldHistory
|
||||||
|
})
|
||||||
|
sharelatexObject : JSON.stringify({
|
||||||
|
siteUrl: Settings.siteUrl,
|
||||||
|
jsPath: res.locals.jsPath
|
||||||
|
})
|
||||||
|
privilegeLevel: privilegeLevel
|
||||||
|
loadPdfjs: (user.ace.pdfViewer == "pdfjs")
|
||||||
|
chatUrl: Settings.apis.chat.url
|
||||||
|
anonymous: anonymous
|
||||||
|
languages: Settings.languages
|
||||||
|
else
|
||||||
|
res.send 401
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
_buildListViewModel = (projects, collabertions, readOnlyProjects, tags, tagsGroupedByProject, subscription)->
|
_buildListViewModel = (projects, collabertions, readOnlyProjects, tags, tagsGroupedByProject, subscription)->
|
||||||
# TODO: Remove this one month after the ability to start free trials was removed
|
# TODO: Remove this one month after the ability to start free trials was removed
|
||||||
if subscription? and subscription.freeTrial? and subscription.freeTrial.expiresAt?
|
if subscription? and subscription.freeTrial? and subscription.freeTrial.expiresAt?
|
||||||
|
|
|
@ -20,130 +20,6 @@ FileStoreHandler = require("../Features/FileStore/FileStoreHandler")
|
||||||
module.exports = class ProjectController
|
module.exports = class ProjectController
|
||||||
constructor: ()->
|
constructor: ()->
|
||||||
|
|
||||||
list: (req, res, next)->
|
|
||||||
timer = new metrics.Timer("project-list")
|
|
||||||
user_id = req.session.user._id
|
|
||||||
startTime = new Date()
|
|
||||||
User.findById user_id, (error, user) ->
|
|
||||||
logger.log user_id: user_id, duration: (new Date() - startTime), "project list timer - User.findById"
|
|
||||||
startTime = new Date()
|
|
||||||
# TODO: Remove this one month after the ability to start free trials was removed
|
|
||||||
SubscriptionLocator.getUsersSubscription user._id, (err, subscription)->
|
|
||||||
logger.log user_id: user_id, duration: (new Date() - startTime), "project list timer - Subscription.getUsersSubscription"
|
|
||||||
startTime = new Date()
|
|
||||||
return next(error) if error?
|
|
||||||
# TODO: Remove this one month after the ability to start free trials was removed
|
|
||||||
if subscription? and subscription.freeTrial? and subscription.freeTrial.expiresAt?
|
|
||||||
freeTrial =
|
|
||||||
expired: !!subscription.freeTrial.downgraded
|
|
||||||
expiresAt: SubscriptionFormatters.formatDate(subscription.freeTrial.expiresAt)
|
|
||||||
TagsHandler.getAllTags user_id, (err, tags, tagsGroupedByProject)->
|
|
||||||
logger.log user_id: user_id, duration: (new Date() - startTime), "project list timer - TagsHandler.getAllTags"
|
|
||||||
startTime = new Date()
|
|
||||||
Project.findAllUsersProjects user_id, 'name lastUpdated publicAccesLevel', (projects, collabertions, readOnlyProjects)->
|
|
||||||
logger.log user_id: user_id, duration: (new Date() - startTime), "project list timer - Project.findAllUsersProjects"
|
|
||||||
startTime = new Date()
|
|
||||||
for project in projects
|
|
||||||
project.accessLevel = "owner"
|
|
||||||
for project in collabertions
|
|
||||||
project.accessLevel = "readWrite"
|
|
||||||
for project in readOnlyProjects
|
|
||||||
project.accessLevel = "readOnly"
|
|
||||||
projects = projects.concat(collabertions).concat(readOnlyProjects)
|
|
||||||
projects = projects.map (project)->
|
|
||||||
project.tags = tagsGroupedByProject[project._id] || []
|
|
||||||
return project
|
|
||||||
tags = _.sortBy tags, (tag)->
|
|
||||||
-tag.project_ids.length
|
|
||||||
logger.log projects:projects, collabertions:collabertions, readOnlyProjects:readOnlyProjects, user_id:user_id, "rendering project list"
|
|
||||||
sortedProjects = _.sortBy projects, (project)->
|
|
||||||
return - project.lastUpdated
|
|
||||||
res.render 'project/list',
|
|
||||||
title:'Your Projects'
|
|
||||||
priority_title: true
|
|
||||||
projects: sortedProjects
|
|
||||||
freeTrial: freeTrial
|
|
||||||
tags:tags
|
|
||||||
projectTabActive: true
|
|
||||||
logger.log user_id: user_id, duration: (new Date() - startTime), "project list timer - Finished"
|
|
||||||
timer.done()
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
loadEditor: (req, res)->
|
|
||||||
timer = new metrics.Timer("load-editor")
|
|
||||||
if !Settings.editorIsOpen
|
|
||||||
res.render("general/closed", {title:"updating site"})
|
|
||||||
else
|
|
||||||
if req.session.user?
|
|
||||||
user_id = req.session.user._id
|
|
||||||
else
|
|
||||||
user_id = 'openUser'
|
|
||||||
project_id = req.params.Project_id
|
|
||||||
Project.findPopulatedById project_id, (err, project)->
|
|
||||||
User.findById user_id, (err, user)->
|
|
||||||
if user_id == 'openUser'
|
|
||||||
anonymous = true
|
|
||||||
user =
|
|
||||||
id : user_id
|
|
||||||
ace:
|
|
||||||
mode:'none'
|
|
||||||
theme:'textmate'
|
|
||||||
fontSize: '12'
|
|
||||||
autoComplete: true
|
|
||||||
spellCheckLanguage: ""
|
|
||||||
pdfViewer: ""
|
|
||||||
subscription:
|
|
||||||
freeTrial:
|
|
||||||
allowed: true
|
|
||||||
featureSwitches:
|
|
||||||
dropbox: false
|
|
||||||
trackChanges: false
|
|
||||||
else
|
|
||||||
anonymous = false
|
|
||||||
SubscriptionLocator.getUsersSubscription user?._id, (err, subscription)->
|
|
||||||
SecurityManager.userCanAccessProject user, project, (canAccess, privilegeLevel)->
|
|
||||||
allowedFreeTrial = true
|
|
||||||
if subscription? and subscription.freeTrial? and subscription.freeTrial.expiresAt?
|
|
||||||
allowedFreeTrial = !!subscription.freeTrial.allowed
|
|
||||||
if canAccess
|
|
||||||
timer.done()
|
|
||||||
res.render 'project/editor',
|
|
||||||
title: project.name
|
|
||||||
priority_title: true
|
|
||||||
bodyClasses: ["editor"]
|
|
||||||
project : project
|
|
||||||
owner : project.owner_ref
|
|
||||||
userObject : JSON.stringify({
|
|
||||||
id : user.id
|
|
||||||
email : user.email
|
|
||||||
first_name : user.first_name
|
|
||||||
last_name : user.last_name
|
|
||||||
referal_id : user.referal_id
|
|
||||||
subscription :
|
|
||||||
freeTrial: {allowed: allowedFreeTrial}
|
|
||||||
})
|
|
||||||
userSettingsObject: JSON.stringify({
|
|
||||||
mode : user.ace.mode
|
|
||||||
theme : user.ace.theme
|
|
||||||
project_id : project._id
|
|
||||||
fontSize : user.ace.fontSize
|
|
||||||
autoComplete: user.ace.autoComplete
|
|
||||||
spellCheckLanguage: user.ace.spellCheckLanguage
|
|
||||||
pdfViewer : user.ace.pdfViewer
|
|
||||||
docPositions: {}
|
|
||||||
oldHistory: !!user.featureSwitches?.oldHistory
|
|
||||||
})
|
|
||||||
sharelatexObject : JSON.stringify({
|
|
||||||
siteUrl: Settings.siteUrl,
|
|
||||||
jsPath: res.locals.jsPath
|
|
||||||
})
|
|
||||||
privilegeLevel: privilegeLevel
|
|
||||||
userCanSeeDropbox: user.featureSwitches.dropbox and project.owner_ref._id+"" == user._id+""
|
|
||||||
loadPdfjs: (user.ace.pdfViewer == "pdfjs")
|
|
||||||
chatUrl: Settings.apis.chat.url
|
|
||||||
anonymous: anonymous
|
|
||||||
languages: Settings.languages,
|
|
||||||
|
|
||||||
startBufferingRequest: (req, res, next) ->
|
startBufferingRequest: (req, res, next) ->
|
||||||
req.bufferedChunks = []
|
req.bufferedChunks = []
|
||||||
|
|
|
@ -96,7 +96,7 @@ module.exports = class Router
|
||||||
app.post '/project/new', AuthenticationController.requireLogin(), ProjectController.newProject
|
app.post '/project/new', AuthenticationController.requireLogin(), ProjectController.newProject
|
||||||
app.get '/project/new/template', TemplatesMiddlewear.saveTemplateDataInSession, AuthenticationController.requireLogin(), TemplatesController.createProjectFromZipTemplate
|
app.get '/project/new/template', TemplatesMiddlewear.saveTemplateDataInSession, AuthenticationController.requireLogin(), TemplatesController.createProjectFromZipTemplate
|
||||||
|
|
||||||
app.get '/Project/:Project_id', SecurityManager.requestCanAccessProject, Project.loadEditor
|
app.get '/Project/:Project_id', SecurityManager.requestCanAccessProject, ProjectController.loadEditor
|
||||||
app.get '/Project/:Project_id/file/:File_id', SecurityManager.requestCanAccessProject, FileStoreController.getFile
|
app.get '/Project/:Project_id/file/:File_id', SecurityManager.requestCanAccessProject, FileStoreController.getFile
|
||||||
|
|
||||||
app.get '/Project/:Project_id/output/output.pdf', SecurityManager.requestCanAccessProject, CompileController.downloadPdf
|
app.get '/Project/:Project_id/output/output.pdf', SecurityManager.requestCanAccessProject, CompileController.downloadPdf
|
||||||
|
|
|
@ -9,7 +9,6 @@
|
||||||
a(href='#exportSettings', data-toggle="tab") Export & Copy
|
a(href='#exportSettings', data-toggle="tab") Export & Copy
|
||||||
li
|
li
|
||||||
a(href='#deleteProjectTab', data-toggle="tab") Delete Project
|
a(href='#deleteProjectTab', data-toggle="tab") Delete Project
|
||||||
- if(userCanSeeDropbox)
|
|
||||||
li#manageDropboxSettiingsTabLink
|
li#manageDropboxSettiingsTabLink
|
||||||
a(href='#dropboxProjectSettings', data-toggle='tab') Dropbox
|
a(href='#dropboxProjectSettings', data-toggle='tab') Dropbox
|
||||||
span.label.label-warning beta
|
span.label.label-warning beta
|
||||||
|
|
|
@ -20,7 +20,6 @@ block content
|
||||||
a(href='#passwordReset', data-toggle="tab") Password
|
a(href='#passwordReset', data-toggle="tab") Password
|
||||||
li
|
li
|
||||||
a(href='#newsletter', data-toggle="tab") Newsletter
|
a(href='#newsletter', data-toggle="tab") Newsletter
|
||||||
- if(userCanSeeDropbox)
|
|
||||||
li
|
li
|
||||||
a(href='#dropboxSettings', data-toggle="tab") Dropbox
|
a(href='#dropboxSettings', data-toggle="tab") Dropbox
|
||||||
span.label.label-warning beta
|
span.label.label-warning beta
|
||||||
|
|
|
@ -12,7 +12,11 @@ describe "ProjectController", ->
|
||||||
|
|
||||||
@project_id = "123213jlkj9kdlsaj"
|
@project_id = "123213jlkj9kdlsaj"
|
||||||
|
|
||||||
@settings = {}
|
@settings =
|
||||||
|
apis:
|
||||||
|
chat:
|
||||||
|
url:"chat.com"
|
||||||
|
siteUrl: "mysite.com"
|
||||||
@ProjectDeleter =
|
@ProjectDeleter =
|
||||||
deleteProject: sinon.stub().callsArgWith(1)
|
deleteProject: sinon.stub().callsArgWith(1)
|
||||||
@ProjectDuplicator =
|
@ProjectDuplicator =
|
||||||
|
@ -26,6 +30,11 @@ describe "ProjectController", ->
|
||||||
getAllTags: sinon.stub()
|
getAllTags: sinon.stub()
|
||||||
@ProjectModel =
|
@ProjectModel =
|
||||||
findAllUsersProjects: sinon.stub()
|
findAllUsersProjects: sinon.stub()
|
||||||
|
findPopulatedById: sinon.stub()
|
||||||
|
@UserModel =
|
||||||
|
findById: sinon.stub()
|
||||||
|
@SecurityManager =
|
||||||
|
userCanAccessProject:sinon.stub()
|
||||||
@ProjectController = SandboxedModule.require modulePath, requires:
|
@ProjectController = SandboxedModule.require modulePath, requires:
|
||||||
"settings-sharelatex":@settings
|
"settings-sharelatex":@settings
|
||||||
"logger-sharelatex": log:->
|
"logger-sharelatex": log:->
|
||||||
|
@ -35,6 +44,8 @@ describe "ProjectController", ->
|
||||||
"../Subscription/SubscriptionLocator": @SubscriptionLocator
|
"../Subscription/SubscriptionLocator": @SubscriptionLocator
|
||||||
"../Tags/TagsHandler":@TagsHandler
|
"../Tags/TagsHandler":@TagsHandler
|
||||||
'../../models/Project': Project:@ProjectModel
|
'../../models/Project': Project:@ProjectModel
|
||||||
|
"../../models/User":User:@UserModel
|
||||||
|
"../../managers/SecurityManager":@SecurityManager
|
||||||
|
|
||||||
@user =
|
@user =
|
||||||
_id:"!£123213kjljkl"
|
_id:"!£123213kjljkl"
|
||||||
|
@ -47,7 +58,9 @@ describe "ProjectController", ->
|
||||||
user: @user
|
user: @user
|
||||||
body:
|
body:
|
||||||
projectName: @projectName
|
projectName: @projectName
|
||||||
@res = {}
|
@res =
|
||||||
|
locals:
|
||||||
|
jsPath:"js path here"
|
||||||
|
|
||||||
describe "deleteProject", ->
|
describe "deleteProject", ->
|
||||||
|
|
||||||
|
@ -69,8 +82,6 @@ describe "ProjectController", ->
|
||||||
done()
|
done()
|
||||||
@ProjectController.cloneProject @req, @res
|
@ProjectController.cloneProject @req, @res
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
describe "newProject", ->
|
describe "newProject", ->
|
||||||
|
|
||||||
it "should call the projectCreationHandler with createExampleProject", (done)->
|
it "should call the projectCreationHandler with createExampleProject", (done)->
|
||||||
|
@ -90,9 +101,6 @@ describe "ProjectController", ->
|
||||||
done()
|
done()
|
||||||
@ProjectController.newProject @req, @res
|
@ProjectController.newProject @req, @res
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
describe "projectListPage", ->
|
describe "projectListPage", ->
|
||||||
|
|
||||||
beforeEach ->
|
beforeEach ->
|
||||||
|
@ -105,8 +113,6 @@ describe "ProjectController", ->
|
||||||
@ProjectModel.findAllUsersProjects.callsArgWith(2, null, @projects, @collabertions, @readOnly)
|
@ProjectModel.findAllUsersProjects.callsArgWith(2, null, @projects, @collabertions, @readOnly)
|
||||||
|
|
||||||
it "should render the project/list page", (done)->
|
it "should render the project/list page", (done)->
|
||||||
|
|
||||||
@req.body.template = "example"
|
|
||||||
@res.render = (pageName, opts)=>
|
@res.render = (pageName, opts)=>
|
||||||
pageName.should.equal "project/list"
|
pageName.should.equal "project/list"
|
||||||
done()
|
done()
|
||||||
|
@ -118,7 +124,6 @@ describe "ProjectController", ->
|
||||||
done()
|
done()
|
||||||
@ProjectController.projectListPage @req, @res
|
@ProjectController.projectListPage @req, @res
|
||||||
|
|
||||||
|
|
||||||
it "should send the projects", (done)->
|
it "should send the projects", (done)->
|
||||||
@res.render = (pageName, opts)=>
|
@res.render = (pageName, opts)=>
|
||||||
opts.projects.length.should.equal (@projects.length + @collabertions.length + @readOnly.length)
|
opts.projects.length.should.equal (@projects.length + @collabertions.length + @readOnly.length)
|
||||||
|
@ -126,3 +131,68 @@ describe "ProjectController", ->
|
||||||
@ProjectController.projectListPage @req, @res
|
@ProjectController.projectListPage @req, @res
|
||||||
|
|
||||||
|
|
||||||
|
describe "loadEditor", ->
|
||||||
|
beforeEach ->
|
||||||
|
@settings.editorIsOpen = true
|
||||||
|
@project =
|
||||||
|
name:"my proj"
|
||||||
|
_id:"213123kjlkj"
|
||||||
|
@user =
|
||||||
|
_id:"123kj21k3lj"
|
||||||
|
ace:
|
||||||
|
fontSize:"massive"
|
||||||
|
theme:"sexy"
|
||||||
|
email: "bob@bob.com"
|
||||||
|
@ProjectModel.findPopulatedById.callsArgWith 1, null, @project
|
||||||
|
@UserModel.findById.callsArgWith(1, null, @user)
|
||||||
|
@SubscriptionLocator.getUsersSubscription.callsArgWith(1, null, {})
|
||||||
|
@SecurityManager.userCanAccessProject.callsArgWith 2, true, "owner"
|
||||||
|
|
||||||
|
it "should render the project/editor page", (done)->
|
||||||
|
@res.render = (pageName, opts)=>
|
||||||
|
pageName.should.equal "project/editor"
|
||||||
|
done()
|
||||||
|
@ProjectController.loadEditor @req, @res
|
||||||
|
|
||||||
|
it "should add the project onto the opts", (done)->
|
||||||
|
@res.render = (pageName, opts)=>
|
||||||
|
opts.project.should.equal @project
|
||||||
|
done()
|
||||||
|
@ProjectController.loadEditor @req, @res
|
||||||
|
|
||||||
|
it "should add userObject", (done)->
|
||||||
|
@res.render = (pageName, opts)=>
|
||||||
|
userObject = JSON.parse(opts.userObject)
|
||||||
|
userObject.email.should.equal @user.email
|
||||||
|
done()
|
||||||
|
@ProjectController.loadEditor @req, @res
|
||||||
|
|
||||||
|
it "should add on userSettingsObject", (done)->
|
||||||
|
@res.render = (pageName, opts)=>
|
||||||
|
userSettingsObject = JSON.parse(opts.userSettingsObject)
|
||||||
|
userSettingsObject.project_id.should.equal @project._id
|
||||||
|
userSettingsObject.fontSize.should.equal @user.ace.fontSize
|
||||||
|
userSettingsObject.theme.should.equal @user.ace.theme
|
||||||
|
done()
|
||||||
|
@ProjectController.loadEditor @req, @res
|
||||||
|
|
||||||
|
it "should add sharelatexObject", (done)->
|
||||||
|
@res.render = (pageName, opts)=>
|
||||||
|
sharelatexObject = JSON.parse(opts.sharelatexObject)
|
||||||
|
sharelatexObject.siteUrl.should.equal @settings.siteUrl
|
||||||
|
done()
|
||||||
|
@ProjectController.loadEditor @req, @res
|
||||||
|
|
||||||
|
it "should render the closed page if the editor is closed", (done)->
|
||||||
|
@settings.editorIsOpen = false
|
||||||
|
@res.render = (pageName, opts)=>
|
||||||
|
pageName.should.equal "general/closed"
|
||||||
|
done()
|
||||||
|
@ProjectController.loadEditor @req, @res
|
||||||
|
|
||||||
|
it "should not render the page if the project can not be accessed", (done)->
|
||||||
|
@SecurityManager.userCanAccessProject = sinon.stub().callsArgWith 2, false
|
||||||
|
@res.send = (resCode, opts)=>
|
||||||
|
resCode.should.equal 401
|
||||||
|
done()
|
||||||
|
@ProjectController.loadEditor @req, @res
|
||||||
|
|
Loading…
Reference in a new issue