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

msv.tahiti.src.com.sun.tahiti.reader.relaxng.TRELAXNGReader Maven / Gradle / Ivy

There is a newer version: 2.2.5.1
Show newest version
/*
 * @(#)$Id: TRELAXNGReader.java 1231 2001-10-16 23:49:03Z Bear $
 *
 * Copyright 2001 Sun Microsystems, Inc. All Rights Reserved.
 * 
 * This software is the proprietary information of Sun Microsystems, Inc.  
 * Use is subject to license terms.
 * 
 */
package com.sun.tahiti.reader.relaxng;

import com.sun.msv.grammar.*;
import com.sun.msv.datatype.DatabindableDatatype;
import com.sun.msv.reader.trex.ng.RELAXNGReader;
import com.sun.msv.reader.ExpressionState;
import com.sun.msv.reader.GrammarReaderController;
import com.sun.msv.util.StartTagInfo;
import com.sun.tahiti.reader.annotator.Annotator;
import com.sun.tahiti.reader.NameUtil;
import com.sun.tahiti.reader.TahitiGrammarReader;
import com.sun.tahiti.grammar.*;
import javax.xml.parsers.SAXParserFactory;
import java.util.Map;
import java.util.Stack;
import java.util.ResourceBundle;
import java.text.MessageFormat;
import org.xml.sax.SAXException;
import org.xml.sax.Attributes;

/**
 * parses Tahiti-annotated RELAX NG grammar.
 * 
 * @author
 *	Kohsuke KAWAGUCHI
 */
public class TRELAXNGReader extends RELAXNGReader implements TahitiGrammarReader {

	public TRELAXNGReader(
		GrammarReaderController controller, SAXParserFactory parserFactory,
		StateFactory stateFactory, ExpressionPool pool ) {
		
		super( controller, parserFactory, stateFactory, pool );
	}

	public TRELAXNGReader(
		GrammarReaderController controller, SAXParserFactory parserFactory ) {
		
		super( controller, parserFactory );
	}
	
	private final AnnotatedGrammar annGrammar = new AnnotatedGrammar( null, pool );
	public AnnotatedGrammar getAnnotatedResult() {
		return annGrammar;
	}

	protected Expression interceptExpression( ExpressionState state, Expression exp ) {
		// if an error was found, stop processing.
		if( hadError )	return exp;
		
		// check Tahiti attributes.
		
		final StartTagInfo tag = state.getStartTag();
		String role = tag.getAttribute(TahitiNamespace,"role");
		
		if( role==null ) {
			// there is no markup.
			
			// insert an ClassItem if this is the  tag.
			// some of those temporarily added ClassItems will be removed
			// in the final wrap up.
			if( tag.localName.equals("element") ) {
//			if( exp instanceof ElementExp ) {
/*
				ElementExp eexp = (ElementExp)exp;
				// add a ClassItem between the ElementExp and the content model.
				ClassItem t = annGrammar.createClassItem( decideName(state,exp,"class"), eexp.contentModel );
				eexp.contentModel = t;
				
				t.isTemporary = true;	// this flag indicates that this class item is a temporary one.
				
				return eexp;
*/
				// to make marshalling easy, ClassItem is generated above the ElementExp
				
				// calculate the unique name to avoid name conflicts.
				String baseName = decideName(state,exp,"class");
				if( annGrammar.classes.containsKey(baseName) ) {
					int cnt = 2;
					while( annGrammar.classes.containsKey(baseName+cnt) )
						cnt++;
					baseName += cnt;
				}
				
				ClassItem t = annGrammar.createClassItem( baseName, exp );
				t.isTemporary = true;
				return t;
			} else {
				// if this element has the t:name attribute, store that
				// information by using a ReferenceExp. This information
				// might be useful to various annotators.
//				String name = tag.getAttribute(TahitiNamespace,"name");
//				if(name!=null)	exp = new ReferenceExp( name, exp );
			}
			
			return exp;	// the "role" attribute is not present.
		}
		
		
		OtherExp roleExp;
		
		if( role.equals("none") ) {
			// do nothing. this will prevent automatic ClassItem insertion.
			return exp;
		} else
		if( role.equals("superClass") ) {
			roleExp = new SuperClassItem();
		} else
		if( role.equals("class") ) {
			/*
			removes silly use of temporary ClassItem.
			Consider the following grammar fragment:
			
				
					...
				
			
			
			Since there is no tahiti markup for  element, a temporary ClassItem
			is inserted. Immediately after that, a  element is processed, and 
			a new ClassItem is inserted.
			
			The following code is a quick hack to prevent this situation by removing
			temporary ClassItem if it is the immediate child of the user-defined class.
			 */
			/*
				We changed the code and now temporary class items are generated
				*above* elements. So now this code is unnecessary.
			*/
/* TODO:
	we have to think in much broader view.
	This could be done as a part of temporary class item removal.
	Otherwise, we have to implement this mechanism to every reader.
	
	Also, this particular situation would interact with the combine attribute.
*/			
/*			if( (exp instanceof ClassItem) && ((ClassItem)exp).isTemporary )
				exp = ((ClassItem)exp).exp;
			if( exp.getClass()==ReferenceExp.class ) {
				ReferenceExp rexp = (ReferenceExp)exp;
				if( (rexp.exp instanceof ClassItem) && ((ClassItem)rexp.exp).isTemporary )
					rexp.exp = ((ClassItem)rexp.exp).exp;
			}
*/			
			roleExp = annGrammar.createClassItem(decideName(state,exp,role),null);
		} else
		if( role.equals("field") ) {
			FieldItem fi = new FieldItem(decideName(state,exp,role));
			roleExp = fi;
			
			// read the additional configuration.
			String collection = tag.getAttribute(TahitiNamespace,"collection");
			if(collection!=null) {
				fi.collectionType = fi.collectionType.parse(collection);
				if(fi.collectionType==null)
					reportError( ERR_INVALID_COLLECTION_TYPE, collection );
			}
			String access = tag.getAttribute(TahitiNamespace,"access");
			if(access!=null) {
				fi.accessModifier = fi.accessModifier.parse(access);
				if(fi.accessModifier==null)
					reportError( ERR_INVALID_ACCESS_MODIFIER, access );
			}
			String method = tag.getAttribute(TahitiNamespace,"method");
			if(method!=null) {
				fi.accessor = fi.accessor.parse(method);
				if(fi.accessModifier==null)
					reportError( ERR_INVALID_ACCESSOR, method );
			}
		} else
		if( role.equals("interface") ) {
			roleExp = annGrammar.createInterfaceItem(decideName(state,exp,role),null);
		} else
		if( role.equals("ignore") ) {
			roleExp = new IgnoreItem();
		} else {
			reportError( ERR_UNDEFINED_ROLE, role );
			return exp;
		}
		
		setDeclaredLocationOf(roleExp);	// memorize where this expression is defined.
		
		
		if( tag.localName.equals("define") ) {
			// if this is , then
			// this tag should be placed as the sole child of .
			ReferenceExp rexp = (ReferenceExp)exp;
			roleExp.exp = rexp.exp;
			rexp.exp = roleExp;
		} else {
			// wrap the expression by this tag.
			roleExp.exp = exp;
			exp = roleExp;
		}
		
		return exp;
	}

	/**
	 * compute the name for the item.
	 * 
	 * @param role
	 *		the role of this expression. One of "field","interface", and "class".
	 */
	protected String decideName( ExpressionState state, Expression exp, String role ) {
		
		final StartTagInfo tag = state.getStartTag();
	
		String name = tag.getAttribute(TahitiNamespace,"name");
		// if we have t:name attribute, use it.
		if(name==null) {
		
			// if the current tag has the name attribute, use it.
			// this is the case of ,, and sometimes
			//  and 
			name = tag.getAttribute("name");
			if(name!=null)	name = NameUtil.xmlNameToJavaName(role,name);
		}
		
		if(name==null) {
			// otherwise, sniff the name.
		
			// if it's element/attribute, then we may be able to use its name.
			if( exp instanceof NameClassAndExpression ) {
				NameClass nc = ((NameClassAndExpression)exp).getNameClass();
				if( nc instanceof SimpleNameClass )
					name = NameUtil.xmlNameToJavaName(role,((SimpleNameClass)nc).localName);
					
				// if it's not a simple type, abort.
			}
		}
		
		if(name==null) {
			// we can't generate a proper name. bail out
			reportError( ERR_NAME_NEEDED );
			return "";
		}
		
		if( role.equals("class") || role.equals("interface") ) {
			// append the default package name, if necessary.
			int idx = name.indexOf('.');
			if(idx<0 && defaultPackageName.length()!=0 )
				name = defaultPackageName+"."+name;
		}
		
		return name;
	}
	
	public void wrapUp() {
		// First, let the super class do its job.
		super.wrapUp();
		
		// if we already have an error, abort further processing.
		if(hadError)	return;

		// if no package name is specified, place it to the root pacakge.
		if(annGrammar.grammarName==null)
			annGrammar.grammarName = "Grammar";
		
		// add missing annotations and normalizes them.
		annGrammar.topLevel = grammar.exp;
		Annotator.annotate( annGrammar, this );
		grammar.exp = annGrammar.topLevel;
	}

	/**
	 * propagatable 't:package' attribute that specifies the default package 
	 * for unqualified java classes/interfaces.
	 */
	private final Stack packageNameStack = new Stack();
	private String defaultPackageName="";
	
	public void startElement( String a, String b, String c, Attributes d ) throws SAXException {
		// handle "t:package" attribute here.
		packageNameStack.push(defaultPackageName);
		if( d.getIndex(TahitiNamespace,"package")!=-1 ) {
			defaultPackageName = d.getValue(TahitiNamespace,"package");
			
			// if this is the first time the package name is specified,
			// then use it for the grammar's name.
			if(annGrammar.grammarName==null)
				annGrammar.grammarName = defaultPackageName+".Grammar";
		}
		
		super.startElement(a,b,c,d);
	}
	public void endElement( String a, String b, String c ) throws SAXException {
		super.endElement(a,b,c);
		defaultPackageName = (String)packageNameStack.pop();
	}
	
	
	

	protected String localizeMessage( String propertyName, Object[] args ) {
		String format;
		try {
			format = ResourceBundle.getBundle(
				"com.sun.tahiti.reader.relaxng.Messages").getString(propertyName);
		} catch( Exception e ) {
			try {
				format = ResourceBundle.getBundle(
					"com.sun.tahiti.reader.Messages").getString(propertyName);
			} catch( Exception ee ) {
				return super.localizeMessage(propertyName,args);
			}
		}
	    return MessageFormat.format(format, args );
	}
	
// TODO: add more arguments to produce user-friendly messages.
	public static final String ERR_UNDEFINED_ROLE = // arg:1
		"UndefinedRole"; // "{0}" is a bad value for the role attribute.
	public static final String ERR_NAME_NEEDED = // arg:0
		"NameNeeded";	// failed to generate a proper name for this role.
				// specify t:name attribute.
	
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy