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

com.sun.msv.writer.relaxng.PatternWriter Maven / Gradle / Ivy

There is a newer version: 2.3.0
Show newest version
/*
 * Copyright (c) 2001-2013 Oracle and/or its affiliates. All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 *
 *   - Redistributions of source code must retain the above copyright
 *     notice, this list of conditions and the following disclaimer.
 *
 *   - Redistributions in binary form must reproduce the above copyright
 *     notice, this list of conditions and the following disclaimer in the
 *     documentation and/or other materials provided with the distribution.
 *
 *   - Neither the name of Oracle nor the names of its
 *     contributors may be used to endorse or promote products derived
 *     from this software without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS
 * IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
 * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
 * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE COPYRIGHT OWNER OR
 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
 * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
 * LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
 * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
 * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 */

package com.sun.msv.writer.relaxng;

import java.util.HashSet;
import java.util.Set;
import java.util.Vector;

import org.relaxng.datatype.Datatype;
import org.relaxng.datatype.ValidationContext;

import com.sun.msv.datatype.SerializationContext;
import com.sun.msv.datatype.xsd.ConcreteType;
import com.sun.msv.datatype.xsd.DataTypeWithFacet;
import com.sun.msv.datatype.xsd.EnumerationFacet;
import com.sun.msv.datatype.xsd.FinalComponent;
import com.sun.msv.datatype.xsd.FractionDigitsFacet;
import com.sun.msv.datatype.xsd.LengthFacet;
import com.sun.msv.datatype.xsd.ListType;
import com.sun.msv.datatype.xsd.MaxLengthFacet;
import com.sun.msv.datatype.xsd.MinLengthFacet;
import com.sun.msv.datatype.xsd.PatternFacet;
import com.sun.msv.datatype.xsd.RangeFacet;
import com.sun.msv.datatype.xsd.TokenType;
import com.sun.msv.datatype.xsd.TotalDigitsFacet;
import com.sun.msv.datatype.xsd.UnionType;
import com.sun.msv.datatype.xsd.WhiteSpaceFacet;
import com.sun.msv.datatype.xsd.XSDatatype;
import com.sun.msv.datatype.xsd.XSDatatypeImpl;
import com.sun.msv.grammar.AttributeExp;
import com.sun.msv.grammar.BinaryExp;
import com.sun.msv.grammar.ChoiceExp;
import com.sun.msv.grammar.ConcurExp;
import com.sun.msv.grammar.DataExp;
import com.sun.msv.grammar.ElementExp;
import com.sun.msv.grammar.Expression;
import com.sun.msv.grammar.ExpressionVisitorVoid;
import com.sun.msv.grammar.InterleaveExp;
import com.sun.msv.grammar.ListExp;
import com.sun.msv.grammar.MixedExp;
import com.sun.msv.grammar.OneOrMoreExp;
import com.sun.msv.grammar.OtherExp;
import com.sun.msv.grammar.ReferenceExp;
import com.sun.msv.grammar.SequenceExp;
import com.sun.msv.grammar.ValueExp;
import com.sun.msv.writer.XMLWriter;

/**
 * Visits Expression and writes it as RELAX NG.
 */
public abstract class PatternWriter implements ExpressionVisitorVoid {
    public PatternWriter(Context ctxt) {
        this.writer = ctxt.getWriter();
        this.context = ctxt;
    }
    
    protected final XMLWriter writer;
    protected final Context context;
    
    public abstract void onOther(OtherExp exp);
    public abstract void onRef(ReferenceExp exp);
    
    public void onElement(ElementExp exp) {
        writer.start("element");
        context.writeNameClass(exp.getNameClass());
        visitUnary(exp.contentModel);
        writer.end("element");
    }
    
    public void onEpsilon() {
        writer.element("empty");
    }
    
    public void onNullSet() {
        writer.element("notAllowed");
    }
    
    public void onAnyString() {
        writer.element("text");
    }
    
    public void onInterleave(InterleaveExp exp) {
        visitBinExp("interleave", exp, InterleaveExp.class);
    }
    
    public void onConcur(ConcurExp exp) {
        throw new IllegalArgumentException("the grammar includes concur, which is not supported");
    }
    
    public void onList(ListExp exp) {
        writer.start("list");
        visitUnary(exp.exp);
        writer.end("list");
    }
    
    protected void onOptional(Expression exp) {
        if (exp instanceof OneOrMoreExp) {
            // (X+)? == X*
            onZeroOrMore((OneOrMoreExp)exp);
            return;
        }
        writer.start("optional");
        visitUnary(exp);
        writer.end("optional");
    }
    
    public void onChoice(ChoiceExp exp) {
        // use optional instead of  p  
        if (exp.exp1 == Expression.epsilon) {
            onOptional(exp.exp2);
            return;
        }
        if (exp.exp2 == Expression.epsilon) {
            onOptional(exp.exp1);
            return;
        }
    
        visitBinExp("choice", exp, ChoiceExp.class);
    }
    
    public void onSequence(SequenceExp exp) {
        visitBinExp("group", exp, SequenceExp.class);
    }
    
    public void visitBinExp(String elementName, BinaryExp exp, Class type) {
        // since AGM is binarized,
        //  a b c  is represented as
        //  a  b c 
        // this method print them as  a b c 
        writer.start(elementName);
        Expression[] children = exp.getChildren();
        for (int i = 0; i < children.length; i++)
            children[i].visit(this);
        writer.end(elementName);
    }
    
    public void onMixed(MixedExp exp) {
        writer.start("mixed");
        visitUnary(exp.exp);
        writer.end("mixed");
    }
    
    public void onOneOrMore(OneOrMoreExp exp) {
        writer.start("oneOrMore");
        visitUnary(exp.exp);
        writer.end("oneOrMore");
    }
    
    protected void onZeroOrMore(OneOrMoreExp exp) {
        // note that this method is not a member of TREXPatternVisitor.
        writer.start("zeroOrMore");
        visitUnary(exp.exp);
        writer.end("zeroOrMore");
    }
    
    public void onAttribute(AttributeExp exp) {
        writer.start("attribute");
        context.writeNameClass(exp.nameClass);
        visitUnary(exp.exp);
        writer.end("attribute");
    }
    
    /**
     * print expression but surpress unnecessary sequence.
     */
    public void visitUnary(Expression exp) {
        // TREX treats  p q 
        // as  p q 
        // This method tries to exploit this property to
        // simplify the result.
        if (exp instanceof SequenceExp) {
            SequenceExp seq = (SequenceExp)exp;
            visitUnary(seq.exp1);
            seq.exp2.visit(this);
        } else
            exp.visit(this);
    }
    
    public void onValue( ValueExp exp ) {
        if (exp.dt instanceof XSDatatypeImpl) {
            XSDatatypeImpl base = (XSDatatypeImpl)exp.dt;

            final Vector ns = new Vector();

            String lex = base.convertToLexicalValue(exp.value, new SerializationContext() {
                public String getNamespacePrefix(String namespaceURI) {
                    int cnt = ns.size() / 2;
                    ns.add("xmlns:ns" + cnt);
                    ns.add(namespaceURI);
                    return "ns" + cnt;
                }
            });

            if (base != TokenType.theInstance) {
                // if the type is token, we don't need @type.
                ns.add("type");
                ns.add(base.getName());
            }

            writer.start("value", (String[])ns.toArray(new String[0]));
            writer.characters(lex);
            writer.end("value");
            return;
        }
            
        throw new UnsupportedOperationException( exp.dt.getClass().getName() );
    }
        
    public void onData(DataExp exp) {
        Datatype dt = exp.dt;

        if (dt instanceof XSDatatypeImpl) {
            XSDatatypeImpl dti = (XSDatatypeImpl)dt;

            if (isPredefinedType(dt)) {
                // it's a pre-defined types.
                writer.element("data", new String[] { "type", dti.getName()});
            } else {
                serializeDataType(dti);
            }
            return;
        }

        // unknown datatype
        writer.element("data-unknown", new String[] { "class", dt.getClass().getName()});
    }
        
        
    /**
     * serializes the given datatype.
     * 
     * The caller should generate events for <simpleType> element
     * if necessary.
     */
    protected void serializeDataType(XSDatatype dt) {

        if (dt instanceof UnionType) {
            serializeUnionType((UnionType)dt);
            return;
        }

        // store names of the applied facets into this set
        Set appliedFacets = new HashSet();

        // store effective facets (those which are not shadowed by another facet).
        Vector effectiveFacets = new Vector();

        XSDatatype x = dt;
        while (x instanceof DataTypeWithFacet || x instanceof FinalComponent) {

            if (x instanceof FinalComponent) {
                // skip FinalComponent
                x = x.getBaseType();
                continue;
            }

            String facetName = ((DataTypeWithFacet)x).facetName;

            if (facetName.equals(XSDatatypeImpl.FACET_ENUMERATION)) {
                // if it contains enumeration, then we will serialize this
                // by using s.
                serializeEnumeration((XSDatatypeImpl)dt, (EnumerationFacet)x);
                return;
            }

            if (facetName.equals(XSDatatypeImpl.FACET_WHITESPACE)) {
                // TODO: better error handling
                System.err.println("warning: unsupported whiteSpace facet is ignored");
                x = x.getBaseType();
                continue;
            }

            // find the same facet twice.
            // pattern is allowed more than once.
            if (!appliedFacets.contains(facetName)) {

                appliedFacets.add(facetName);
                effectiveFacets.add(x);
            }

            x = ((DataTypeWithFacet)x).baseType;
        }

        if (x instanceof ListType) {
            // the base type is list.
            serializeListType((XSDatatypeImpl)dt);
            return;
        }

        // it cannot be the union type. Union type cannot be derived by
        // restriction.

        // so this must be one of the pre-defined types.
        if (!(x instanceof ConcreteType))
            throw new Error(x.getClass().getName());

        if (x instanceof com.sun.msv.grammar.relax.EmptyStringType) {
            // empty token will do.
            writer.element("value");
            return;
        }
        if (x instanceof com.sun.msv.grammar.relax.NoneType) {
            // "none" is equal to 
            writer.element("notAllowed");
            return;
        }

        writer.start("data", new String[] { "type", x.getName()});

        // serialize effective facets
        for (int i = effectiveFacets.size() - 1; i >= 0; i--) {
            DataTypeWithFacet dtf = (DataTypeWithFacet)effectiveFacets.get(i);

            if (dtf instanceof LengthFacet) {
                param("length", Long.toString(((LengthFacet)dtf).length));
            } else if (dtf instanceof MinLengthFacet) {
                param("minLength", Long.toString(((MinLengthFacet)dtf).minLength));
            } else if (dtf instanceof MaxLengthFacet) {
                param("maxLength", Long.toString(((MaxLengthFacet)dtf).maxLength));
            } else if (dtf instanceof PatternFacet) {
                String pattern = "";
                PatternFacet pf = (PatternFacet)dtf;
                for (int j = 0; j < pf.getRegExps().length; j++) {
                    if (pattern.length() != 0)
                        pattern += "|";
                    pattern += pf.patterns[j];
                }
                param("pattern", pattern);
            } else if (dtf instanceof TotalDigitsFacet) {
                param("totalDigits", Long.toString(((TotalDigitsFacet)dtf).precision));
            } else if (dtf instanceof FractionDigitsFacet) {
                param("fractionDigits", Long.toString(((FractionDigitsFacet)dtf).scale));
            } else if (dtf instanceof RangeFacet) {
                param(dtf.facetName, dtf.convertToLexicalValue(((RangeFacet)dtf).limitValue, null));
                // we don't need to pass SerializationContext because it is only
                // for QName.
            } else if (dtf instanceof WhiteSpaceFacet) {
                ; // do nothing.
            } else
                // undefined facet type
                throw new Error();
        }

        writer.end("data");
    }
    
    protected void param(String name, String value) {
        writer.start("param", new String[] { "name", name });
        writer.characters(value);
        writer.end("param");
    }
        
    /**
     * returns true if the specified type is a pre-defined XSD type
     * without any facet.
     */
    protected boolean isPredefinedType(Datatype x) {
        return !(
            x instanceof DataTypeWithFacet
                || x instanceof UnionType
                || x instanceof ListType
                || x instanceof FinalComponent
                || x instanceof com.sun.msv.grammar.relax.EmptyStringType
                || x instanceof com.sun.msv.grammar.relax.NoneType);
    }
        
    /**
     * serializes a union type.
     * this method is called by serializeDataType method.
     */
    protected void serializeUnionType(UnionType dt) {
        writer.start("choice");

        // serialize member types.
        for (int i = 0; i < dt.memberTypes.length; i++)
            serializeDataType(dt.memberTypes[i]);

        writer.end("choice");
    }
        
    /**
     * serializes a list type.
     * this method is called by serializeDataType method.
     */
    protected void serializeListType(XSDatatypeImpl dt) {

        ListType base = (ListType)dt.getConcreteType();

        if (dt.getFacetObject(XSDatatype.FACET_LENGTH) != null) {
            // with the length facet.
            int len = ((LengthFacet)dt.getFacetObject(XSDatatype.FACET_LENGTH)).length;
            writer.start("list");
            for (int i = 0; i < len; i++)
                serializeDataType(base.itemType);
            writer.end("list");

            return;
        }

        if (dt.getFacetObject(XSDatatype.FACET_MAXLENGTH) != null)
            throw new UnsupportedOperationException("warning: maxLength facet to list type is not properly converted.");

        MinLengthFacet minLength = (MinLengthFacet)dt.getFacetObject(XSDatatype.FACET_MINLENGTH);

        writer.start("list");
        if (minLength != null) {
            // list n times
            for (int i = 0; i < minLength.minLength; i++)
                serializeDataType(base.itemType);
        }
        writer.start("zeroOrMore");
        serializeDataType(base.itemType);
        writer.end("zeroOrMore");
        writer.end("list");
    }

    /**
     * serializes a type with enumeration.
     * this method is called by serializeDataType method.
     */
    protected void serializeEnumeration(XSDatatypeImpl dt, EnumerationFacet enums) {

        Object[] values = enums.values.toArray();

        if (values.length > 1)
            writer.start("choice");

        for (int i = 0; i < values.length; i++) {
            final Vector ns = new Vector();

            String lex = dt.convertToLexicalValue(values[i], new SerializationContext() {
                public String getNamespacePrefix(String namespaceURI) {
                    int cnt = ns.size() / 2;
                    ns.add("xmlns:ns" + cnt);
                    ns.add(namespaceURI);
                    return "ns" + cnt;
                }
            });

            // make sure that the converted lexical value is allowed by this type.
            // sometimes, facets that are added later rejects some of
            // enumeration values.

            boolean allowed = dt.isValid(lex, new ValidationContext() {

                public String resolveNamespacePrefix(String prefix) {
                    if (!prefix.startsWith("ns"))
                        return null;
                    int i = Integer.parseInt(prefix.substring(2));
                    return (String)ns.get(i * 2 + 1);
                }

                public boolean isUnparsedEntity(String name) {
                    return true;
                }
                public boolean isNotation(String name) {
                    return true;
                }
                public String getBaseUri() {
                    return null;
                }
            });

            ns.add("type");
            ns.add(dt.getConcreteType().getName());

            if (allowed) {
                writer.start("value", (String[])ns.toArray(new String[0]));
                writer.characters(lex);
                writer.end("value");
            }
        }

        if (values.length > 1)
            writer.end("choice");
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy