// Licensed to the Apache Software Foundation (ASF) under one // or more contributor license agreements. See the NOTICE file // distributed with this work for additional information // regarding copyright ownership. The ASF licenses this file // to you under the Apache License, Version 2.0 (the // "License"); you may not use this file except in compliance // with the License. You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, // software distributed under the License is distributed on an // "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY // KIND, either express or implied. See the License for the // specific language governing permissions and limitations // under the License. import { Field } from '../schema.js'; import { DataType, Map_, Struct } from '../type.js'; import { Builder, VariableWidthBuilder } from '../builder.js'; /** @ignore */ type MapValue = Map_['TValue']; /** @ignore */ type MapValues = Map | undefined>; /** @ignore */ type MapValueExt = MapValue | { [key: string]: V } | { [key: number]: V }; /** @ignore */ export class MapBuilder extends VariableWidthBuilder, TNull> { declare protected _pending: MapValues | undefined; public set(index: number, value: MapValueExt | TNull) { return super.set(index, value as MapValue | TNull); } public setValue(index: number, value: MapValueExt) { const row = (value instanceof Map ? value : new Map(Object.entries(value))) as MapValue; const pending = this._pending || (this._pending = new Map() as MapValues); const current = pending.get(index) as Map | undefined; current && (this._pendingLength -= current.size); this._pendingLength += row.size; pending.set(index, row); } public addChild(child: Builder>, name = `${this.numChildren}`) { if (this.numChildren > 0) { throw new Error('ListBuilder can only have one child.'); } this.children[this.numChildren] = child; this.type = new Map_(new Field(name, child.type, true), this.type.keysSorted); return this.numChildren - 1; } protected _flushPending(pending: MapValues) { const offsets = this._offsets; const [child] = this.children; for (const [index, value] of pending) { if (value === undefined) { offsets.set(index, 0); } else { let { [index]: idx, [index + 1]: end } = offsets.set(index, value.size).buffer; for (const val of value.entries()) { child.set(idx, val); if (++idx >= end) break; } } } } }