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

com.aoindustries.aoserv.client.GlobalTable Maven / Gradle / Ivy

There is a newer version: 1.92.0
Show newest version
/*
 * aoserv-client - Java client for the AOServ Platform.
 * Copyright (C) 2001-2009, 2016, 2017, 2018, 2019  AO Industries, Inc.
 *     [email protected]
 *     7262 Bull Pen Cir
 *     Mobile, AL 36695
 *
 * This file is part of aoserv-client.
 *
 * aoserv-client is free software: you can redistribute it and/or modify
 * it under the terms of the GNU Lesser General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *
 * aoserv-client is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public License
 * along with aoserv-client.  If not, see .
 */
package com.aoindustries.aoserv.client;

import com.aoindustries.aoserv.client.schema.AoservProtocol;
import com.aoindustries.aoserv.client.schema.Table;
import java.io.IOException;
import java.sql.SQLException;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.BitSet;
import java.util.Collections;
import java.util.HashMap;
import java.util.Iterator;
import java.util.List;
import java.util.Map;

/**
 * A GlobalTable is shared between all users.
 * The data is cached so that subsequent lookups need not
 * access the server.
 *
 * @author  AO Industries, Inc.
 */
abstract public class GlobalTable> extends AOServTable {

	private static final int numTables = Table.TableID.values().length;

	/**
	 * Each table has its own lock because we were getting deadlocks with one lock on GlobalTable.class.
	 */
	private static class Lock {}
	private static final Lock[] locks = new Lock[numTables];
	static {
		for(int c=0;c-1 if not yet loaded.
	 */
	private static final long[] lastLoadeds = new long[numTables];
	static {
		Arrays.fill(lastLoadeds, -1);
	}

	/**
	 * The internal objects are stored in a HashMap
	 * based on the server and then the table ID, and then the
	 * column number.
	 */
	private static final List>> tableHashes=new ArrayList<>(numTables);
	static {
		for(int c=0;cHashMap
	 * based on the server and then the table ID, and then the
	 * column number.  Each element of the HashMap is
	 * a ArrayList or AOServObject[].
	 * All of the List stored here are unmodifiable.
	 */
	private static final List>>>> indexHashes=new ArrayList<>(numTables);
	static {
		for(int c=0;c is unmodifiable.
	 */
	private static final List>> tableObjs=new ArrayList<>(numTables);
	static {
		for(int c=0;c clazz) {
		super(connector, clazz);
	}

	/**
	 * Gets the number of accessible rows in the table or -1 if the
	 * table is not yet loaded.
	 */
	public final int getGlobalRowCount() {
		int ordinal = getTableID().ordinal();
		List> objs;
		synchronized(locks[ordinal]) {
			synchronized(tableObjs) {
				objs=tableObjs.get(ordinal);
			}
		}
		if(objs!=null) return objs.size();
		return -1;
	}

	@Override
	@SuppressWarnings({"unchecked"})
	final public List getIndexedRows(int col, Object value) throws IOException, SQLException {
		Table.TableID tableID=getTableID();
		int ordinal = tableID.ordinal();
		synchronized(locks[ordinal]) {
			validateCache();

			BitSet tableLoadeds=indexLoadeds[ordinal];
			if(tableLoadeds==null) indexLoadeds[ordinal]=tableLoadeds=new BitSet(col+1);
			boolean isHashed=tableLoadeds.get(col);

			List>>> tableValues;
			synchronized(indexHashes) {
				tableValues = indexHashes.get(ordinal);
				if(tableValues==null) indexHashes.set(ordinal, tableValues=new ArrayList<>(col+1));
			}
			while(tableValues.size()<=col) tableValues.add(null);
			Map>> colIndexes=tableValues.get(col);
			if(colIndexes==null) tableValues.set(col, colIndexes=new HashMap<>());

			if(!isHashed) {
				// Build the modifiable lists in a temporary Map
				Map>> modifiableIndexes=new HashMap<>();
				for(GlobalObject O : getRows()) {
					Object cvalue=O.getColumn(col);
					List> list=modifiableIndexes.get(cvalue);
					if(list==null) modifiableIndexes.put(cvalue, list=new ArrayList<>());
					list.add(O);
				}
				// Wrap each of the newly-created indexes to be unmodifiable
				colIndexes.clear();
				Iterator keys=modifiableIndexes.keySet().iterator();
				while(keys.hasNext()) {
					Object key=keys.next();
					List> list=modifiableIndexes.get(key);
					colIndexes.put(key, Collections.unmodifiableList(list));
				}
				tableLoadeds.set(col);
			}
			// This returns unmodifable lists.
			List> list=colIndexes.get(value);
			if(list==null) return Collections.emptyList();
			return (List)list;
		}
	}

	@Override
	@SuppressWarnings({"unchecked"})
	final protected V getUniqueRowImpl(int col, Object value) throws SQLException, IOException {
		if(value == null) return null;
		Table.TableID tableID=getTableID();
		int ordinal = tableID.ordinal();
		synchronized(locks[ordinal]) {
			validateCache();

			BitSet tableLoadeds=hashLoadeds[ordinal];
			if(tableLoadeds==null) hashLoadeds[ordinal]=tableLoadeds=new BitSet(col+1);
			boolean isHashed=tableLoadeds.get(col);

			List table=getRows();
			int size=table.size();

			List> tableValues;
			synchronized(tableHashes) {
				tableValues = tableHashes.get(ordinal);
				if(tableValues==null) tableHashes.set(ordinal, tableValues=new ArrayList<>(col+1));
			}
			while(tableValues.size()<=col) tableValues.add(null);
			Map colValues=tableValues.get(col);
			if(colValues==null) tableValues.set(col, colValues=new HashMap<>(size*13/9));

			if(!isHashed) {
				colValues.clear();
				for(int c=0;c getRows() throws IOException, SQLException {
		Table.TableID tableID = getTableID();
		int ordinal = tableID.ordinal();
		// We synchronize here to make sure tableObjs is not cleared between validateCache and get, but only on a per-table ID basis
		synchronized(locks[ordinal]) {
			validateCache();
			synchronized(tableObjs) {
				List> objs=tableObjs.get(ordinal);
				return (List)objs;
			}
		}
	}

	@Override
	public List getRowsCopy() throws IOException, SQLException {
		return new ArrayList<>(getRows());
	}

	/**
	 * Determines if the contents are currently hashed in a hashmap.
	 */
	boolean isHashed(int column) {
		int ordinal = getTableID().ordinal();
		synchronized(locks[ordinal]) {
			BitSet table=hashLoadeds[ordinal];
			return table!=null && table.get(column);
		}
	}

	/**
	 * Determines if the contents are currently indexed.
	 */
	boolean isIndexed(int column) {
		int ordinal = getTableID().ordinal();
		synchronized(locks[ordinal]) {
			BitSet table=indexLoadeds[ordinal];
			return table!=null && table.get(column);
		}
	}

	@Override
	final public boolean isLoaded() {
		Table.TableID tableID=getTableID();
		int ordinal = tableID.ordinal();
		synchronized(locks[ordinal]) {
			return lastLoadeds[ordinal] != -1;
		}
	}

	@Override
	public void clearCache() {
		super.clearCache();
		Table.TableID tableID=getTableID();
		int ordinal = tableID.ordinal();
		synchronized(locks[ordinal]) {
			lastLoadeds[ordinal] = -1;
		}
	}

	/**
	 * Reloads the cache if the cache time has expired.
	 */
	@SuppressWarnings({"unchecked"})
	private void validateCache() throws IOException, SQLException {
		Table.TableID tableID=getTableID();
		int ordinal = tableID.ordinal();
		synchronized(locks[ordinal]) {
			long currentTime=System.currentTimeMillis();
			long lastLoaded=lastLoadeds[ordinal];
			if(lastLoaded==-1) {
				List> list=(List)getObjects(true, AoservProtocol.CommandID.GET_TABLE, ordinal);
				synchronized(tableObjs) {
					tableObjs.set(ordinal, Collections.unmodifiableList(list));
				}
				BitSet loaded=hashLoadeds[ordinal];
				if(loaded!=null) loaded.clear();
				BitSet indexed=indexLoadeds[ordinal];
				if(indexed!=null) indexed.clear();
				lastLoadeds[ordinal]=currentTime;
			}
		}
	}
}