/*******************************************************************************
 * Copyright (c) 2001, 2005 IBM Corporation and others.
 * All rights reserved. This program and the accompanying materials
 * are made available under the terms of the Eclipse Public License v1.0
 * which accompanies this distribution, and is available at
 * http://www.eclipse.org/legal/epl-v10.html
 * 
 * Contributors:
 *     IBM Corporation - initial API and implementation
 *******************************************************************************/
package org.eclipse.jem.internal.proxy.remote;

/*


 */

import java.text.MessageFormat;
import java.util.*;

import org.eclipse.core.runtime.IStatus;
import org.eclipse.core.runtime.Status;
import org.eclipse.jem.internal.proxy.core.*;
import org.eclipse.jem.internal.proxy.core.ExpressionProxy.ProxyEvent;
import org.eclipse.jem.internal.proxy.common.CommandException;
import org.eclipse.jem.internal.proxy.common.MapTypes;
import org.eclipse.jem.internal.proxy.common.remote.CommandErrorException;
import org.eclipse.jem.internal.proxy.common.remote.Commands;

/**
 * We are a beanTypeProxyFactory for creating REMBeanTypeProxy objects Primitive types as well as common types like String are cached for speed of
 * lookup. We are the standard one that the VCE uses. Creation date: (12/3/99 2:32:48 PM)
 * 
 * @author: Joe Winchester
 */
public final class REMStandardBeanTypeProxyFactory implements IStandardBeanTypeProxyFactory {

	protected final REMProxyFactoryRegistry fFactoryRegistry;

	protected REMStandardBeanProxyFactory fBeanFactory;

	// Hashtable to cache proxies for classes so they are found on second and subsequent lookups
	protected Map fBeanProxies = new HashMap(1000);

	// A Cache of bean type proxies that should not ever be released. These are the standard ones
	// that we create here. They are never released because they wouldn't be correctly re-created
	// if they were. Also they are standard ones with standard ID's that don't change so they
	// don't need to be re-created later.
	protected Set fPermanentProxies = new HashSet(30);

	// Cache of requested but not found bean types. If not maintaining the list, this variable will be null.
	// The values are strings (classnames in JNI format).
	protected Set fNotFoundTypes = null;

	// Cached copy of a few typical bean type proxies.
	REMBeanTypeProxy objectClass;

	REMClassBeanTypeProxy classClass;

	REMVoidBeanTypeProxy voidType;

	REMBooleanTypeBeanTypeProxy booleanType;

	REMBooleanClassBeanTypeProxy booleanClass;

	REMIntegerTypeBeanTypeProxy intType;

	REMIntegerClassBeanTypeProxy integerClass;

	REMByteTypeBeanTypeProxy byteType;

	REMByteClassBeanTypeProxy byteClass;

	REMShortClassBeanTypeProxy shortClass;

	REMShortTypeBeanTypeProxy shortType;

	REMLongClassBeanTypeProxy longClass;

	REMLongTypeBeanTypeProxy longType;

	REMDoubleClassBeanTypeProxy doubleClass;

	REMDoubleTypeBeanTypeProxy doubleType;

	REMFloatClassBeanTypeProxy floatClass;

	REMFloatTypeBeanTypeProxy floatType;

	REMBigDecimalBeanTypeProxy bigDecimalClass;

	REMBigIntegerBeanTypeProxy bigIntegerClass;

	REMCharacterTypeBeanTypeProxy charType;

	REMCharacterClassBeanTypeProxy characterClass;

	REMStringBeanTypeProxy stringClass;

	/**
	 * Available to all REM beantypes.
	 */
	public REMProxyConstants proxyConstants;

	/**
	 * IDEBeanTypeProxyFactory constructor comment.
	 */
	REMStandardBeanTypeProxyFactory(REMProxyFactoryRegistry aRegistry) {
		fFactoryRegistry = aRegistry;
		aRegistry.registerBeanTypeProxyFactory(this);

		proxyConstants = new REMProxyConstants(aRegistry);

		// Now initialize the cache.
		objectClass = new REMBeanTypeProxy(fFactoryRegistry, new Integer(Commands.OBJECT_CLASS), Object.class.getName(), null);
		classClass = new REMClassBeanTypeProxy(fFactoryRegistry, objectClass);
		voidType = new REMVoidBeanTypeProxy(fFactoryRegistry);
		booleanType = new REMBooleanTypeBeanTypeProxy(fFactoryRegistry);
		booleanClass = new REMBooleanClassBeanTypeProxy(fFactoryRegistry, objectClass);
		IBeanTypeProxy numberClass = objectClass.newBeanTypeForClass(new Integer(Commands.NUMBER_CLASS), Number.class.getName(), true);
		intType = new REMIntegerTypeBeanTypeProxy(fFactoryRegistry);
		integerClass = new REMIntegerClassBeanTypeProxy(fFactoryRegistry, numberClass);
		byteType = new REMByteTypeBeanTypeProxy(fFactoryRegistry);
		byteClass = new REMByteClassBeanTypeProxy(fFactoryRegistry, numberClass);
		shortType = new REMShortTypeBeanTypeProxy(fFactoryRegistry);
		shortClass = new REMShortClassBeanTypeProxy(fFactoryRegistry, numberClass);
		longType = new REMLongTypeBeanTypeProxy(fFactoryRegistry);
		longClass = new REMLongClassBeanTypeProxy(fFactoryRegistry, numberClass);
		doubleType = new REMDoubleTypeBeanTypeProxy(fFactoryRegistry);
		doubleClass = new REMDoubleClassBeanTypeProxy(fFactoryRegistry, numberClass);
		floatType = new REMFloatTypeBeanTypeProxy(fFactoryRegistry);
		floatClass = new REMFloatClassBeanTypeProxy(fFactoryRegistry, numberClass);
		bigDecimalClass = new REMBigDecimalBeanTypeProxy(fFactoryRegistry, numberClass);
		bigIntegerClass = new REMBigIntegerBeanTypeProxy(fFactoryRegistry, numberClass);
		charType = new REMCharacterTypeBeanTypeProxy(fFactoryRegistry);
		characterClass = new REMCharacterClassBeanTypeProxy(fFactoryRegistry, objectClass);
		stringClass = new REMStringBeanTypeProxy(fFactoryRegistry, objectClass);
		IBeanTypeProxy throwableClass = new REMThrowableBeanTypeProxy(fFactoryRegistry, new Integer(Commands.THROWABLE_CLASS), Throwable.class
				.getName(), objectClass);
		IBeanTypeProxy threadClass = objectClass.newBeanTypeForClass(new Integer(Commands.THREAD_CLASS), Thread.class.getName(), false);
		IBeanTypeProxy expressionProcessorControllerClass = objectClass.newBeanTypeForClass(new Integer(Commands.EXPRESSIONPROCESSERCONTROLLER_CLASS), "org.eclipse.jem.internal.proxy.vm.remote.ExpressionProcesserController", false); //$NON-NLS-1$

		// Initialize the hashtable with the primitives, their lang equivalents, and also common classes like String

		// Primitives
		fBeanProxies.put(voidType.getTypeName(), voidType);
		fBeanProxies.put(intType.getTypeName(), intType);
		fBeanProxies.put(booleanType.getTypeName(), booleanType);
		fBeanProxies.put(charType.getTypeName(), charType);
		fBeanProxies.put(byteType.getTypeName(), byteType);
		fBeanProxies.put(shortType.getTypeName(), shortType);
		fBeanProxies.put(longType.getTypeName(), longType);
		fBeanProxies.put(floatType.getTypeName(), floatType);
		fBeanProxies.put(doubleType.getTypeName(), doubleType);

		// java.lang primitive peers
		// Note that special classes are used for some of these which allow the IDE to get the
		// lang objects from the objects that are holding proxies
		fBeanProxies.put(integerClass.getTypeName(), integerClass);
		fBeanProxies.put(booleanClass.getTypeName(), booleanClass);
		fBeanProxies.put(characterClass.getTypeName(), characterClass);
		fBeanProxies.put(byteClass.getTypeName(), byteClass);
		fBeanProxies.put(shortClass.getTypeName(), shortClass);
		fBeanProxies.put(longClass.getTypeName(), longClass);
		fBeanProxies.put(floatClass.getTypeName(), floatClass);
		fBeanProxies.put(doubleClass.getTypeName(), doubleClass);
		fBeanProxies.put(bigDecimalClass.getTypeName(), bigDecimalClass);
		fBeanProxies.put(bigIntegerClass.getTypeName(), bigIntegerClass);
		fBeanProxies.put(stringClass.getTypeName(), stringClass);
		fBeanProxies.put(throwableClass.getTypeName(), throwableClass);
		fBeanProxies.put(objectClass.getTypeName(), objectClass);
		fBeanProxies.put(classClass.getTypeName(), classClass);
		fBeanProxies.put(numberClass.getTypeName(), numberClass);
		fBeanProxies.put(threadClass.getTypeName(), threadClass);
		fBeanProxies.put(expressionProcessorControllerClass.getTypeName(), expressionProcessorControllerClass);

		// Mark these as permanent.
		fPermanentProxies.addAll(fBeanProxies.values());
	}

	/**
	 * Initialize AFTER BeanProxyFactory has been created. This is REQUIRED! NOTE: It is package protected so that only REMStandardBeanProxyFactory
	 * can call it when ready.
	 */
	synchronized void initialize(REMStandardBeanProxyFactory proxyFactory) {
		fBeanFactory = proxyFactory;
		fBeanFactory.registerProxies(fBeanProxies.values());
	}

	/*
	 * This is called when we know we don't have the class registered, so we need to create the proxy. We have a connection passed in and will reuse
	 * it as necessary
	 * 
	 * It is important that this be called only from within a transaction.
	 */
	private synchronized IREMBeanTypeProxy createBeanTypeProxy(String typeName, IREMConnection connection) throws CommandException {

		// We don't have the beantype proxy, so create it.
		IREMBeanTypeProxy beanTypeProxy = null;
		Commands.GetClassReturn ret = null;
		try {
			ret = getClassReturn(connection, typeName);
		} catch (ThrowableProxy ep) {
			// Print out the trace and return proxy with init error.
			ProxyPlugin.getPlugin().getLogger().log(new Status(IStatus.WARNING, ProxyPlugin.getPlugin().getBundle().getSymbolicName(), 0, "", ep)); //$NON-NLS-1$
			IREMBeanTypeProxy proxy = new REMInitErrorBeanTypeProxy(fFactoryRegistry, MessageFormat.format(ProxyRemoteMessages.ExceptionErrorMsg_EXC_, new Object[] { ep.getTypeProxy().getTypeName(), ep.getProxyLocalizedMessage()}), typeName); //$NON-NLS-1$
			registerBeanTypeProxy(proxy, false);
			return proxy;
		}
		if (ret == null) {
			if (fNotFoundTypes != null)
				fNotFoundTypes.add(typeName);
			return null; // The class doesn't even exist on the server.
		}

		if (typeName.charAt(0) != '[') {
			// It is not an array.
			IREMBeanTypeProxy superTypeProxy = null;
			if (!ret.isInterface && ret.superClassname.length() != 0) {
				// Get the beantype proxy of the superclass.
				superTypeProxy = getBeanTypeProxy(ret.superClassname, connection);
			}

			// First check with the factory for the package of the class.
			// Inner classes have to use the dollar notation since if they didn't we couldn't tell where
			// the package ended and the class started. This check is here in case the
			// extension factory can handle this class but needs the id from the server to
			// create it.
			int packageIndex = typeName.lastIndexOf('.');
			if (packageIndex != -1) {
				String packageName = typeName.substring(0, packageIndex);
				IREMBeanTypeProxyFactory packageFactory = (IREMBeanTypeProxyFactory) fFactoryRegistry.getBeanTypeProxyFactoryExtension(packageName);
				if (packageFactory != null) {
					beanTypeProxy = packageFactory.getExtensionBeanTypeProxy(typeName, new Integer(ret.classID), superTypeProxy);
					if (beanTypeProxy != null) {
						registerBeanTypeProxy(beanTypeProxy, false);
						return beanTypeProxy;
					}
				}
			}

			if (ret.isInterface) {
				// Interface never have a super type, so we will create a specific type.
				beanTypeProxy = new REMInterfaceBeanTypeProxy(fFactoryRegistry, new Integer(ret.classID), typeName);
			} else {
				// Ask the supertype
				// to create a beantype proxy of the same beantype proxy class.
				// This is so that any subclasses will get the same beantype proxy class
				// for it if it is special.
				if (superTypeProxy != null)
					beanTypeProxy = superTypeProxy.newBeanTypeForClass(new Integer(ret.classID), typeName, ret.isAbstract);
			}
		} else
			beanTypeProxy = new REMArrayBeanTypeProxy(fFactoryRegistry, new Integer(ret.classID), typeName, objectClass);

		// Cache the instance so we can re-use it again
		if (beanTypeProxy != null)
			registerBeanTypeProxy(beanTypeProxy, false);
		return beanTypeProxy;
	}

	/**
	 * Using the helper class to find a class by name, then create the proxy.
	 */
	public IBeanTypeProxy getBeanTypeProxy(String typeName) {
		try {
			return getBeanTypeProxy(MapTypes.getJNIFormatName(typeName), (IREMConnection) null);
		} catch (CommandException e) {
			// Try once more (we won't have received recoverable exceptions here, they were already caught and handled)
			try {
				return getBeanTypeProxy(typeName, (IREMConnection) null);
			} catch (CommandException eAgain) {
				// Failed again.
				ProxyPlugin.getPlugin().getLogger().log(
						new Status(IStatus.WARNING, ProxyPlugin.getPlugin().getBundle().getSymbolicName(), 0, "", eAgain)); //$NON-NLS-1$
			}
		}
		return null;
	}

	/*
	 * (non-Javadoc)
	 * 
	 * @see org.eclipse.jem.internal.proxy.core.IStandardBeanTypeProxyFactory#getBeanTypeProxy(org.eclipse.jem.internal.proxy.core.IExpression,
	 *      java.lang.String)
	 */
	public synchronized IProxyBeanType getBeanTypeProxy(IExpression expression, String typeName) {
		typeName = MapTypes.getJNIFormatName(typeName);
		// See whether we already have the proxy for the argument name
		IProxyBeanType beanTypeProxy = (IProxyBeanType) fBeanProxies.get(typeName);
		if (beanTypeProxy != null)
			return beanTypeProxy;
		
		// Now see if an expression proxy cached.
		beanTypeProxy = ((REMExpression) expression).getBeanType(typeName);
		if (beanTypeProxy != null)
			return beanTypeProxy;		

		// If not an array, then see if the package extension mechanism can find it.
		// Do this here so that if it is found in the package extension we won't necessarily create an
		// extra connection when not needed.
		if (typeName.charAt(0) != '[') {
			// It is not an array
			// First check with the factory for the package of the class.
			// Inner classes have to use the dollar notation since if they didn't we couldn't tell where
			// the package ended and the class started.
			int packageIndex = typeName.lastIndexOf('.');
			if (packageIndex != -1) {
				String packageName = typeName.substring(0, packageIndex);
				IREMBeanTypeProxyFactory packageFactory = (IREMBeanTypeProxyFactory) fFactoryRegistry.getBeanTypeProxyFactoryExtension(packageName);
				if (packageFactory != null) {
					beanTypeProxy = packageFactory.getExtensionBeanTypeProxy(typeName, expression);
					if (beanTypeProxy != null) {
						registerBeanTypeProxy(beanTypeProxy, false);
						return beanTypeProxy;
					}
				}
			}
		}

		// Need to create the expression proxy for it.
		beanTypeProxy = ((Expression) expression).createBeanTypeExpressionProxy(typeName);
		registerBeanTypeProxy(beanTypeProxy, false);
		return beanTypeProxy;

	}

	/**
	 * One that internally allows that we already have a connection to work with. If the connection is null, then one will be created.
	 * 
	 * It is important that if the connection is not null, then we are in a transaction.
	 */
	private synchronized IREMBeanTypeProxy getBeanTypeProxy(String typeName, IREMConnection inConnect) throws CommandException {

		// See whether we already have the proxy for the argument name
		IProxyBeanType beanTypeProxy = (IProxyBeanType) fBeanProxies.get(typeName);
		// See if there and resolved, if so, return it. If not resolved, that means we need it NOW
		// so we must go for it. When finally resolved the original ExpressionProxy will be deregistered and
		// the resolved beantypeproxy will be in its place.
		if (beanTypeProxy != null && beanTypeProxy.isBeanProxy()) { return (IREMBeanTypeProxy) beanTypeProxy; }

		// If not an array, then see if the package extension mechanism can find it.
		// Do this here so that if it is found in the package extension we won't necessarily create an
		// extra connection when not needed.
		if (typeName.charAt(0) != '[') {
			// It is not an array
			// First check with the factory for the package of the class.
			// Inner classes have to use the dollar notation since if they didn't we couldn't tell where
			// the package ended and the class started.
			int packageIndex = typeName.lastIndexOf('.');
			if (packageIndex != -1) {
				String packageName = typeName.substring(0, packageIndex);
				IREMBeanTypeProxyFactory packageFactory = (IREMBeanTypeProxyFactory) fFactoryRegistry.getBeanTypeProxyFactoryExtension(packageName);
				if (packageFactory != null) {
					beanTypeProxy = packageFactory.getExtensionBeanTypeProxy(typeName);
					if (beanTypeProxy != null) {
						registerBeanTypeProxy((IREMBeanTypeProxy) beanTypeProxy, false);
						return (IREMBeanTypeProxy) beanTypeProxy;
					}
				}
			}
		}

		IREMConnection connect = inConnect != null ? inConnect : fFactoryRegistry.getFreeConnection();
		if (inConnect == null)
			fBeanFactory.startTransaction(); // Start a transation.
		try {
			return createBeanTypeProxy(typeName, connect);
		} catch (CommandException e) {
			if (inConnect == null) {
				// Need to close the connection, not return it.
				fFactoryRegistry.closeConnection(connect);
				connect = null; // So that it won't be returned.
			}
			throw e; // Pass it on up
		} finally {
			if (inConnect == null)
				fBeanFactory.stopTransaction();
			if (inConnect == null && connect != null)
				fFactoryRegistry.returnConnection(connect);
		}
	}

	/*
	 * It is important this be called only from within a transaction.
	 */
	private Commands.GetClassReturn getClassReturn(IREMConnection connection, String className) throws CommandException, ThrowableProxy {
		try {
			return connection.getClass(className);
		} catch (CommandErrorException e) {
			fBeanFactory.processErrorReturn(e); // Let proxy factory handle the error return
		}
		return null;
	}

	/**
	 * Return an Array type proxy for the given class name of the specified dimensions. This is a helper method. The same result can be gotton from
	 * getBeanTypeProxy. e.g. getBeanTypeProxy("java.lang.Object", 3) is the same as: getBeanTypeProxy("[[[Ljava.lang.Object;")
	 * 
	 * They both result in a type of: Object [][][]
	 * 
	 * or getBeanTypeProxy("[Ljava.langObject;", 3) becomes Object [][][][]
	 */
	public IBeanTypeProxy getBeanTypeProxy(String componentClassName, int dimensions) {
		return getBeanTypeProxy(getArrayClassName(componentClassName, dimensions));
	}

	/**
	 * @param componentClassName
	 * @param dimensions
	 * @return
	 * 
	 * @since 1.1.0
	 */
	private String getArrayClassName(String componentClassName, int dimensions) {
		String jniComponentClassName = MapTypes.getJNIFormatName(componentClassName);
		String compType = jniComponentClassName;
		if (jniComponentClassName.charAt(0) != '[') {
			// We're not already an array, so create correct template.
			compType = (String) MapTypes.MAP_TYPENAME_TO_SHORTSIG.get(jniComponentClassName);
			if (compType == null) {
				// It is a class, and not a type.
				compType = "L" + jniComponentClassName + ";"; //$NON-NLS-1$ //$NON-NLS-2$
			}
		}

		// Now create it with the appropriate number of '[' in front.
		StringBuffer buffer = new StringBuffer(dimensions + compType.length());
		for (int i = 0; i < dimensions; i++)
			buffer.append('[');
		buffer.append(compType);
		return buffer.toString();
	}

	/*
	 * (non-Javadoc)
	 * 
	 * @see org.eclipse.jem.internal.proxy.core.IStandardBeanTypeProxyFactory#getBeanTypeProxy(org.eclipse.jem.internal.proxy.core.IExpression,
	 *      java.lang.String, int)
	 */
	public IProxyBeanType getBeanTypeProxy(IExpression expression, String componentClassName, int dimensions) {
		return getBeanTypeProxy(expression, getArrayClassName(componentClassName, dimensions));
	}

	/**
	 * Get the bean type proxy from a class id. This means that a new class id was sent back from the server that we don't have yet. We need to go ask
	 * the server for information on this type so that we can create it.
	 * 
	 * NOTE: This is package protected so that only the standard bean proxy factory can call it.
	 * 
	 * It is important that this has been called within a transaction.
	 */
	IREMBeanTypeProxy createBeanTypeProxy(Integer classID) {
		IREMConnection connect = fFactoryRegistry.getFreeConnection();
		try {
			return createBeanTypeProxy(classID, connect);
		} catch (CommandException e) {
			if (e.isRecoverable()) {
				// It is recoverable, print message, keep connection live and return it.
				ProxyPlugin.getPlugin().getLogger().log(new Status(IStatus.WARNING, ProxyPlugin.getPlugin().getBundle().getSymbolicName(), 0, "", e)); //$NON-NLS-1$
			} else {
				// Try again, close connection, get a new one.
				fFactoryRegistry.closeConnection(connect);
				connect = null;
				connect = fFactoryRegistry.getFreeConnection();
				try {
					return createBeanTypeProxy(classID, connect);
				} catch (CommandException eAgain) {
					// Failed again. Close connection, don't return it.
					ProxyPlugin.getPlugin().getLogger().log(
							new Status(IStatus.WARNING, ProxyPlugin.getPlugin().getBundle().getSymbolicName(), 0, "", eAgain)); //$NON-NLS-1$
					fFactoryRegistry.closeConnection(connect);
					connect = null;
				}
			}
		} finally {
			if (connect != null)
				fFactoryRegistry.returnConnection(connect);
		}
		return null;
	}

	/*
	 * One that internally allows that we already have a connection to work with. If the connection is null, then one will be created.
	 * 
	 * It is important this be called only from within a transaction.
	 */
	private synchronized IREMBeanTypeProxy createBeanTypeProxy(Integer classID, IREMConnection connect) throws CommandException {
		// We don't have the beantype proxy, so create it.
		IREMBeanTypeProxy beanTypeProxy = null;

		Commands.GetClassIDReturn ret = null;
		try {
			ret = getClassIDReturn(connect, classID);
		} catch (ThrowableProxy ep) {
			// Just print out the trace and return proxy not found.
			ProxyPlugin.getPlugin().getLogger().log(new Status(IStatus.WARNING, ProxyPlugin.getPlugin().getBundle().getSymbolicName(), 0, "", ep)); //$NON-NLS-1$
			return null;
		}

		// If the signature is that of a class.
		if (ret.className.charAt(0) != '[') {
			// It is not an array.
			IREMBeanTypeProxy superTypeProxy = null;
			if (!ret.isInterface && ret.superClassname.length() != 0) {
				// Get the beantype proxy of the superclass.
				superTypeProxy = getBeanTypeProxy(ret.superClassname, connect);
			}

			// First check with the factory for the package of the class.
			// Inner classes have to use the dollar notation since if they didn't we couldn't tell where
			// the package ended and the class started.
			int packageIndex = ret.className.lastIndexOf('.');
			if (packageIndex != -1) {
				String packageName = ret.className.substring(0, packageIndex);
				IREMBeanTypeProxyFactory packageFactory = (IREMBeanTypeProxyFactory) fFactoryRegistry.getBeanTypeProxyFactoryExtension(packageName);
				if (packageFactory != null) {
					beanTypeProxy = packageFactory.getExtensionBeanTypeProxy(ret.className, classID, superTypeProxy);
					if (beanTypeProxy != null) {
						registerBeanTypeProxy(beanTypeProxy, false);
						return beanTypeProxy;
					}
				}
			}

			if (ret.isInterface) {
				// Interface never have a super type, so we will let the object class do it for us.
				beanTypeProxy = new REMInterfaceBeanTypeProxy(fFactoryRegistry, classID, ret.className);
			} else {
				// Ask the beantype proxy of the superclass
				// to create a beantype proxy of the same beantype proxy class.
				// This is so that any subclasses will get the same beantype proxy class
				// for it if it is special.
				if (superTypeProxy != null)
					beanTypeProxy = superTypeProxy.newBeanTypeForClass(classID, ret.className, ret.isAbstract);
			}

			// Cache the instance so we can re-use it again
			if (beanTypeProxy != null)
				registerBeanTypeProxy(beanTypeProxy, false);
			return beanTypeProxy;
		} else {
			// It is an array.
			beanTypeProxy = new REMArrayBeanTypeProxy(fFactoryRegistry, classID, ret.className, objectClass);
			registerBeanTypeProxy(beanTypeProxy, false);
			return beanTypeProxy;
		}
	}

	/*
	 * It is important that this be called only from within a transaction.
	 */
	private Commands.GetClassIDReturn getClassIDReturn(IREMConnection connection, Integer classID) throws CommandException, ThrowableProxy {
		try {
			return connection.getClassFromID(classID.intValue());
		} catch (CommandErrorException e) {
			fBeanFactory.processErrorReturn(e); // Let proxy factory handle the error return
		}
		return null;
	}

	/*
	 * (non-Javadoc)
	 * 
	 * @see org.eclipse.jem.internal.proxy.core.IStandardBeanTypeProxyFactory#registerBeanTypeProxy(org.eclipse.jem.internal.proxy.core.IBeanTypeProxy,
	 *      boolean)
	 */
	public synchronized void registerBeanTypeProxy(IBeanTypeProxy aBeanTypeProxy, boolean permanent) {
		fBeanProxies.put(aBeanTypeProxy.getTypeName(), aBeanTypeProxy);
		fBeanFactory.registerProxy((IREMBeanProxy) aBeanTypeProxy);
		if (permanent)
			fPermanentProxies.add(aBeanTypeProxy);
	}

	/*
	 * (non-Javadoc)
	 * 
	 * @see org.eclipse.jem.internal.proxy.core.IStandardBeanTypeProxyFactory#registerBeanTypeProxy(org.eclipse.jem.internal.proxy.core.IProxyBeanType,
	 *      boolean)
	 */
	public void registerBeanTypeProxy(IProxyBeanType aProxyBeanType, final boolean permanent) {
		if (aProxyBeanType.isBeanProxy())
			registerBeanTypeProxy((IBeanTypeProxy) aProxyBeanType, permanent); // A regular kind, do regular registration.
		else {
			ExpressionProxy beanExpressionProxy = ((ExpressionProxy) aProxyBeanType);
			final String typeName = aProxyBeanType.getTypeName();
			((REMExpression) beanExpressionProxy.getExpression()).addBeanType(typeName, aProxyBeanType);
			beanExpressionProxy.addProxyListener(new ExpressionProxy.ProxyAdapter() {

				public void proxyResolved(ProxyEvent event) {
					String typeName = ((IProxyBeanType) event.getSource()).getTypeName();
					synchronized (REMStandardBeanTypeProxyFactory.this) {
						if (!fBeanProxies.containsKey(typeName)) {
							// It hasn't been resolved through some other means. So this is good. Actually this should never
							// occur because upon resolution we've already registered the bean type proxy through the
							// normal mechanisms. But to be safe, we'll do it here.
							fBeanProxies.put(typeName, event.getProxy());
							if (permanent)
								fPermanentProxies.add(event.getProxy());
						}
					}
				}
				
				public void proxyNotResolved(ExpressionProxy.ProxyEvent event) {
					((REMExpression) ((ExpressionProxy) event.getSource()).getExpression()).removeBeanType(typeName);
				}
			});
		}

	}

	/**
	 * A beantype proxy is asked to be released. We can only release ones that were not in the permanent set that we initialized with. Those in the
	 * permanent set can't be changed so we can't release them.
	 * 
	 * Answer whether it can be released from the server too.
	 * 
	 * NOTE: Package protected since only REMStandardBeanProxyFactory should call it.
	 */
	boolean releaseProxy(IBeanTypeProxy proxy) {
		/**
		 * Currently we won't allow any bean type proxies to be released. We don't have a good strategy for handling that there may be hard refs from
		 * subtypes. One thought is that beanproxies table should store SoftReferences so that only when space is needed, that any beantype that
		 * doesn't have a subtype (since subtypes hold a strong ref) or, is in the permanent table (since that is hardref) could be GC'd. Then what
		 * would happen is on releaseProxy we don't actually release, we change it to a WeakRef so that it would definitely be released on a GC. These
		 * are complicated arch. and we're not sure if it should be allowed or not. So for now, we don't allow them to be released. if
		 * (!fPermanentProxies.contains(proxy)) { // We can release it. It is not one of the permanent ones. synchronized(this) {
		 * fBeanProxies.remove(proxy.getTypeName()); return true; } }
		 */
		return false;
	}

	/*
	 * Terminate this factory. Since all of the proxies are registered in the proxy factory, there is no need to release them here. There is no need
	 * to clear out any fields since this factory will not be held onto by anything and so it will be GC'd.
	 */
	public void terminateFactory(boolean wait) {
	}

	/*
	 * (non-Javadoc)
	 * 
	 * @see org.eclipse.jem.internal.proxy.core.IStandardBeanTypeProxyFactory#isBeanTypeRegistered(String)
	 */
	public synchronized boolean isBeanTypeRegistered(String className) {
		return fBeanProxies.containsKey(MapTypes.getJNIFormatName(className));
	}

	/*
	 * (non-Javadoc)
	 * 
	 * @see org.eclipse.jem.internal.proxy.core.IStandardBeanTypeProxyFactory#registeredTypes()
	 */
	public Set registeredTypes() {
		return fBeanProxies.keySet();
	}

	/*
	 * (non-Javadoc)
	 * 
	 * @see org.eclipse.jem.internal.proxy.core.IStandardBeanTypeProxyFactory#isBeanTypeNotFound(String)
	 */
	public synchronized boolean isBeanTypeNotFound(String className) {
		return fNotFoundTypes != null && fNotFoundTypes.contains(MapTypes.getJNIFormatName(className));
	}

	/*
	 * (non-Javadoc)
	 * 
	 * @see org.eclipse.jem.internal.proxy.core.IStandardBeanTypeProxyFactory#isMaintainNotFoundTypes()
	 */
	public synchronized boolean isMaintainNotFoundTypes() {
		return fNotFoundTypes != null;
	}

	/*
	 * (non-Javadoc)
	 * 
	 * @see org.eclipse.jem.internal.proxy.core.IStandardBeanTypeProxyFactory#setMaintainNotFoundTypes(boolean)
	 */
	public synchronized void setMaintainNotFoundTypes(boolean maintain) {
		if (maintain) {
			if (fNotFoundTypes == null)
				fNotFoundTypes = new HashSet();
		} else
			fNotFoundTypes = null;
	}

}

