Как отслеживать запросы ScriptService WebService? - PullRequest
7 голосов
/ 20 июня 2009

У меня есть SoapExtension, которое предназначено для регистрации всех запросов и ответов SOAP. Он прекрасно работает для вызовов из приложения, использующего MS Soap Toolkit (OnBase Workflow). Но это не работает для вызовов, сделанных $ .ajax () на html-странице. Вот пример:

$.ajax({
    type: "POST",
    url: url,
    data: data,
    contentType: "application/json; charset=utf-8",
    dataType: "json"
});

Он вызывает веб-сервис ASP.NET 3.5, помеченный атрибутами WebService и ScriptService:

[WebService(Namespace = XmlSerializationService.DefaultNamespace)]
[ScriptService]
[WebServiceBinding(ConformsTo = WsiProfiles.BasicProfile1_1)]
public class DepartmentAssigneeService : WebService
{
    private readonly DepartmentAssigneeController _controller = new DepartmentAssigneeController();

    /// <summary>
    /// Fetches the role items.
    /// </summary>
    /// <returns></returns>
    [WebMethod]
    [SoapLog]
    public ListItem[] FetchDepartmentItems()
    {
        return CreateListItems(_controller.FetchDepartments());
    }
}

А вот основные сведения об атрибутах SoapExtension и SoapExtensionAttribute:

public class LoggingSoapExtension : SoapExtension, IDisposable { /*...*/ }

[AttributeUsage(AttributeTargets.Method)]
public sealed class SoapLogAttribute : SoapExtensionAttribute { /*...*/ }

Я что-то упустил, что позволило бы LoggingSoapExtension выполнять запросы $ .ajax ()?

Обновление

@ Крис Брандсма

Возможно, это связано с тем, что вы запрашиваете результаты Json вместо XML через ваш веб-сервис (dataType: "json"). Таким образом, атрибут ScriptService активируется, но вы не отправляете SOAP-сообщения.

Это отвечает, почему SoapExtension не работает. Есть предложения по отслеживанию с помощью ScriptService? Единственное, что приходит на ум, - это базовый класс ScriptService, который предоставляет метод для регистрации запроса. Но тогда мне придется вызывать этот метод в каждом WebMethod в каждом ScriptService WebService (у меня их немало). Я хотел бы использовать что-то простое и понятное, как атрибут SoapExtension, если это возможно.

Ответы [ 3 ]

2 голосов
/ 23 июня 2009

Я нашел решение. Используя IHttpModule, я могу регистрировать запросы от чего угодно (SOAP, JSON, формы и т. Д.). В приведенной ниже реализации я решил регистрировать все запросы .asmx и .ashx. Это заменяет LoggingSoapExtension из вопроса.

public class ServiceLogModule : IHttpModule
{
    private HttpApplication _application;
    private bool _isWebService;
    private int _requestId;
    private string _actionUrl;

    #region IHttpModule Members

    public void Dispose()
    {
    }

    public void Init(HttpApplication context)
    {
        _application = context;
        _application.BeginRequest += ContextBeginRequest;
        _application.PreRequestHandlerExecute += ContextPreRequestHandlerExecute;
        _application.PreSendRequestContent += ContextPreSendRequestContent;
    }

    #endregion

    private void ContextPreRequestHandlerExecute(object sender, EventArgs e)
    {
        _application.Response.Filter = new CapturedStream(_application.Response.Filter,
                                                          _application.Response.ContentEncoding);
    }

    private void ContextBeginRequest(object sender, EventArgs e)
    {
        string ext = VirtualPathUtility.GetExtension(_application.Request.FilePath).ToLower();
        _isWebService = ext == ".asmx" || ext == ".ashx";

        if (_isWebService)
        {
            ITraceLog traceLog = TraceLogFactory.Create();
            _actionUrl = _application.Request.Url.PathAndQuery;

            StreamReader reader = new StreamReader(_application.Request.InputStream);
            string message = reader.ReadToEnd();
            _application.Request.InputStream.Position = 0;

            _requestId = traceLog.LogRequest(_actionUrl, message);
        }
    }

    private void ContextPreSendRequestContent(object sender, EventArgs e)
    {
        if (_isWebService)
        {
            CapturedStream stream = _application.Response.Filter as CapturedStream;
            if (stream != null)
            {
                ITraceLog traceLog = TraceLogFactory.Create();
                traceLog.LogResponse(_actionUrl, stream.StreamContent, _requestId);
            }
        }
    }
}

Я сильно позаимствовал у Захват HTML, сгенерированный из ASP.NET .

1 голос
/ 20 июня 2009

Это может быть связано с тем, что вы запрашиваете результаты Json вместо XML через веб-сервис (dataType: "json"). Таким образом, атрибут ScriptService активируется, но вы не отправляете SOAP-сообщения.

Вы можете изменить dataType на xml и посмотреть, работает ли он.

http://docs.jquery.com/Ajax/jQuery.ajax#options

Кроме того, другой вариант для регистрации будет Log4Net. Это может быть гораздо более универсальным для вас.

0 голосов
/ 22 июня 2009

Fiddler (в первую очередь для IE, но теперь и для Firefox) или Firebug (для Firefox) являются бесценными инструментами для просмотра ваших запросов и ответов на стороне клиента.

...