Как извлечь внутреннее исключение из исключения soap в ASP.NET?

У меня есть простая операция веб-службы, подобная этой:

    [WebMethod]
    public string HelloWorld()
    {
        throw new Exception("HelloWorldException");
        return "Hello World";
    }

И затем у меня есть клиентское приложение, которое использует веб-службу, а затем вызывает операцию. Очевидно, что это вызовет исключение: -)

    try
    {
        hwservicens.Service1 service1 = new hwservicens.Service1();
        service1.HelloWorld();
    }
    catch(Exception e)
    {
        Console.WriteLine(e.ToString());
    }

В моем блоке catch я хотел бы извлечь сообщение о фактическом исключении, чтобы использовать его в моем коде. Исключением является SoapException, это нормально, но свойство Message похоже на это...

System.Web.Services.Protocols.SoapException: Server was unable to process request. ---> System.Exception: HelloWorldException
   at WebService1.Service1.HelloWorld() in C:\svnroot\Vordur\WebService1\Service1.asmx.cs:line 27
   --- End of inner exception stack trace ---

... и InnerException - null.

Что бы я хотел сделать, это извлечь свойство Message в тексте InnerException (текст HelloWorldException в моем примере), может ли кто-нибудь помочь с этим? Если вы можете избежать этого, не предлагайте разбора свойства Message SoapException.

Ответы

Ответ 1

К сожалению, я не думаю, что это возможно.

Исключение, которое вы поднимаете в коде веб-службы, кодируется в мыльную ошибку, которая затем передается в виде строки обратно к вашему коду клиента.

То, что вы видите в сообщении SoapException, - это просто текст из ошибки Soap, который не преобразуется обратно в исключение, а просто сохраняется как текст.

Если вы хотите вернуть полезную информацию в условиях ошибки, я рекомендую вернуть пользовательский класс из вашего веб-сервиса, который может иметь свойство "Ошибка", которое содержит вашу информацию.

[WebMethod]
public ResponseClass HelloWorld()
{
  ResponseClass c = new ResponseClass();
  try 
  {
    throw new Exception("Exception Text");
    // The following would be returned on a success
    c.WasError = false;
    c.ReturnValue = "Hello World";
  }
  catch(Exception e)
  {
    c.WasError = true;
    c.ErrorMessage = e.Message;
    return c;
  }
}

Ответ 2

Это возможно!

Пример служебной операции:

try
{
   // do something good for humanity
}
catch (Exception e)
{
   throw new SoapException(e.InnerException.Message,
                           SoapException.ServerFaultCode);
}

Клиент, потребляющий услугу:

try
{
   // save humanity
}
catch (Exception e)
{
   Console.WriteLine(e.Message);    
}

Только одно: вам нужно установить режим customErrors = "RemoteOnly" или 'On' в ваш web.config (проекта службы).

кредиты об обнаружении customErrors - http://forums.asp.net/t/236665.aspx/1

Ответ 3

Я столкнулся с чем-то подобным чуть-чуть, и написал об этом в блоге. Я не уверен, что это точно применимо, но может быть. Код достаточно прост, если вы поймете, что вам нужно пройти через объект MessageFault. В моем случае я знал, что в детали содержится GUID, который я мог бы использовать для повторного запроса службы SOAP для деталей. Код выглядит следующим образом:

catch (FaultException soapEx)
{
    MessageFault mf = soapEx.CreateMessageFault();
    if (mf.HasDetail)
    {
        XmlDictionaryReader reader = mf.GetReaderAtDetailContents();
        Guid g = reader.ReadContentAsGuid();
    }
}