จำเป็นต้องบันทึกคำขอ asp.net webapi 2 และเนื้อหาตอบกลับไปยังฐานข้อมูล


104

ฉันใช้ Microsoft Asp.net WebApi2 ที่โฮสต์บน IIS ฉันต้องการบันทึกเนื้อหาคำขอ (XML หรือ JSON) และเนื้อหาการตอบสนองสำหรับแต่ละโพสต์

ไม่มีอะไรพิเศษเกี่ยวกับโปรเจ็กต์นี้หรือผู้ควบคุมกำลังประมวลผลโพสต์ ฉันไม่สนใจที่จะใช้เฟรมเวิร์กการบันทึกเช่น nLog, elmah, log4net หรือคุณสมบัติการติดตามในตัวของเว็บ API เว้นแต่ว่าจำเป็นต้องทำเช่นนั้น

ฉันแค่อยากรู้ว่าจะใส่รหัสบันทึกของฉันไว้ที่ไหนและจะรับ JSON หรือ XML จริงจากคำขอและการตอบกลับขาเข้าและขาออกได้อย่างไร

วิธีการโพสต์คอนโทรลเลอร์ของฉัน:

public HttpResponseMessage Post([FromBody])Employee employee)
{
   if (ModelState.IsValid)
   {
      // insert employee into to the database
   }

}

คุณต้องการบันทึกคำขอ / การตอบกลับสำหรับการดำเนินการบางอย่างชุดหรือการกระทำทั้งหมดของคุณในคอนโทรลเลอร์เฉพาะหรือไม่?
LB2

สนใจเฉพาะการบันทึกโพสต์ (a) เวลาในการโพสต์ (b) เนื้อหาของ xml หรือ json ที่โพสต์ (c) การตอบสนอง (เนื้อหา xml หรือ json) พร้อมกับรหัสสถานะ Http
user2315985

เหตุผลที่ฉันถามคือแนะนำว่าจะนำโค้ดไปสู่การปฏิบัติโดยตรงหรือไม่หรือวิธีแก้ปัญหาทั่วไปสำหรับการกระทำทั้งหมด ดูคำตอบของฉันด้านล่าง
LB2

FYI ฉันลบ asp.net เนื่องจากไม่ได้ตอบคำถามนี้
Dalorzo

การสร้าง filer ไม่ใช่ตัวเลือก?
Prerak K

คำตอบ:


195

ฉันอยากจะแนะนำให้ใช้ไฟล์DelegatingHandler. จากนั้นคุณจะไม่ต้องกังวลเกี่ยวกับรหัสบันทึกใด ๆ ในคอนโทรลเลอร์ของคุณ

public class LogRequestAndResponseHandler : DelegatingHandler
{
    protected override async Task<HttpResponseMessage> SendAsync(
        HttpRequestMessage request, CancellationToken cancellationToken)
    {
        if (request.Content != null)
        {
            // log request body
            string requestBody = await request.Content.ReadAsStringAsync();
            Trace.WriteLine(requestBody);
        }
        // let other handlers process the request
        var result = await base.SendAsync(request, cancellationToken);

        if (result.Content != null)
        {
            // once response body is ready, log it
            var responseBody = await result.Content.ReadAsStringAsync();
            Trace.WriteLine(responseBody);
        }

        return result;
    }
}

เพียงแทนที่Trace.WriteLineด้วยรหัสบันทึกของคุณและลงทะเบียนตัวจัดการในWebApiConfigลักษณะนี้:

config.MessageHandlers.Add(new LogRequestAndResponseHandler());

นี่คือเอกสารไมโครซอฟท์เต็มรูปแบบสำหรับข้อความ Handlers


3
task.Result.ContentSystem.Net.Http.ObjectContentผลตอบแทน มีวิธีรับ Raw xml / json แทนหรือไม่?
พีซี.

4
@SoftwareFactor: ContinueWithและResultเป็น API ที่อันตราย มันจะดีกว่าถ้าใช้awaitแทนเช่นvar result = await base.SendAsync(request, cancellationToken); var resposeBody = await response.Content.ReadAsStringAsync(); Trace.WriteLine(responseBody); return response;
Stephen Cleary

9
นี่เป็นวิธีแก้ปัญหาที่ยอดเยี่ยมมาก แต่จะทำให้เกิดข้อผิดพลาดเมื่อการตอบสนองไม่มีเนื้อความ แต่นั่นง่ายพอที่จะตรวจสอบและแก้ไข :)
buddybubble

6
การโทรawait request.Content.ReadAsStringAsync();ไม่ส่งผลให้เกิดข้อผิดพลาดที่บอกว่าสตรีมคำขอถูกอ่านแล้วในสถานการณ์บางอย่างหรือไม่?
Gavin

6
หากตัวจัดการการมอบสิทธิ์อ่านเนื้อหาของคำขอจะไม่ทำให้ตัวจัดการเทอร์มินัลใช้งานไม่ได้จริง (เช่น mvc / webapi) หรือไม่
LB2

15

มีหลายวิธีในการจัดการการบันทึกคำร้อง / การตอบกลับโดยทั่วไปสำหรับการเรียกใช้เมธอด WebAPI ทุกวิธี:

  1. ActionFilterAttribute: สามารถเขียนแบบกำหนดเองActionFilterAttributeและตกแต่งวิธีการควบคุม / การดำเนินการเพื่อเปิดใช้งานการบันทึก

    Con: คุณต้องตกแต่งคอนโทรลเลอร์ / วิธีการทั้งหมด (คุณยังสามารถทำได้บนตัวควบคุมพื้นฐาน แต่ก็ยังไม่ได้ระบุถึงข้อกังวลในการตัดข้าม

  2. แทนที่BaseControllerและจัดการการบันทึกที่นั่น

    Con: เราคาดหวัง / บังคับให้คอนโทรลเลอร์สืบทอดจากตัวควบคุมพื้นฐานที่กำหนดเอง

  3. การใช้DelegatingHandler.

    ข้อได้เปรียบ: เราไม่ได้สัมผัสตัวควบคุม / วิธีการที่นี่ด้วยวิธีนี้ ตัวจัดการการมอบหมายจะอยู่ในการแยกและจัดการการบันทึกคำขอ / การตอบกลับอย่างสง่างาม

สำหรับบทความเชิงลึกเพิ่มเติมโปรดดูนี้http://weblogs.asp.net/fredriknormen/log-message-request-and-response-in-asp-net-webapi


คุณสามารถกำหนดตัวกรองการดำเนินการดังต่อไปนี้: WebApiConfig คลาสคงที่สาธารณะ {public static void Register (HttpConfiguration config) {// Web API configuration and services config.Filters.Add (new MyFilter ()) // Web API route config.MapHttpAttributeRoutes (); config.Routes.MapHttpRoute (ชื่อ: "DefaultApi", routeTemplate: "api / {controller} / {id}", ค่าเริ่มต้น: ใหม่ {id = RouteParameter.Optional}); }}
Mika Karjunen

11

หนึ่งในตัวเลือกที่คุณมีคือใช้การสร้างตัวกรองการดำเนินการและตกแต่ง WebApiController / ApiMethod ของคุณด้วย

กรองแอตทริบิวต์

public class MyFilterAttribute : System.Web.Http.Filters.ActionFilterAttribute
    {
        public override void OnActionExecuting(HttpActionContext actionContext)
        {
            if (actionContext.Request.Method == HttpMethod.Post)
            {
                var postData = actionContext.ActionArguments;
                //do logging here
            }
        }
    }

ตัวควบคุม WebApi

[MyFilterAttribute]
public class ValuesController : ApiController{..}

หรือ

[MyFilterAttribute]
public void Post([FromBody]string value){..}

หวังว่านี่จะช่วยได้


ฉันชอบวิธีนี้ แต่เพื่อให้ได้การตอบสนองฉันต้องแทนที่ OnActionExecuted แทน ปัญหาคือคำขอ ณ จุดนั้นได้ถูกแปลงเป็น POCO ของฉันแล้วแทนที่จะเป็น xml หรือ json ความคิดใด ๆ ?
user2315985

แต่เดิมฉันหมายถึงบันทึกข้อมูลใน OnActionExecuting จากนั้นปล่อยให้โพสต์ทำงานได้อย่างง่ายดาย สิ่งที่ฉันเข้าใจจากคำถามของคุณคือคุณต้องการบันทึกข้อมูลสำหรับแต่ละโพสต์ที่ทำเสร็จ
Prerak K

3
ฉันต้องการบันทึกทั้งคำขอและข้อมูลการตอบกลับทุกครั้งที่มีคนโพสต์
user2315985

2
คุณสามารถใช้ OnActionExecuted และลอง "(actionExecutedContext.ActionContext.Response.Content เป็น ObjectContent) .Value ToString ()" เพื่อรับการตอบสนองและบันทึก
Prerak K

ฉันจะรับคำขอจากภายใน OnActionExecuted ได้อย่างไร
user2315985

3

การเข้าถึงข้อความขอเป็นเรื่องง่าย คลาสฐานApiControllerของคุณมี.Requestคุณสมบัติซึ่งตามชื่อที่แนะนำมีคำขอในรูปแบบแยกวิเคราะห์ คุณเพียงตรวจสอบสิ่งที่คุณต้องการบันทึกและส่งต่อไปยังสถานที่บันทึกของคุณไม่ว่าจะเป็นแบบใดก็ตาม รหัสนี้คุณสามารถใส่ไว้ในจุดเริ่มต้นของการกระทำของคุณหากคุณต้องการทำเพียงหนึ่งหรือหยิบมือเดียว

หากคุณจำเป็นต้องทำในทุกการกระทำ (ทั้งหมดมีความหมายมากกว่ากำมือที่จัดการได้) สิ่งที่คุณทำได้คือลบล้าง.ExecuteAsyncวิธีการเพื่อจับทุกการเรียกร้องการกระทำสำหรับคอนโทรลเลอร์ของคุณ

public override Task<HttpResponseMessage> ExecuteAsync(
    HttpControllerContext controllerContext,
    CancellationToken cancellationToken
)
{
    // Do logging here using controllerContext.Request
    return base.ExecuteAsync(controllerContext, cancellationToken);
}

ฉันกำลังทำสิ่งนี้และฉันยังไม่ได้เปรียบเทียบมันแค่สัญชาตญาณของฉันบอกฉันว่ามันอาจจะช้ามาก?
Marcus

ทำไมคิดว่าจะช้า ExecuteAsyncคือสิ่งที่เรียกโดยเฟรมเวิร์กและการใช้คลาสคอนโทรลเลอร์พื้นฐานคือสิ่งที่ทำให้การดำเนินการดำเนินการได้จริง นี่เป็นเพียงการเรียกเข้าสู่การบันทึกของคุณซึ่งเป็นส่วนหนึ่งของการดำเนินการที่เกิดขึ้นแล้ว พวกเขาลงโทษเฉพาะที่นี่คือเวลาในการตัดไม้จริง
LB2

ไม่ฉันหมายถึง 'ช้ามาก' เหมือนกับการบันทึกทุกคำขอ
Marcus

2
นั่นเป็นคำถามเกี่ยวกับข้อกำหนดและนั่นคือข้อกำหนดที่ระบุโดย OP เป็นคำถามเกี่ยวกับปริมาณที่ไซต์จัดการประสิทธิภาพของสิ่งอำนวยความสะดวกในการบันทึก ฯลฯ ซึ่งอยู่นอกเหนือจากโพสต์ OPs
LB2

0

ดูเหมือนว่าจะเป็นกระทู้เก่า แต่กำลังแบ่งปันวิธีแก้ปัญหาอื่น

คุณสามารถเพิ่มวิธีนี้ในไฟล์ global.asax ของคุณซึ่งจะถูกทริกเกอร์ทุกครั้งหลังจากที่คำขอ HTTP สิ้นสุดลง

void Application_EndRequest(Object Sender, EventArgs e)
    {
        var request = (Sender as HttpApplication).Request;
        var response = (Sender as HttpApplication).Response;

        if (request.HttpMethod == "POST" || request.HttpMethod == "PUT")
        {


            byte[] bytes = request.BinaryRead(request.TotalBytes);
            string body = Encoding.UTF7.GetString(bytes);
            if (!String.IsNullOrEmpty(body))
            {


                // Do your logic here (Save in DB, Log in IIS etc.)
            }
        }
    }

0

นี่เป็นหัวข้อเก่าจริงๆ แต่ฉันใช้เวลามาก (ค้นหาทางอินเทอร์เน็ต) เพื่อทำสิ่งเหล่านี้ดังนั้นฉันจะโพสต์วิธีแก้ปัญหาที่นี่

แนวคิด

  1. แทนที่ ExecuteAsync ของเมธอด APicontroller สำหรับการติดตามคำขอขาเข้าในโซลูชันของฉันฉันสร้าง Base_ApiController เป็นพาเรนต์ของตัวควบคุม API ของโครงการของฉัน
  2. ใช้ System.Web.Http.Filters.ActionFilterAttribute เพื่อติดตามการตอบสนองขาออกของตัวควบคุม api
  3. *** (เพิ่มเติม) *** ใช้ System.Web.Http.Filters.ExceptionFilterAttribute เพื่อบันทึกเมื่อมีข้อยกเว้นเกิดขึ้น

1. MyController.cs

    [APIExceptionFilter]  // use 3.
    [APIActionFilter]     // use 2.
    public class Base_APIController : ApiController
    {
        public   bool  IsLogInbound
        {
            get
            { return   ConfigurationManager.AppSettings["LogInboundRequest"] =="Y"? true:false ;     }
        }
        /// <summary>
        /// for logging exception
        /// </summary>
        /// <param name="controllerContext"></param>
        /// <param name="cancellationToken"></param>
        /// <returns></returns>
        public override Task<HttpResponseMessage> ExecuteAsync(
         HttpControllerContext controllerContext,
         CancellationToken cancellationToken
         )
        {
            // Do logging here using controllerContext.Request
            // I don't know why calling the code below make content not null Kanit P.
            var content = controllerContext.Request.Content.ReadAsStringAsync().Result.ToString(); // keep request json content
             // Do your own logging!
            if (IsLogInbound)
            {
                try
                {
                    ErrLog.Insert(ErrLog.type.InboundRequest, controllerContext.Request,
                         controllerContext.Request.RequestUri.AbsoluteUri
                         , content);
                }
                catch (Exception e) { }
            }

            // will not log err when go to wrong controller's action (error here but not go to APIExceptionFilter)
            var t = base.ExecuteAsync(controllerContext, cancellationToken);
            if (!t.Result.IsSuccessStatusCode)
            { 
            }
            return t;

        }

2. APIActionFilter.cs

    public class APIActionFilter : System.Web.Http.Filters.ActionFilterAttribute
    {
        public bool LogOutboundRequest
        {
            get
            { return ConfigurationManager.AppSettings["LogInboundRequest"] == "Y" ? true : false; }
        }

        public override void OnActionExecuted(HttpActionExecutedContext actionExecutedContext)
        {
            try {

                var returndata = actionExecutedContext.Response.Content.ReadAsStringAsync().Result.ToString(); 
             //keep Json response content
             // Do your own logging!
                if (LogOutboundRequest)
                {
                    ErrLog.Insert(ErrLog.type.OutboundResponse, actionExecutedContext.Response.Headers,
                       actionExecutedContext.ActionContext.ControllerContext.ControllerDescriptor.ControllerName
                      + "/"
                      + actionExecutedContext.ActionContext.ActionDescriptor.ActionName
                      , returndata );
                }
            } catch (Exception e) {

            }
     

        } 
    }
}

3. APIExceptionFilter.cs

    public class APIExceptionFilter : ExceptionFilterAttribute
    {
    public bool IsLogErr
    {
        get
        { return ConfigurationManager.AppSettings["LogExceptionRequest"] == "Y" ? true : false; }
    }


    public override void OnException(HttpActionExecutedContext context)
    {
        try
        { 
            //Do your own logging!
            if (IsLogErr)
            {
                ErrLog.Insert(ErrLog.type.APIFilterException, context.Request,
                    context.ActionContext.ControllerContext.ControllerDescriptor.ControllerName
                    + "/"
                    + context.ActionContext.ActionDescriptor.ActionName
                    , context.Exception.ToString() + context.Exception.StackTrace);
            }
        }catch(Exception e){

        }

        if (context.Exception is NotImplementedException)
        {
            context.Response = new HttpResponseMessage(HttpStatusCode.NotImplemented);
        }
        else {
            context.Response = new HttpResponseMessage(HttpStatusCode.InternalServerError);

        }
    }
}
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.