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
1 month 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;
|
||
|
}
|