You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					91 lines
				
				2.8 KiB
			
		
		
			
		
	
	
					91 lines
				
				2.8 KiB
			| 
											3 years ago
										 | 'use strict'; | ||
|  | 
 | ||
|  | const arrayAtomicsSymbol = require('../helpers/symbols').arrayAtomicsSymbol; | ||
|  | const sessionNewDocuments = require('../helpers/symbols').sessionNewDocuments; | ||
|  | 
 | ||
|  | module.exports = function trackTransaction(schema) { | ||
|  |   schema.pre('save', function() { | ||
|  |     const session = this.$session(); | ||
|  |     if (session == null) { | ||
|  |       return; | ||
|  |     } | ||
|  |     if (session.transaction == null || session[sessionNewDocuments] == null) { | ||
|  |       return; | ||
|  |     } | ||
|  | 
 | ||
|  |     if (!session[sessionNewDocuments].has(this)) { | ||
|  |       const initialState = {}; | ||
|  |       if (this.isNew) { | ||
|  |         initialState.isNew = true; | ||
|  |       } | ||
|  |       if (this.$__schema.options.versionKey) { | ||
|  |         initialState.versionKey = this.get(this.$__schema.options.versionKey); | ||
|  |       } | ||
|  | 
 | ||
|  |       initialState.modifiedPaths = new Set(Object.keys(this.$__.activePaths.states.modify)); | ||
|  |       initialState.atomics = _getAtomics(this); | ||
|  | 
 | ||
|  |       session[sessionNewDocuments].set(this, initialState); | ||
|  |     } else { | ||
|  |       const state = session[sessionNewDocuments].get(this); | ||
|  | 
 | ||
|  |       for (const path of Object.keys(this.$__.activePaths.states.modify)) { | ||
|  |         state.modifiedPaths.add(path); | ||
|  |       } | ||
|  |       state.atomics = _getAtomics(this, state.atomics); | ||
|  |     } | ||
|  |   }); | ||
|  | }; | ||
|  | 
 | ||
|  | function _getAtomics(doc, previous) { | ||
|  |   const pathToAtomics = new Map(); | ||
|  |   previous = previous || new Map(); | ||
|  | 
 | ||
|  |   const pathsToCheck = Object.keys(doc.$__.activePaths.init).concat(Object.keys(doc.$__.activePaths.modify)); | ||
|  | 
 | ||
|  |   for (const path of pathsToCheck) { | ||
|  |     const val = doc.$__getValue(path); | ||
|  |     if (val != null && | ||
|  |         val instanceof Array && | ||
|  |         val.isMongooseDocumentArray && | ||
|  |         val.length && | ||
|  |         val[arrayAtomicsSymbol] != null && | ||
|  |         Object.keys(val[arrayAtomicsSymbol]).length > 0) { | ||
|  |       const existing = previous.get(path) || {}; | ||
|  |       pathToAtomics.set(path, mergeAtomics(existing, val[arrayAtomicsSymbol])); | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   const dirty = doc.$__dirty(); | ||
|  |   for (const dirt of dirty) { | ||
|  |     const path = dirt.path; | ||
|  | 
 | ||
|  |     const val = dirt.value; | ||
|  |     if (val != null && val[arrayAtomicsSymbol] != null && Object.keys(val[arrayAtomicsSymbol]).length > 0) { | ||
|  |       const existing = previous.get(path) || {}; | ||
|  |       pathToAtomics.set(path, mergeAtomics(existing, val[arrayAtomicsSymbol])); | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   return pathToAtomics; | ||
|  | } | ||
|  | 
 | ||
|  | function mergeAtomics(destination, source) { | ||
|  |   destination = destination || {}; | ||
|  | 
 | ||
|  |   if (source.$pullAll != null) { | ||
|  |     destination.$pullAll = (destination.$pullAll || []).concat(source.$pullAll); | ||
|  |   } | ||
|  |   if (source.$push != null) { | ||
|  |     destination.$push = destination.$push || {}; | ||
|  |     destination.$push.$each = (destination.$push.$each || []).concat(source.$push.$each); | ||
|  |   } | ||
|  |   if (source.$addToSet != null) { | ||
|  |     destination.$addToSet = (destination.$addToSet || []).concat(source.$addToSet); | ||
|  |   } | ||
|  |   if (source.$set != null) { | ||
|  |     destination.$set = Object.assign(destination.$set || {}, source.$set); | ||
|  |   } | ||
|  | 
 | ||
|  |   return destination; | ||
|  | } |