import {InternMap} from "internmap"; import identity from "./identity.js"; export default function group(values, ...keys) { return nest(values, identity, identity, keys); } export function groups(values, ...keys) { return nest(values, Array.from, identity, keys); } function flatten(groups, keys) { for (let i = 1, n = keys.length; i < n; ++i) { groups = groups.flatMap(g => g.pop().map(([key, value]) => [...g, key, value])); } return groups; } export function flatGroup(values, ...keys) { return flatten(groups(values, ...keys), keys); } export function flatRollup(values, reduce, ...keys) { return flatten(rollups(values, reduce, ...keys), keys); } export function rollup(values, reduce, ...keys) { return nest(values, identity, reduce, keys); } export function rollups(values, reduce, ...keys) { return nest(values, Array.from, reduce, keys); } export function index(values, ...keys) { return nest(values, identity, unique, keys); } export function indexes(values, ...keys) { return nest(values, Array.from, unique, keys); } function unique(values) { if (values.length !== 1) throw new Error("duplicate key"); return values[0]; } function nest(values, map, reduce, keys) { return (function regroup(values, i) { if (i >= keys.length) return reduce(values); const groups = new InternMap(); const keyof = keys[i++]; let index = -1; for (const value of values) { const key = keyof(value, ++index, values); const group = groups.get(key); if (group) group.push(value); else groups.set(key, [value]); } for (const [key, values] of groups) { groups.set(key, regroup(values, i)); } return map(groups); })(values, 0); }