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

com.predic8.schema.Schema.groovy Maven / Gradle / Ivy

Go to download

Membrane SOA Model is an open-source toolkit and Java API for WSDL and XML Schema, licensed under ASF 2.0. that can: Parse, create or modify a WSDL or XML Schema Document from Java Compare two WSDL or XML Schema Documents Create a SOAP Request or Template Analyze a WSDL or Schema document and generate an HMTL report

There is a newer version: 2.0.1
Show newest version
/* Copyright 2012 predic8 GmbH, www.predic8.com

   Licensed under the Apache License, Version 2.0 (the "License");
   you may not use this file except in compliance with the License.
   You may obtain a copy of the License at

   http://www.apache.org/licenses/LICENSE-2.0

   Unless required by applicable law or agreed to in writing, software
   distributed under the License is distributed on an "AS IS" BASIS,
   WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
   See the License for the specific language governing permissions and
   limitations under the License. */

package com.predic8.schema;

import static com.predic8.soamodel.Consts.SCHEMA_NS

import groovy.xml.*

import javax.xml.namespace.QName as JQName

import org.slf4j.Logger
import org.slf4j.LoggerFactory

import com.predic8.schema.creator.*
import com.predic8.soamodel.*
import com.predic8.wsdl.Definitions
import com.predic8.xml.util.PrefixedName
import com.predic8.xml.util.ResourceResolver

class Schema extends SchemaComponent{
  
  public final static JQName ELEMENTNAME = new JQName(SCHEMA_NS, 'schema')
  public final static JQName STRING = new JQName(SCHEMA_NS, 'string')
  public final static JQName INT = new JQName(SCHEMA_NS, 'int')
  public final static JQName INTEGER = new JQName(SCHEMA_NS, 'integer')
  public final static JQName DATE = new JQName(SCHEMA_NS, 'date')
  
  private static final Logger log = LoggerFactory.getLogger(Schema.class)

  /**
   * if the schema is embedded in a WSDL document, definitions is a reference 
   * to the schema. Definitions is null if the schema is not embedded in a 
   * WSDL document.
   */
  Definitions definitions
  ResourceResolver resourceResolver
  
  String baseDir = ''
  String schemaLocation = ''
  String targetNamespace = ''
  String attributeFormDefault = "unqualified"
  String elementFormDefault = "unqualified"
  List imports = []
	/**
	 * includes contains a list of Include objects for each included schema.
	 */
  List includes = []
	
	/**
	 * includedPaths contains a list of Strings with the real schema location 
	 * and is used to prevent a cycling include circle to come to a deadlock.
	 */
	List includedPaths = []
  List complexTypes = []
  List groups = []
  List simpleTypes = []
  List elements = []
  Annotation annotation
  List attributes = []
  List attributeGroups = []

  final Map getTypeCache = [:]

  @Lazy List allSchemasCache = { [this] + importedSchemas }.call()
  
  Schema(){}
  
  Schema(String tns){
    this.targetNamespace = tns
    namespaces['tns'] = tns
  }
  
  protected parseAttributes(token, ctx){
    targetNamespace = ctx.targetNamespace ?: (token.getAttributeValue( null , 'targetNamespace') ?: '')
    ctx.setImportedSchema(this)
    attributeFormDefault = token.getAttributeValue( null , 'attributeFormDefault') ?: "unqualified"
    elementFormDefault = token.getAttributeValue( null , 'elementFormDefault') ?: "unqualified"
  }

  protected parseChildren(token, child, ctx){
    switch (child ){
      case 'annotation' :
      annotation = new Annotation(schema: this)
      annotation.parse(token, ctx) ; break
      case 'complexType':
      def complexType = new ComplexType(schema:this)
      complexType.parse(token, ctx)
      complexTypes << complexType ; break
      case 'group':
      def group = new Group(schema:this)
      group.parse(token, ctx)
      groups << group ;break
      case 'simpleType':
			def simpleType = new SimpleType(schema:this)
      simpleType.parse(token, ctx)
      simpleTypes << simpleType ; break
      case 'element':
      def element = new Element(schema:this,toplevel:true)
      element.parse(token, ctx)
      elements << element ; break
      case 'import':
      def imp = new Import(schema : this)
      ctx.token = token
      imp.parse(token, ctx)
      imports << imp ; break
      case 'include':
      ctx.token = token
      def inc = new Include(schema : this)
      log.debug "including schema $inc"
      inc.parse(token, ctx)
      includes << inc ; break
      case 'attribute' :
      def attribute = new Attribute(schema: this)
      attribute.parse(token, ctx)
      attributes << attribute ; break
      case 'attributeGroup' :
      def attributeGroup = new AttributeGroup(schema: this)
      attributeGroup.parse(token, ctx)
      attributeGroups << attributeGroup ; break
    }
  }

  protected getElementName(){
    'schema'
  }
  
  def asXml(namespaces) {
    def strWriter = new StringWriter()
    namespaces.each { k,v->
      this.namespaces[k] = this.namespaces[k] ?: v 
    }
    def creator = new SchemaCreator(builder : new MarkupBuilder(strWriter))
    create(creator, new SchemaCreatorContext())
    strWriter.toString()
  }

  List getAllElements() {
    (elements + importedSchemas.elements).flatten()
  }
  
	/**
	 * Resolves the element object for the given qname from the known schemas.
	 * If the schema is embedded in a WSDL document, all other schemas in the
	 * WSDl will be checked, even if there are no imports of them in this schema.
	 * @param groovy.xml.QName
	 * @return Element
	 */
  Element getElement(QName elementRef, String prefix = '') {
		if(!elementRef) return
    Element refElement = allSchemas.elements.flatten().find {
      it.name == elementRef.localPart && it.schema.targetNamespace == elementRef.namespaceURI
		} ?: definitions?.getSchemaLoadKnownSchemaIfNeeded(elementRef.namespaceURI)?.getElement(elementRef.localPart)
		if(!refElement) throw new ElementRefAccessException(
			"Could not find the referenced element '${elementRef.localPart}' in namespace '${elementRef.namespaceURI}'.",
			elementRef, prefix ?: getPrefix(elementRef.namespaceURI))
		refElement
  }
	
	Element getElement(PrefixedName elementPN){
		def uri
		if(elementPN.prefix){
			uri = getNamespace(elementPN.prefix)
		} else {
			return elements.find{ it.name==elementPN.localName }
		}
		allSchemas.elements.flatten().find {
			it.name == elementPN.localName && it.schema.targetNamespace == uri
		}
	}
	
  Element getElement(String elementName){
    getElement(new PrefixedName(elementName))
  }

  Attribute getAttribute(QName ref){
		allSchemas.attributes.flatten().find{it.qname == ref}
  }
  
  Attribute getAttribute(String name){
    attributes.find{it.name == name}
  }
  
  AttributeGroup getAttributeGroup(QName ref){
    allSchemas.attributeGroups.flatten().find{it.name == ref.localPart}
  }
  
  AttributeGroup getAttributeGroup(String name){
    attributeGroups.find{it.name == name}
  }

    /**
     * Resolves the type object for the given qname from the known schemas, and caches
     * the reference if found.
     * If the schema is embedded in a WSDL document, all other schemas in the
     * WSDl will be checked, even if there are no imports of them in this schema.
     * @param groovy.xml.QName
     * @return TypeDefinition
     *
     * @param typeRef
     * @return the cached or found typeDefinition
     */
    TypeDefinition getType(QName typeRef){
        def result = getTypeCache[typeRef]

        if (!result) {
            result = findType(typeRef)
            getTypeCache[typeRef] = result
        }

        result
    }

	/**
	 * Resolves the type object for the given qname from the known schemas.
	 * If the schema is embedded in a WSDL document, all other schemas in the 
	 * WSDl will be checked, even if there are no imports of them in this schema.
	 * @param groovy.xml.QName
	 * @return TypeDefinition
	 */
  TypeDefinition findType(QName typeRef){
		if(!typeRef) return
		if(typeRef.namespaceURI == Consts.SCHEMA_NS) return new BuiltInSchemaType(qname: typeRef)
    TypeDefinition refType = (allSchemas.complexTypes + allSchemas.simpleTypes + allSchemas.groups).flatten().find{
      it.qname == typeRef
		} ?: definitions?.getSchemaLoadKnownSchemaIfNeeded(typeRef.namespaceURI)?.getType(typeRef.localPart)
		if(!refType) throw new TypeRefAccessException(
			"Could not find the referenced type '${typeRef.localPart}' in namespace '${typeRef.namespaceURI}'.",
			typeRef, getPrefix(typeRef.namespaceURI))
		refType
  }
  
  TypeDefinition getType(String typeName){
		(complexTypes + simpleTypes + groups).flatten().find{
			it.name == typeName
		}
  }
  
  ComplexType getComplexType(String typeName){
    getType(new QName(targetNamespace,typeName))
  }
	
	SimpleType getSimpleType(String typeName){
		getType(new QName(targetNamespace,typeName))
	}

  Group getGroup(QName qname){
    (allSchemas.groups).flatten().find{
      it.qname == qname
    }
  }


  List getAllSchemas(){
    allSchemasCache
  }
	
  List getImportedSchemas(){
    getImportedSchemas([])
  }

  Schema getImportedSchema(targetNamespace) {
    importedSchemas.find { it.targetNamespace==targetNamespace }
  }

  List getImportedSchemas(importedSchemas){
    log.debug "imported Schemas: ${importedSchemas.targetNamespace}"
    def schemas = []
    imports.each { imp ->
      def schema
      if(definitions) {
				if(!imp.namespace) schema = definitions.localSchemas.find{ it.targetNamespace == ''}
				else schema = definitions.localSchemas.find{ it.targetNamespace == imp.namespace}
      }
      if(imp.importSchema) {
        schema = imp.importSchema
      }
      if(schema && !(importedSchemas.contains(schema))) {
        schemas << schema
        schemas.addAll(schema.getImportedSchemas(schemas+importedSchemas))
      }
    }

    schemas
  }
  
  public void add(ComplexType ct){
    ct.parent = this
    ct.schema = this
    complexTypes << ct
  }
  
  ComplexType newComplexType(String name){
    def ct = new ComplexType(name: name, qname: new QName(targetNamespace, name), schema: this, parent: this)
    complexTypes << ct
    ct
  }
  
  ComplexType newComplexType(){
    new ComplexType(schema:this, parent: this)
  }
	
	SimpleType newSimpleType(){
		new SimpleType(schema: this, parent: this)
	}
	
	SimpleType newSimpleType(String name){
		def st = new SimpleType(qname: new QName(targetNamespace, name), schema: this, parent: this, name: name)
		simpleTypes << st
		st
	}
  
  Element newElement(String name, JQName type){
    def e = new Element(name: name, type: new QName(type.namespaceURI, type.localPart), schema: this, parent: this)
    elements << e
    e
  }
  
  Element newElement(String name, String type){
    def e = new Element(name: name, type: new QName(targetNamespace, type), schema: this, parent: this)
    elements << e
    e
  }
  
  Element newElement(String name){
    def e = new Element(name: name, schema: this, parent: this)
    elements << e
    e
  }
	
	Schema getSchema() {
		//SchemaComponent.schema is used in parse() and getNamspaceUri() methods
		this
	}
  
  def create(creator, CreatorContext ctx) {
    creator.createSchema(this, ctx.clone())
  }
	
	List validate(AbstractParserContext ctx) {
		new SchemaValidator().validate(this, ctx)
	}
  
  boolean equals(obj){
    if(!(obj instanceof Schema)) 
    return false
    if (obj.is(this))
    return true
    if(obj.targetNamespace != targetNamespace)
    return false
    if(obj.complexTypes && obj.complexTypes.qname != complexTypes.qname)
    return false
    if(obj.simpleTypes && obj.simpleTypes.qname != simpleTypes.qname)
    return false
    if(obj.elements && obj.elements.name != elements.name)
    return false
    true
  }
  
  String toString(){
    "schema[ baseDir=$baseDir, targetNamespace=$targetNamespace, namespaces=$namespaceContext, imports=$imports, includes=$includes, complexTypes=${complexTypes.name}, simpleTypes=${simpleTypes.name}, elements=${elements.name} ]\n"
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy