|
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
// See the LICENSE file in the project root for more information.
namespace System.ServiceModel.Description
{
using System.Collections;
using System.Collections.Generic;
using System.Collections.ObjectModel;
using System.Runtime;
using Microsoft.Xml;
using Microsoft.Xml.Schema;
internal static class SchemaHelper
{
static internal void AddElementForm(XmlSchemaElement element, XmlSchema schema)
{
if (schema.ElementFormDefault != XmlSchemaForm.Qualified)
element.Form = XmlSchemaForm.Qualified;
}
static internal void AddElementToSchema(XmlSchemaElement element, XmlSchema schema, XmlSchemaSet schemaSet)
{
XmlSchemaElement existingElement = (XmlSchemaElement)schema.Elements[new XmlQualifiedName(element.Name, schema.TargetNamespace)];
if (existingElement != null)
{
if (element.SchemaType == existingElement.SchemaType && element.SchemaTypeName == existingElement.SchemaTypeName)
return;
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(string.Format(SRServiceModel.SFxConflictingGlobalElement, element.Name, schema.TargetNamespace, GetTypeName(element), GetTypeName(existingElement))));
}
schema.Items.Add(element);
if (!element.SchemaTypeName.IsEmpty)
AddImportToSchema(element.SchemaTypeName.Namespace, schema);
schemaSet.Reprocess(schema);
}
static internal void AddImportToSchema(string ns, XmlSchema schema)
{
if (NamespacesEqual(ns, schema.TargetNamespace)
|| NamespacesEqual(ns, XmlSchema.Namespace)
|| NamespacesEqual(ns, XmlSchema.InstanceNamespace))
return;
foreach (object item in schema.Includes)
{
if (item is XmlSchemaImport && NamespacesEqual(ns, ((XmlSchemaImport)item).Namespace))
return;
}
XmlSchemaImport import = new XmlSchemaImport();
if (ns != null && ns.Length > 0)
import.Namespace = ns;
schema.Includes.Add(import);
}
static internal void AddTypeToSchema(XmlSchemaType type, XmlSchema schema, XmlSchemaSet schemaSet)
{
XmlSchemaType existingType = (XmlSchemaType)schema.SchemaTypes[new XmlQualifiedName(type.Name, schema.TargetNamespace)];
if (existingType != null)
{
if (existingType == type)
return;
throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new InvalidOperationException(string.Format(SRServiceModel.SFxConflictingGlobalType, type.Name, schema.TargetNamespace)));
}
schema.Items.Add(type);
schemaSet.Reprocess(schema);
}
static internal XmlSchema GetSchema(string ns, XmlSchemaSet schemaSet)
{
if (ns == null) { ns = String.Empty; }
ICollection schemas = schemaSet.Schemas();
foreach (XmlSchema existingSchema in schemas)
{
if ((existingSchema.TargetNamespace == null && ns.Length == 0) || ns.Equals(existingSchema.TargetNamespace))
return existingSchema;
}
XmlSchema schema = new XmlSchema();
schema.ElementFormDefault = XmlSchemaForm.Qualified;
if (ns.Length > 0)
schema.TargetNamespace = ns;
schemaSet.Add(schema);
return schema;
}
private static string GetTypeName(XmlSchemaElement element)
{
if (element.SchemaType != null)
return "anonymous";
if (!element.SchemaTypeName.IsEmpty)
return element.SchemaTypeName.ToString();
return String.Empty;
}
// Compare XmlSchemaElement properties set by WsdlExporter: do not check element QName,
// this code only called for elements with matching name and namespace
internal static bool IsMatch(XmlSchemaElement e1, XmlSchemaElement e2)
{
// this code only called for elements with matching name and namespace
Fx.Assert(e1.Name == e2.Name, "");
// Anonymous types never match
if (e1.SchemaType != null || e2.SchemaType != null)
return false;
if (e1.SchemaTypeName != e2.SchemaTypeName)
return false;
// do not need to check parent Schema.ElementFormDefault: see AddElementForm in this class.
if (e1.Form != e2.Form)
return false;
if (e1.IsNillable != e2.IsNillable)
return false;
return true;
}
static internal bool NamespacesEqual(string ns1, string ns2)
{
if (ns1 == null || ns1.Length == 0)
return (ns2 == null || ns2.Length == 0);
else
return ns1 == ns2;
}
static internal void Compile(XmlSchemaSet schemaSet, Collection<MetadataConversionError> errors)
{
ValidationEventHandler validationEventHandler = new ValidationEventHandler(delegate (object sender, ValidationEventArgs args)
{
HandleSchemaValidationError(sender, args, errors);
}
);
schemaSet.ValidationEventHandler += validationEventHandler;
schemaSet.Compile();
schemaSet.ValidationEventHandler -= validationEventHandler;
}
static internal void HandleSchemaValidationError(object sender, ValidationEventArgs args, Collection<MetadataConversionError> errors)
{
MetadataConversionError warning = null;
if (args.Exception != null && args.Exception.SourceUri != null)
{
XmlSchemaException ex = args.Exception;
warning = new MetadataConversionError(string.Format(SRServiceModel.SchemaValidationError, ex.SourceUri, ex.LineNumber, ex.LinePosition, ex.Message));
}
else
{
warning = new MetadataConversionError(string.Format(SRServiceModel.GeneralSchemaValidationError, args.Message));
}
if (!errors.Contains(warning))
errors.Add(warning);
}
private static IList<string> s_xsdValueTypePrimitives = new string[]
{
"boolean", "float", "double", "decimal", "long", "unsignedLong", "int", "unsignedInt", "short", "unsignedShort", "byte", "unsignedByte",
"duration", "dateTime", "integer", "positiveInteger", "negativeInteger", "nonPositiveInteger"
};
private static IList<string> s_dataContractPrimitives = new string[]
{
"char", "guid"
};
private static IList<string> s_xmlSerializerPrimitives = new string[]
{
"char", "guid"
};
private static string s_dataContractSerializerNamespace = "http://schemas.microsoft.com/2003/10/Serialization/";
private static string s_xmlSerializerNamespace = "http://microsoft.com/wsdl/types/";
internal static bool IsElementValueType(XmlSchemaElement element)
{
XmlQualifiedName typeName = element.SchemaTypeName;
if (typeName == null || typeName.IsEmpty)
return false;
if (typeName.Namespace == XmlSchema.Namespace)
return s_xsdValueTypePrimitives.Contains(typeName.Name);
if (typeName.Namespace == s_dataContractSerializerNamespace)
return s_dataContractPrimitives.Contains(typeName.Name);
if (typeName.Namespace == s_xmlSerializerNamespace)
return s_dataContractPrimitives.Contains(typeName.Name);
return false;
}
}
}
|