public string ContinueFromManyToMany(string clazz, string[] joinColumns, QueryTranslator q)
		{
			Start(q);
			continuation = true;
			currentName = q.CreateNameFor(clazz);
			q.AddType(currentName, clazz);
			IQueryable classPersister = q.GetPersister(clazz);
			AddJoin(currentName, TypeFactory.ManyToOne(clazz), joinColumns);
			currentPropertyMapping = classPersister;
			return currentName;
		}
		/// <summary>
		/// 
		/// </summary>
		/// <param name="propertyName"></param>
		/// <param name="propertyType"></param>
		/// <param name="q"></param>
		/// <remarks>NOTE: we avoid joining to the next table if the named property is just the foreign key value</remarks>
		private void DereferenceEntity(string propertyName, EntityType propertyType, QueryTranslator q)
		{
			//if its "id"
			bool isIdShortcut = EntityID.Equals(propertyName) && !propertyType.IsUniqueKeyReference;

			//or its the id property name
			string idPropertyName;
			try
			{
				idPropertyName = propertyType.GetIdentifierOrUniqueKeyPropertyName(q.Factory);
			}
			catch (MappingException me)
			{
				throw new QueryException(me);
			}
			bool isNamedIdPropertyShortcut = idPropertyName != null && idPropertyName.Equals(propertyName);

			if (isIdShortcut || isNamedIdPropertyShortcut)
			{
				// special shortcut for id properties, skip the join!
				// this must only occur at the _end_ of a path expression
				DereferenceProperty(propertyName);
			}
			else
			{
				string entityClass = propertyType.GetAssociatedEntityName();
				string name = q.CreateNameFor(entityClass);
				q.AddType(name, entityClass);
				//String[] keyColNames = memberPersister.getIdentifierColumnNames();
				AddJoin(name, propertyType);
                if (propertyType.IsOneToOne)
                {
                    oneToOneOwnerName = currentName;
                }
                else
                {
                    oneToOneOwnerName = null;
                }
				ownerAssociationType = propertyType;
				currentName = name;
				currentProperty = propertyName;
				q.AddPathAliasAndJoin(path.ToString(0, path.ToString().LastIndexOf(StringHelper.Dot)), name, joinSequence.Copy());
				componentPath.Length = 0;
				currentPropertyMapping = q.GetPersister(entityClass);
			}
		}
		/// <summary>
		/// 
		/// </summary>
		/// <param name="q"></param>
		/// <returns></returns>
		public string AddFromCollection(QueryTranslator q)
		{
			IType collectionElementType = PropertyType;

			if (collectionElementType == null)
			{
				throw new QueryException(
					string.Format("must specify 'elements' for collection valued property in from clause: {0}", path));
			}
			if (collectionElementType.IsEntityType)
			{
				// an association
				IQueryableCollection collectionPersister = q.GetCollectionPersister(collectionRole);
				IQueryable entityPersister = (IQueryable) collectionPersister.ElementPersister;
				string clazz = entityPersister.EntityName;

				string elementName;
				if (collectionPersister.IsOneToMany)
				{
					elementName = collectionName;
					// allow index() function
					q.DecoratePropertyMapping(elementName, collectionPersister);
				}
				else
				{
					// many to many
					q.AddCollection(collectionName, collectionRole);
					elementName = q.CreateNameFor(clazz);
					AddJoin(elementName, (IAssociationType) collectionElementType);
				}
				q.AddFrom(elementName, clazz, joinSequence);
				currentPropertyMapping = new CollectionPropertyMapping(collectionPersister);
				return elementName;
			}
			
			// collection of values
			q.AddFromCollection(collectionName, collectionRole, joinSequence);
			return collectionName;
		}
		public void Token(string token, QueryTranslator q)
		{
			// start by looking for HQL keywords....
			string lcToken = token.ToLowerInvariant();
			if (lcToken.Equals(StringHelper.Comma))
			{
				if (!(expectingJoin | expectingAs))
					throw new QueryException("unexpected token: ,");

				expectingJoin = false;
				expectingAs = false;
			}
			else if ("join".Equals(lcToken))
			{
				if (!afterJoinType)
				{
					if (!(expectingJoin | expectingAs))
						throw new QueryException("unexpected token: join");

					// inner joins can be abbreviated to 'join'
					joinType = JoinType.InnerJoin;
					expectingJoin = false;
					expectingAs = false;
				}
				else
				{
					afterJoinType = false;
				}
			}
			else if ("fetch".Equals(lcToken))
			{
				if (q.IsShallowQuery)
					throw new QueryException("fetch may not be used with scroll() or iterate()");

				if (joinType == JoinType.None)
					throw new QueryException("unexpected token: fetch");

				if (joinType == JoinType.FullJoin || joinType == JoinType.RightOuterJoin)
					throw new QueryException("fetch may only be used with inner join or left outer join");

				afterFetch = true;
			}
			else if ("outer".Equals(lcToken))
			{
				// 'outer' is optional and is ignored)
				if (!afterJoinType || (joinType != JoinType.LeftOuterJoin && joinType != JoinType.RightOuterJoin))
					throw new QueryException("unexpected token: outer");
			}
			else if (joinTypes.ContainsKey(lcToken))
			{
				if (!(expectingJoin | expectingAs))
					throw new QueryException("unexpected token: " + token);

				joinType = joinTypes[lcToken];
				afterJoinType = true;
				expectingJoin = false;
				expectingAs = false;
			}
			else if ("class".Equals(lcToken))
			{
				if (!afterIn)
					throw new QueryException("unexpected token: class");

				if (joinType != JoinType.None)
					throw new QueryException("outer or full join must be followed by path expression");

				afterClass = true;
			}
			else if ("in".Equals(lcToken))
			{
				if (!expectingIn)
					throw new QueryException("unexpected token: in");

				afterIn = true;
				expectingIn = false;
			}
			else if ("as".Equals(lcToken))
			{
				if (!expectingAs)
					throw new QueryException("unexpected token: as");

				afterAs = true;
				expectingAs = false;
			}
			else
			{
				if (afterJoinType)
					throw new QueryException("join expected: " + token);

				if (expectingJoin)
					throw new QueryException("unexpected token: " + token);

				if (expectingIn)
					throw new QueryException("in expected: " + token);

				// now anything that is not a HQL keyword

				if (afterAs || expectingAs)
				{
					// (AS is always optional, for consistency with SQL/OQL

					// process the "new" HQL stype where aliases are assigned
					// _after_ the class name or path expression ie using the
					// AS construction

					if (entityName != null)
					{
						q.SetAliasName(token, entityName);
					}
					else
						throw new QueryException("unexpected: as " + token);

					afterAs = false;
					expectingJoin = true;
					expectingAs = false;
					entityName = null;
				}
				else if (afterIn)
				{
					// process the "old" HQL style where aliases appear _first
					// ie using the IN or IN CLASS constructions

					if (alias == null)
						throw new QueryException("alias not specified for: " + token);

					if (joinType != JoinType.None)
						throw new QueryException("outer or full join must be followed by path expressions");

					if (afterClass)
					{
						// treat it as a classname
						IQueryable p = q.GetPersisterUsingImports(token);
						if (p == null)
						{
							throw new QueryException("persister not found: " + token);
						}
						q.AddFromClass(alias, p);
					}
					else
					{
						// treat it as a path expression
						peParser.JoinType = JoinType.InnerJoin;
						peParser.UseThetaStyleJoin = true;
						ParserHelper.Parse(peParser, q.Unalias(token), ParserHelper.PathSeparators, q);
						if (!peParser.IsCollectionValued)
						{
							throw new QueryException("pathe expression did not resolve to collection: " + token);
						}
						string nm = peParser.AddFromCollection(q);
						q.SetAliasName(alias, nm);
					}

					alias = null;
					afterIn = false;
					afterClass = false;
					expectingJoin = true;
				}
				else
				{
					// handle a path expression or class name that appears
					// at the start, in the "new" HQL style or an alias that
					// appears at the start in the "old HQL stype
					IQueryable p = q.GetPersisterUsingImports(token);
					if (p != null)
					{
						// starts with the name of a mapped class (new style)
						if (joinType != JoinType.None)
							throw new QueryException("outer or full join must be followed by path expression");

						entityName = q.CreateNameFor(p.EntityName);
						q.AddFromClass(entityName, p);
						expectingAs = true;
					}
					else if (token.IndexOf('.') < 0)
					{
						// starts with an alias (old style)
						// semi-bad thing about this: can't re-alias another alias...
						alias = token;
						expectingIn = true;
					}
					else
					{
						// starts with a path expression (new style)

						// force HQL style: from Person p inner join p.cars c
						//if (joinType==JoinType.None) throw new QueryException("path expression must be preceded by full, left, right or inner join");

						//allow ODMG OQL style: from Person p, p.cars c
						if (joinType != JoinType.None)
							peParser.JoinType = joinType;
						else
							peParser.JoinType = JoinType.InnerJoin;

						peParser.UseThetaStyleJoin = q.IsSubquery;

						ParserHelper.Parse(peParser, q.Unalias(token), ParserHelper.PathSeparators, q);
						entityName = peParser.AddFromAssociation(q);

						joinType = JoinType.None;
						peParser.JoinType = JoinType.InnerJoin;

						if (afterFetch)
						{
							peParser.Fetch(q, entityName);
							afterFetch = false;
						}

						expectingAs = true;
					}
				}
			}
		}