XmlSchemaClassGenerator-beta 2.1.1183

XmlSchemaClassGenerator

Nuget Build status codecov.io netstandard2.0 net462

A console program and library to generate XmlSerializer compatible C# classes from XML Schema files.

Features

  • Map XML namespaces to C# namespaces, either explicitly or through a (configurable) function
  • Generate C# XML comments from schema annotations
  • Generate DataAnnotations attributes from schema restrictions
  • Use Collection<T> properties (initialized in constructor and with private setter)
  • Map xs:integer and derived types to the closest possible .NET type, if not possible - fall back to string. Can be overriden by explicitly defined type (int, long, or decimal)
  • Automatic properties
  • Pascal case for classes and properties
  • Generate nullable adapter properties for optional elements and attributes without default values (see below)
  • Optional support for PCL
  • Optional support for INotifyPropertyChanged
  • Optional support for Entity Framework Code First (automatically generate key properties)
  • Optionally generate interfaces for groups and attribute groups
  • Optionally generate one file per class
  • Support for nullable reference types (NRTs) through AllowNullAttribute and MaybeNullAttribute
  • Optionally generate a common specific type for union member types

Unsupported:

  • Some restriction types
  • Recursive choices and choices whose elements have minOccurs > 0 or nillable="true" (see below)
  • Possible name clashes and invalid identifiers when names contain non-alphanumeric characters
  • Groups with maxOccurs > 0

Usage

For command line use, choose your preferred installation:

Usage: xscgen [OPTIONS]+ xsdFile...
Generate C# classes from XML Schema files.
Version ...
xsdFiles may contain globs, e.g. "content\{schema,xsd}\**\*.xsd", and URLs.
Append - to option to disable it, e.g. --interface-.

Options:
  -h, --help                 show this message and exit
  -n, --namespace=VALUE      map an XML namespace to a C# namespace
                               Separate XML namespace and C# namespace by '='.
                               A single value (no '=') is taken as the C#
                               namespace the empty XML namespace is mapped to.
                               One option must be given for each namespace to
                               be mapped.
                               A file name may be given by appending a pipe
                               sign (|) followed by a file name (like schema.
                               xsd) to the XML namespace.
                               If no mapping is found for an XML namespace, a
                               name is generated automatically (may fail).
      --nf, --namespaceFile=VALUE
                             file containing mappings from XML namespaces to C#
                               namespaces
                               The line format is one mapping per line: XML
                               namespace = C# namespace [optional file name].
                               Lines starting with # and empty lines are
                               ignored.
      --tns, --typeNameSubstitute=VALUE
                             substitute a generated type/member name
                               Separate type/member name and substitute name by
                               '='.
                               Prefix type/member name with an appropriate kind
                               ID as documented at: https://t.ly/HHEI.
                               Prefix with 'A:' to substitute any type/member.
      --tnsf, --typeNameSubstituteFile=VALUE
                             file containing generated type/member name
                               substitute mappings
                               The line format is one mapping per line:
                               prefixed type/member name = substitute name.
                               Lines starting with # and empty lines are
                               ignored.
  -o, --output=FOLDER        the FOLDER to write the resulting .cs files to
  -d, --datetime-offset      map xs:datetime and derived types to System.
                               DateTimeOffset instead of System.DateTime
  -i, --integer=TYPE         map xs:integer and derived types to TYPE instead
                               of automatic approximation
                               TYPE can be i[nt], l[ong], or d[ecimal]
      --fb, --fallback, --use-integer-type-as-fallback
                             use integer type specified via -i only if no type
                               can be deduced
  -e, --edb, --enable-data-binding
                             enable INotifyPropertyChanged data binding
  -r, --order                emit order for all class members stored as XML
                               element
  -c, --pcl                  PCL compatible output
  -p, --prefix=PREFIX        the PREFIX to prepend to auto-generated namespace
                               names
  -v, --verbose              print generated file names on stdout
  -0, --nullable             generate nullable adapter properties for optional
                               elements/attributes w/o default values
  -f, --ef                   generate Entity Framework Code First compatible
                               classes
  -t, --interface            generate interfaces for groups and attribute
                               groups (default is enabled)
  -a, --pascal               use Pascal case for class and property names (
                               default is enabled)
      --av, --assemblyVisible
                             use the internal visibility modifier (default is
                               false)
  -u, --enableUpaCheck       should XmlSchemaSet check for Unique Particle
                               Attribution (UPA) (default is enabled)
      --ct, --collectionType=VALUE
                             collection type to use (default is System.
                               Collections.ObjectModel.Collection`1)
      --cit, --collectionImplementationType=VALUE
                             the default collection type implementation to use (
                               default is null)
      --csm, --collectionSettersMode=Private, Public, PublicWithoutConstructorInitialization, Init, InitWithoutConstructorInitialization
                             generate a private, public, or init-only setter
                               with or without backing field initialization for
                               collections
                               (default is Private; can be: Private, Public,
                               PublicWithoutConstructorInitialization, Init,
                               InitWithoutConstructorInitialization)
      --ctro, --codeTypeReferenceOptions=GlobalReference, GenericTypeParameter
                             the default CodeTypeReferenceOptions Flags to use (
                               default is unset; can be: GlobalReference,
                               GenericTypeParameter)
      --tvpn, --textValuePropertyName=VALUE
                             the name of the property that holds the text value
                               of an element (default is Value)
      --dst, --debuggerStepThrough
                             generate DebuggerStepThroughAttribute (default is
                               enabled)
      --dc, --disableComments
                             do not include comments from xsd
      --nu, --noUnderscore   do not generate underscore in private member name (
                               default is false)
      --da, --description    generate DescriptionAttribute (default is true)
      --cc, --complexTypesForCollections
                             generate complex types for collections (default is
                               true)
  -s, --useShouldSerialize   use ShouldSerialize pattern instead of Specified
                               pattern (default is false)
      --sf, --separateFiles  generate a separate file for each class (default
                               is false)
      --nh, --namespaceHierarchy
                             generate a separate folder for namespace hierarchy.
                                Implies "separateFiles" if true (default is
                               false)
      --sg, --separateSubstitutes
                             generate a separate property for each element of a
                               substitution group (default is false)
      --dnfin, --doNotForceIsNullable
                             do not force generator to emit IsNullable = true
                               in XmlElement annotation for nillable elements
                               when element is nullable (minOccurs < 1 or
                               parent element is choice) (default is false)
      --cn, --compactTypeNames
                             use type names without namespace qualifier for
                               types in the using list (default is false)
      --cl, --commentLanguages=VALUE
                             comment languages to use (default is en; supported
                               are en, de)
      --un, --uniqueTypeNames
                             generate type names that are unique across
                               namespaces (default is false)
      --gc, --generatedCodeAttribute
                             add version information to GeneratedCodeAttribute (
                               default is true)
      --nc, --netCore        generate .NET Core specific code that might not
                               work with .NET Framework (default is false)
      --nr, --nullableReferenceAttributes
                             generate attributes for nullable reference types (
                               default is false)
      --ar, --useArrayItemAttribute
                             use ArrayItemAttribute for sequences with single
                               elements (default is true)
      --es, --enumAsString   Use string instead of enum for enumeration
      --ca, --commandArgs    generate a comment with the exact command line
                               arguments that were used to generate the source
                               code (default is true)
      --uc, --unionCommonType
                             generate a common type for unions if possible (
                               default is false)
      --dtd, --allowDtdParse
                             allow DTD parsing (default is false)
      --ns, --namingScheme   use the specified naming scheme for class and
                               property names (default is Pascal; can be:
                               Direct, Pascal, Legacy)
      --fu, --forceUriScheme=VALUE
                             force URI scheme when resolving URLs (default is
                               none; can be: none, same, or any defined value
                               for scheme, like https or http)

For use from code use the library NuGet package:

var generator = new Generator
{
    OutputFolder = outputFolder,
    Log = s => Console.Out.WriteLine(s),
    GenerateNullables = true,
    NamespaceProvider = new Dictionary<NamespaceKey, string> 
    { 
        { new NamespaceKey("http://wadl.dev.java.net/2009/02"), "Wadl" } 
    }
    .ToNamespaceProvider(new GeneratorConfiguration { NamespacePrefix = "Wadl" }.NamespaceProvider.GenerateNamespace)
};

generator.Generate(files);

Specifying the NamespaceProvider is optional. If you don't provide one, C# namespaces will be generated automatically. The example above shows how to create a custom NamespaceProvider that has a dictionary for a number of specific namespaces as well as a generator function for XML namespaces that are not in the dictionary. In the example the generator function is the default function but with a custom namespace prefix. You can also use a custom generator function, e.g.

var generator = new Generator
{
    NamespaceProvider = new NamespaceProvider
    { 
        GenerateNamespace = key => ...
    }
};

Mapping xsd files to C# namespaces

Using the optional | syntax of the -n command line option you can map individual xsd files to C# namespaces. If you have several input files using the same XML namespace you can still generate an individual C# namespace for the types defined within a single xsd file. For example, if you have two input files a.xsd and b.xsd both of which have the same targetNamespace of http://example.com/namespace you can generate the C# namespaces Example.NamespaceA and Example.NamespaceB:

xscgen -n "|a.xsd=Example.NamespaceA" -n "|b.xsd=Example.NamespaceB" a.xsd b.xsd

Mapping empty XML namespaces

In order to provide a C# namespace name for an empty XML namespace you can specify it on the command line like this:

xscgen -n Example example.xsd

An alternative form that is also valid is -n =Example. Note the space between -n and =Example.

Using mapping files

Instead of specifying the namespace mappings on the command line you can also use a mapping file which should contain one mapping per line in the following format:

# Comment

http://example.com = Example.NamespaceA a.xsd
http://example.com = Example.NamespaceB b.xsd
Empty
# or alternatively
= Empty

Use the --nf option to specify the mapping file.

Substituting generated C# type and member names

If a xsd file specifies obscure names for their types (classes, enums) or members (properties), you can substitute these using the --tns/--typeNameSubstitute= parameter:

xscgen --tns T:Example_RootType=Example --tns T:Example_RootTypeExampleScope=ExampleScope --tns P:StartDateDateTimeValue=StartDate example.xsd

The syntax for substitution is: {kindId}:{generatedName}={substituteName}

The {kindId} is a single character identifier based on documentation/analysis ID format, where valid values are:

ID Scope
P Property
T Type: class, enum, interface
A Any property and/or type

Using substitution files

Instead of specifying the substitutions on the command line you can also use a substitution file which should contain one substitution per line in the following format:

# Comment
T:Example_RootType = Example
T:Example_RootTypeExampleScope = ExampleScope
P:StartDateDateTimeValue = StartDate

Use the --tnsf/--typeNameSubstituteFile option to specify the substitution file.

Nullables

XmlSerializer has been present in the .NET Framework since version 1.1 and has never been updated to provide support for nullables which are a natural fit for the problem of signaling the absence or presence of a value type but have only been present since .NET Framework 2.0.

Instead XmlSerializer has support for a pattern where you provide an additional bool property with "Specified" appended to the name to signal if the original property should be serialized. For example:

<xs:attribute name="id" type="xs:int" use="optional">...</xs:attribute>
[System.Xml.Serialization.XmlAttributeAttribute("id", Form=System.Xml.Schema.XmlSchemaForm.Unqualified, DataType="int")]
public int Id { get; set; }

[System.Xml.Serialization.XmlIgnoreAttribute()]
public bool IdSpecified { get; set; }

XmlSchemaClassGenerator can optionally generate an additional nullable property that works as an adapter to both properties:

[System.Xml.Serialization.XmlAttributeAttribute("id", Form=System.Xml.Schema.XmlSchemaForm.Unqualified, DataType="int")]
public int IdValue { get; set; }
        
[System.Xml.Serialization.XmlIgnoreAttribute()]
public bool IdValueSpecified { get; set; }

[System.Xml.Serialization.XmlIgnoreAttribute()]
public System.Nullable<int> Id
{
    get
    {
        if (this.IdValueSpecified)
        {
            return this.IdValue;
        }
        else
        {
            return null;
        }
    }
    set
    {
        this.IdValue = value.GetValueOrDefault();
        this.IdValueSpecified = value.HasValue;
    }
}

Choice Elements

The support for choice elements differs from that provided by xsd.exe. Xsd.exe generates a property called Item of type object and, if not all choices have a distinct type, another enum property that selects the chosen element. Besides being non-typesafe and non-intuitive, this approach breaks apart if the choices have a more complicated structure (e.g. sequences), resulting in possibly schema-invalid XML.

XmlSchemaClassGenerator currently simply pretends choices are sequences. This means you'll have to take care only to set a schema-valid combination of these properties to non-null values.

Interfaces

Groups and attribute groups in XML Schema are reusable components that can be included in multiple type definitions. XmlSchemaClassGenerator can optionally generate interfaces from these groups to make it easier to access common properties on otherwise unrelated classes. So

<xs:attributeGroup name="Common">
  <xs:attribute name="name" type="xs:string"></xs:attribute>
</xs:attributeGroup>

<xs:complexType name="A">
  <xs:attributeGroup ref="Common"/>
</xs:complexType>

<xs:complexType name="B">
  <xs:attributeGroup ref="Common"/>
</xs:complexType>

becomes

public partial interface ICommon
{
  string Name { get; set; }
}

public partial class A: ICommon
{
  public string Name { get; set; }
}

public partial class B: ICommon
{
  public string Name { get; set; }
}

Collection types

Values for the --collectionType and --collectionImplementationType options have to be given in the format accepted by the Type.GetType() method. For the System.Collections.Generic.List<T> class this means System.Collections.Generic.List`1. Make sure to escape the backtick character (`) to prevent it from being interpreted by the shell.

Integer and derived types

Not all numeric types defined by XML Schema can be safely and accurately mapped to .NET numeric data types, however, it's possible to approximate the mapping based on the integer bounds and restrictions such as totalDigits.
If an explicit integer type mapping is specified via --integer=TYPE, that type will be used, otherwise an approximation will be made based on the table below. If you additionally specify --fallback, the type specified via --integer=TYPE will be used only if no type can be deduced by applying the rules below.

If the restrictions minInclusive and maxInclusive are present on the integer element, then the smallest CLR type that fully encompasses the specified range will be used. Unsigned types are given precedence over signed types. The following table shows the possible ranges and their corresponding CLR type, in the order they will be applied.

Minimum (Inclusive) Maximum (Inclusive) C# type
sbyte.MinValuesbyte.MaxValuesbyte
byte.MinValuebyte.MaxValuebyte
ushort.MinValueushort.MaxValueushort
short.MinValueshort.MaxValueshort
uint.MinValueuint.MaxValueuint
int.MinValueint.MaxValueint
ulong.MinValueulong.MaxValueulong
long.MinValuelong.MaxValuelong
decimal.MinValuedecimal.MaxValuedecimal

If the range specified by minInclusive and maxInclusive does not fit in any CLR type, or if those restrictions are not present, then the totalDigits restriction will be used, as shown in the following table.

XML Schema type totalDigits C# type
xs:positiveInteger
xs:nonNegativeInteger
<3byte
<5ushort
<10uint
<20ulong
<30decimal
>=30string
xs:integer
xs:nonPositiveInteger
xs:negativeInteger
<3sbyte
<5short
<10int
<19long
<29decimal
>=29string

Unions

If you specify --unionCommonType, XmlSchemaClassGenerator will try to determine a common type for a union's member types. If, for example, the member types are all integer types, then the narrowest integer type will be used that can fit all member types.

Note that semantic issues might arise with this approach. For example, DateTime values are serialized with both date and time information included. See discussion at #397.

Contributing

Contributions are welcome. Here are some guidelines:

  • If it's not a trivial fix, please submit an issue first
  • Try and blend new code with the existing code's style
  • Add unit tests

Showing the top 20 packages that depend on XmlSchemaClassGenerator-beta.

Packages Downloads
dotnet-xscgen
A .NET Core CLI tool to generate XmlSerializer compatible C# classes from XML Schema files.
39
dotnet-xscgen
A .NET Core CLI tool to generate XmlSerializer compatible C# classes from XML Schema files.
33
dotnet-xscgen
A .NET Core CLI tool to generate XmlSerializer compatible C# classes from XML Schema files.
32
dotnet-xscgen
A .NET Core CLI tool to generate XmlSerializer compatible C# classes from XML Schema files.
27

Version Downloads Last updated
2.1.1183 3 04/22/2025
2.1.1182 6 04/03/2025
2.1.1181 8 03/26/2025
2.1.1179 11 02/18/2025
2.1.1178 11 02/05/2025
2.1.1174 16 12/15/2024
2.1.1173 14 12/04/2024
2.1.1167 15 11/16/2024
2.1.1166 17 11/08/2024
2.1.1164 18 10/24/2024
2.1.1162 20 09/05/2024
2.1.1160 23 07/19/2024
2.1.1158 18 07/04/2024
2.1.1156 27 07/04/2024
2.1.1155 26 07/01/2024
2.1.1144 23 06/02/2024
2.1.1142 27 05/18/2024
2.1.1141 29 05/19/2024
2.1.1130 19 04/22/2024
2.1.1127 23 04/17/2024
2.1.1107 25 02/24/2024
2.1.1094 30 02/24/2024
2.1.1092 23 02/24/2024
2.1.1057 29 01/07/2024
2.1.1051 23 01/07/2024
2.1.963 28 11/04/2023
2.1.959 34 11/04/2023
2.1.956 25 11/04/2023
2.1.954 24 10/19/2023
2.1.915 24 10/03/2023
2.1.910 28 09/29/2023
2.1.908 24 09/23/2023
2.0.894 23 10/03/2023
2.0.878 24 08/25/2023
2.0.874 25 08/27/2023
2.0.864 28 08/28/2023
2.0.863 23 08/27/2023
2.0.862-beta 33 08/28/2023
2.0.861 23 08/27/2023
2.0.859 27 08/28/2023
2.0.858 31 08/28/2023
2.0.810 33 08/28/2023
2.0.805 28 08/28/2023
2.0.792 34 08/27/2023
2.0.732 26 08/28/2023
2.0.719 29 08/28/2023
2.0.718 30 08/28/2023
2.0.717 25 08/27/2023
2.0.714 28 10/03/2023
2.0.711 34 08/28/2023
2.0.703 31 08/28/2023
2.0.692 22 08/27/2023
2.0.662 27 08/27/2023
2.0.660 31 08/27/2023
2.0.646 23 08/27/2023
2.0.642 33 08/28/2023
2.0.634 30 08/28/2023
2.0.633 32 08/28/2023
2.0.631 26 08/27/2023
2.0.630 25 08/28/2023
2.0.629 28 08/28/2023
2.0.627 28 08/28/2023
2.0.594 31 08/28/2023
2.0.579 30 08/28/2023
2.0.567 26 08/27/2023
2.0.566 29 08/28/2023
2.0.565 30 09/27/2023
2.0.560 25 08/27/2023
2.0.521 27 08/27/2023
2.0.517 32 08/27/2023
2.0.513 26 10/02/2023
2.0.512 28 10/19/2023
2.0.509 25 08/26/2023
2.0.502 27 08/28/2023
2.0.495 29 08/26/2023
2.0.494 34 08/28/2023
2.0.490 23 08/28/2023
2.0.488 26 08/26/2023
2.0.487 24 08/26/2023
2.0.486 25 08/27/2023
2.0.484 32 08/27/2023
2.0.479 30 08/27/2023
2.0.478 25 08/28/2023
2.0.460 25 08/28/2023
2.0.459 33 08/28/2023
2.0.444 32 08/27/2023
2.0.439 29 08/27/2023
2.0.436 23 08/27/2023
2.0.435 27 08/27/2023
2.0.434 26 08/27/2023
2.0.425 32 08/27/2023
2.0.424 22 08/28/2023
2.0.421 26 08/26/2023
2.0.419 28 08/28/2023
2.0.414 24 08/28/2023
2.0.412 23 08/28/2023
2.0.405 24 08/28/2023
2.0.395 25 08/28/2023
2.0.390 30 08/28/2023
2.0.373 24 08/28/2023
2.0.370 25 08/28/2023
2.0.365 23 08/28/2023
2.0.349 25 08/28/2023
2.0.347 19 10/25/2023
2.0.346 25 08/28/2023
2.0.345 34 08/27/2023
2.0.341 27 08/27/2023
2.0.340 25 08/27/2023
2.0.334 25 08/27/2023
2.0.322 25 08/28/2023
2.0.310 37 08/27/2023
2.0.301 28 09/23/2023
2.0.300 27 08/28/2023
2.0.296 24 08/28/2023
2.0.267 24 08/28/2023
2.0.261 30 08/26/2023
2.0.257 23 08/27/2023
2.0.254 24 08/28/2023
2.0.252 30 08/26/2023
2.0.241 33 08/28/2023
2.0.232 27 08/27/2023
2.0.230 25 08/27/2023
2.0.214 32 08/28/2023
2.0.212 29 08/28/2023
2.0.210 24 08/27/2023
2.0.209 27 08/27/2023
2.0.208 25 08/27/2023
2.0.206 30 08/28/2023
2.0.199 32 08/28/2023
2.0.197 26 08/28/2023
2.0.195 28 08/28/2023
2.0.187 24 08/27/2023
2.0.183 25 08/28/2023
2.0.180 28 08/27/2023
2.0.177 23 08/28/2023
2.0.174 25 08/26/2023
2.0.172 26 08/28/2023
2.0.160 25 08/28/2023
2.0.149 26 08/28/2023
2.0.147 28 08/28/2023
2.0.142 23 08/27/2023
2.0.140 32 08/28/2023
2.0.135 28 08/26/2023
2.0.133 25 08/27/2023
2.0.106 25 08/28/2023
2.0.105 24 08/27/2023
2.0.104 28 08/27/2023
2.0.102 24 08/28/2023
2.0.99 35 08/27/2023
2.0.97 33 08/26/2023
2.0.85-beta 27 08/28/2023
2.0.83-beta 33 08/27/2023
2.0.82-beta 27 08/27/2023
2.0.78-beta 32 08/28/2023
2.0.77-beta 25 08/28/2023
2.0.73-beta 33 08/28/2023
1.0.58 32 08/28/2023
1.0.50 24 08/27/2023
1.0.47 32 08/28/2023
1.0.45 36 08/28/2023
1.0.34 26 08/27/2023
1.0.32 28 08/27/2023
1.0.31 34 08/27/2023
1.0.28 31 08/26/2023
1.0.20 30 08/28/2023
1.0.15 23 08/28/2023
1.0.13 30 08/28/2023
1.0.12 28 08/28/2023
0.8.5767.29668-beta 23 08/26/2023
0.8.5750.32814-beta 33 08/27/2023
0.8.5718.30343-beta 21 08/27/2023
0.7.5703.30363-beta 26 08/27/2023
0.7.5668.20189-beta 21 08/28/2023
0.6.5640.32393-beta 24 08/28/2023
0.5.5540.23230-beta 31 08/27/2023
0.5.5540.22937-beta 21 08/28/2023
0.5.5514.24827-beta 23 08/27/2023
0.5.5514.24272-beta 25 08/27/2023
0.5.5514.24013-beta 20 10/27/2023
0.5.5501.32065-beta 27 08/27/2023
0.5.5494.29733-beta 27 08/27/2023
0.5.5493.32546-beta 24 08/28/2023
0.5.5493.25243-beta 32 08/26/2023
0.5.5493.24694-beta 27 08/27/2023
0.4.5463.24795-beta 21 10/06/2023
0.3.5436.30355-beta 24 08/27/2023
0.3.5235.30369-beta 24 08/27/2023
0.2.5145.38480-beta 26 08/27/2023
0.2.5143.24328 29 08/28/2023
0.1.5109.28584 29 08/28/2023
0.1.5101.32625 24 08/28/2023