После поиска нашего интернета я и моя команда решили пойти другим путем, чтобы избавиться от nillable = "true".Мы сделали следующее:
- Зарегистрировал специальный
HttpModule
во время запуска приложения. - Этот модуль зарегистрировал обработчик для события BeginRequest.
- Весли запрашиваются WSDL или XSD (проверяется путем просмотра строки запроса), обработчик перехвата заменит значение по умолчанию
Response.Filter
декоратором (потоком). - Этот декоратор буферизует записанные данные до концадокумента было достигнуто.
- Когда все данные записаны, декоратор создает
XDocument
на основе этих данных и просматривает этот документ, чтобы установить (среди прочего, например, добавление документации) nillable = "false".
Это прекрасно работает.
Идея использования Response.Filter
была найдена здесь .
Вот модуль HttpModule:
public class WsdlInterceptionHttpModule : IHttpModule
{
public void Init(HttpApplication application)
{
application.BeginRequest += (sender, e) =>
{
var context = application.Context;
if (IsRequestForWsdl(context.Request))
{
context.Response.Filter =
new WsdlAnnotationsFilterDecorator(context.Response.Filter);
}
};
}
private static bool IsRequestForWsdl(HttpRequest request) { ... }
}
Вот WsdlAnnotationsFilterDecorator
:
public class WsdlAnnotationsFilterDecorator : Stream
{
private const string DefinitionsEndOfFileMarker = "</wsdl:definitions>";
private const string SchemaEndOfFileMarker = "</xs:schema>";
private readonly Stream inputStream;
private readonly StringBuilder responseXml = new StringBuilder();
private bool firstWrite = true;
private string endOfFileMarker = DefinitionsEndOfFileMarker;
public WsdlAnnotationsFilterDecorator(Stream inputStream)
{
this.inputStream = inputStream;
this.responseXml = new StringBuilder();
}
public override bool CanRead { get { return true; } }
public override bool CanSeek { get { return true; } }
public override bool CanWrite { get { return true; } }
public override long Length { get { return 0; } }
public override long Position { get; set; }
public override void Close()
{
inputStream.Close();
}
public override void Flush()
{
inputStream.Flush();
}
public override long Seek(long offset, SeekOrigin origin)
{
return inputStream.Seek(offset, origin);
}
public override void SetLength(long length)
{
inputStream.SetLength(length);
}
public override int Read(byte[] buffer, int offset, int count)
{
return inputStream.Read(buffer, offset, count);
}
public override void Write(byte[] buffer, int offset, int count)
{
string valueToWrite = UTF8Encoding.UTF8.GetString(buffer, offset, count);
SetEndOfFileMarker(valueToWrite);
if (!valueToWrite.EndsWith(this.endOfFileMarker))
{
responseXml.Append(valueToWrite);
}
else
{
responseXml.Append(valueToWrite);
string finalXml = responseXml.ToString();
finalXml = WsdlAnnotator.Annotate(finalXml);
byte[] data = UTF8Encoding.UTF8.GetBytes(finalXml);
inputStream.Write(data, 0, data.Length);
}
}
private void SetEndOfFileMarker(string valueToWrite)
{
if (firstWrite)
{
int definitionTagIndex = valueToWrite.IndexOf("<wsdl:definitions");
int schemaTagIndex = valueToWrite.IndexOf("<xs:schema");
if (definitionTagIndex > -1 || schemaTagIndex > -1)
{
firstWrite = false;
if (definitionTagIndex > -1 && schemaTagIndex > -1)
{
endOfFileMarker =
definitionTagIndex < schemaTagIndex
? DefinitionsEndOfFileMarker : SchemaEndOfFileMarker;
}
else if (definitionTagIndex > -1)
{
endOfFileMarker = DefinitionsEndOfFileMarker;
}
else if (schemaTagIndex > -1)
{
endOfFileMarker = SchemaEndOfFileMarker;
}
}
}
}
}
* WsdlAnnotator
- это место, где происходит вся магия:
internal static class WsdlAnnotator
{
internal static string Annotate(string xml)
{
XDocument document = XDocument.Parse(xml);
try
{
// Your magic here.
}
catch (Exception ex)
{
throw new InvalidOperationException(
ex.Message + " Document: " + document.ToString(), ex);
}
return document.ToString(SaveOptions.None);
}