All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.aerospike.client.lua.LuaMapLib Maven / Gradle / Ivy

The newest version!
/*
 * Copyright 2012-2021 Aerospike, Inc.
 *
 * Portions may be licensed to Aerospike, Inc. under one or more contributor
 * license agreements WHICH ARE COMPATIBLE WITH THE APACHE LICENSE, VERSION 2.0.
 *
 * Licensed 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.
 */
package com.aerospike.client.lua;

import java.util.HashMap;
import java.util.Iterator;
import java.util.Map.Entry;

import org.luaj.vm2.LuaFunction;
import org.luaj.vm2.LuaTable;
import org.luaj.vm2.LuaValue;
import org.luaj.vm2.Varargs;
import org.luaj.vm2.lib.OneArgFunction;
import org.luaj.vm2.lib.VarArgFunction;

public final class LuaMapLib extends OneArgFunction {

	private final LuaInstance instance;

	public LuaMapLib(LuaInstance instance) {
		this.instance = instance;
		instance.load(new MetaLib(instance));
	}

	public LuaValue call(LuaValue env) {
		LuaTable meta = new LuaTable(0,2);
		meta.set("__call", new create(instance));

		LuaTable table = new LuaTable(0,11);
		table.setmetatable(meta);
		table.set("create", new create(instance));

		new mapcode(table, 0, "size");
		new mapcode(table, 4, "pairs");
		new mapcode(table, 5, "keys");
		new mapcode(table, 6, "values");
		new mapcode(table, 7, "remove");
		new mapcode(table, 8, "clone");
		new mapcode(table, 9, "merge");
		new mapcode(table, 10, "diff");

		instance.registerPackage("map", table);
		return table;
	}

	private static final class MetaLib extends OneArgFunction {
		private final LuaInstance instance;

		public MetaLib(LuaInstance instance) {
			this.instance = instance;
		}

		public LuaValue call(LuaValue env) {
			LuaTable meta = new LuaTable(0,5);
			new mapcode(meta, 0, "__len");
			new mapcode(meta, 1, "__tostring");
			new mapcode(meta, 2, "__index");
			new mapcode(meta, 3, "__newindex");
			instance.registerPackage("Map", meta);
			return meta;
		}
	}

	private static final class create extends VarArgFunction {
		private final LuaInstance instance;

		public create(LuaInstance instance) {
			this.instance = instance;
		}

		@Override
		public Varargs invoke(Varargs args) {
			int capacity = 32;

			if (args.isnumber(1)) {
				capacity = args.toint(1);
			}

			LuaMap map = new LuaMap(instance, new HashMap(capacity));

			if (args.istable(2)) {
				LuaTable table = args.checktable(2);
				LuaValue k = LuaValue.NIL;

				while (true) {
					 Varargs n = table.next(k);

					 if ((k = n.arg1()).isnil())
						 break;

					 LuaValue v = n.arg(2);
					 map.put(k, v);
				 }
			}
			return map;
		}
	}

	private static final class mapcode extends VarArgFunction {
		public mapcode(LuaTable table, int id, String name) {
			super.opcode = id;
			super.name = name;
			table.set(name, this);
		}

		@Override
		public Varargs invoke(Varargs args) {
			LuaMap map = (LuaMap)args.arg(1);

			switch (opcode) {
			case 0: // __len, size
				return map.size();

			case 1: // __tostring
				return map.toLuaString();

			case 2: // __index
				return map.get(args.arg(2));

			case 3: // __newindex
				map.put(args.arg(2), args.arg(3));
				return NIL;

			case 4: // pairs
				return new nextLuaValue(map.entrySetIterator());

			case 5: // keys
				return new LuaListLib.nextLuaValue(map.keySetIterator());

			case 6: // values
				return new LuaListLib.nextLuaValue(map.valuesIterator());

			case 7: // remove
				map.remove(args.arg(2));
				return NIL;

			case 8: // clone
				return map.clone();

			case 9: // merge
				return map.merge((LuaMap)args.arg(2), (LuaFunction)args.arg(3));

			case 10: // diff
				return map.diff((LuaMap)args.arg(2));

			default:
				return NIL;
			}
		}
	}

	private static final class nextLuaValue extends VarArgFunction {
		private final Iterator> iter;

		public nextLuaValue(Iterator> iter) {
			this.iter = iter;
		}

		@Override
		public Varargs invoke(Varargs args) {
			if (iter.hasNext()) {
				Entry entry = iter.next();
				return LuaValue.varargsOf(new LuaValue[] {entry.getKey(), entry.getValue()});
			}
			return NONE;
		}
	}
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy