私たちのためにインターネットを検索した後、私と私のチームは異なるアプローチで、nillable = "true"を取り除くことに決めました。私たちがしたことは以下の通りでした:
- アプリケーションの起動時に特別な
HttpModule
が登録されました。
- このモジュールは、BeginRequestイベントへのハンドラを登録しました。
- WSDLまたはXSDが要求された場合(クエリ文字列を調べることによってチェックされている場合)、フックされたハンドラはデフォルトの
Response.Filter
をデコレータ(ストリーム)に置き換えます。
- このデコレータは、文書の終わりに達するまで、書かれたデータをバッファします。
- すべてのデータが書き込まれると、デコレータはそのデータに基づいて
XDocument
を構築し、そのドキュメントを参照して(ドキュメントの追加など)、nillable = "false"に設定します。
これはすべてかなりかわいいです。
Response.Filter
を使用するアイデアは、hereです。ここで
は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);
}
あなたがあなた自身のWsdlExportExtensionを記述する必要があります。 サンプルについては、この回答を参照してください:http://stackoverflow.com/questions/10602749/setting-nillable-false-with-wcf/23366064#23366064 – Vizu