const { findLast, get } = require('lodash') const Fragment = require('./Fragment') class Collection { constructor({ collection = {} }) { this.collection = collection } async updateStatusByRecommendation({ recommendation, isHandlingEditor = false, fragments, }) { let newStatus if (isHandlingEditor) { newStatus = 'pendingApproval' if (['minor', 'major'].includes(recommendation)) { newStatus = 'revisionRequested' } } else { if (recommendation === 'minor') { newStatus = this.hasAtLeastOneReviewReport(fragments) ? 'reviewCompleted' : 'heAssigned' } if (recommendation === 'major') { newStatus = 'underReview' } } return this.updateStatus({ newStatus }) } async updateFinalStatusByRecommendation({ recommendation }) { let newStatus switch (recommendation) { case 'reject': newStatus = 'rejected' break case 'publish': newStatus = 'accepted' break case 'return-to-handling-editor': newStatus = 'reviewCompleted' break default: break } await this.updateStatus({ newStatus }) } async updateStatus({ newStatus }) { this.collection.status = newStatus await this.collection.save() } async addHandlingEditor({ user, invitation }) { this.collection.handlingEditor = { id: user.id, name: `${user.firstName} ${user.lastName}`, invitedOn: invitation.invitedOn, respondedOn: invitation.respondedOn, email: user.email, hasAnswer: invitation.hasAnswer, isAccepted: invitation.isAccepted, } await this.updateStatus({ newStatus: 'heInvited' }) } async updateHandlingEditor({ isAccepted }) { const { collection: { handlingEditor } } = this handlingEditor.hasAnswer = true handlingEditor.isAccepted = isAccepted handlingEditor.respondedOn = Date.now() const newStatus = isAccepted ? 'heAssigned' : 'submitted' await this.updateStatus({ newStatus }) } async updateStatusByNumberOfReviewers({ invitations }) { const reviewerInvitations = invitations.filter( inv => inv.role === 'reviewer', ) if (reviewerInvitations.length === 0) await this.updateStatus({ newStatus: 'heAssigned' }) } async updateStatusOnRecommendation({ isEditorInChief, recommendation }) { if (isEditorInChief) { if (recommendation === 'return-to-handling-editor') { return this.updateStatus({ newStatus: 'reviewCompleted' }) } return this.updateFinalStatusByRecommendation({ recommendation, }) } return this.updateStatusByRecommendation({ recommendation, isHandlingEditor: true, }) } getHELastName() { const [firstName, lastName] = this.collection.handlingEditor.name.split(' ') return lastName || firstName } // eslint-disable-next-line class-methods-use-this hasAtLeastOneReviewReport(fragments) { return fragments.some(fragment => new Fragment({ fragment }).hasReviewReport(), ) } canHEMakeRecommendation(fragments, fragmentHelper) { if (this.collection.fragments.length === 1) { return fragmentHelper.hasReviewReport() } const previousVersionRecommendations = get( fragments[fragments.length - 2], 'recommendations', [], ) const lastRecommendationByHE = findLast( previousVersionRecommendations, recommendation => recommendation.userId === this.collection.handlingEditor.id && recommendation.recommendationType === 'editorRecommendation', ) if (lastRecommendationByHE.recommendation === 'minor') { return this.hasAtLeastOneReviewReport(fragments) } else if (lastRecommendationByHE.recommendation === 'major') { return fragmentHelper.hasReviewReport() } } async getAllFragments({ FragmentModel }) { return Promise.all( this.collection.fragments.map(async fragment => FragmentModel.find(fragment), ), ) } } module.exports = Collection