UNPKG

mingo

Version:

MongoDB query language for in-memory objects

46 lines (45 loc) 1.52 kB
import { compare, has, isEqual, isNumber, isObject, resolve } from "../../util"; import { applyUpdate, clone, DEFAULT_OPTIONS, walkExpression } from "./_internal"; const MODIFIERS = ["$each", "$slice", "$sort", "$position"]; const $push = (obj, expr, arrayFilters = [], options = DEFAULT_OPTIONS) => { return walkExpression(expr, arrayFilters, options, (val, node, queries) => { const args = { $each: [val] }; if (isObject(val) && MODIFIERS.some((m) => has(val, m))) { Object.assign(args, val); } return applyUpdate( obj, node, queries, (o, k) => { const arr = o[k] ||= []; const prev = arr.slice(0, args.$slice || arr.length); const oldsize = arr.length; const pos = isNumber(args.$position) ? args.$position : arr.length; arr.splice(pos, 0, ...clone(args.$each, options)); if (args.$sort) { const sortKey = isObject(args.$sort) ? Object.keys(args.$sort || {}).pop() : ""; const order = !sortKey ? args.$sort : args.$sort[sortKey]; const f = !sortKey ? (a) => a : (a) => resolve(a, sortKey); arr.sort((a, b) => order * compare(f(a), f(b))); } if (isNumber(args.$slice)) { if (args.$slice < 0) arr.splice(0, arr.length + args.$slice); else arr.splice(args.$slice); } return oldsize != arr.length || !isEqual(prev, arr); }, { descendArray: true, buildGraph: true } ); }); }; export { $push };