18

aspx ページによってレンダリングされた HTML (私の例ではログ用) をキャプチャするにはどうすればよいですか?

サイトのレイアウトが台無しになるため、Response.Write を使用してページに書き戻す必要はありません。

Response.OutputStream または Response.Output のストリームを使用すると、ArgumentException が発生します ({System.ArgumentException: ストリームは読み取れませんでした。)

4

3 に答える 3

25

良い質問です。あなたが説明していることを行うために HttpModule を作成できるかどうかを試してみなければなりませんでした。

応答ストリームから読み取ろうとしてもうまくいきませんでしたが、ResponseFilter を使用することでコンテンツを取得できました。

次のコードはかなりうまく機能しているようで、このコードをベースとして使用できるのではないかと考えました。しかし、これは私がすぐにまとめたものであることを覚えておいてください。決してテストされていません. したがって、適切なレビュー/テストなどを行わずに、本番環境で使用しないでください。でも、気軽にコメントしてね(;_;)

public class ResponseLoggerModule : IHttpModule
{
    private class ResponseCaptureStream : Stream
    {
        private readonly Stream _streamToCapture;
        private readonly Encoding _responseEncoding;

        private string _streamContent;
        public string StreamContent
        {
            get { return _streamContent; }
            private set
            {
                _streamContent = value;
            }
        }

        public ResponseCaptureStream(Stream streamToCapture, Encoding responseEncoding)
        {
            _responseEncoding = responseEncoding;
            _streamToCapture = streamToCapture;

        }

        public override bool CanRead
        {
            get { return _streamToCapture.CanRead; }
        }

        public override bool CanSeek
        {
            get { return _streamToCapture.CanSeek; }
        }

        public override bool CanWrite
        {
            get { return _streamToCapture.CanWrite; }
        }

        public override void Flush()
        {
            _streamToCapture.Flush();
        }

        public override long Length
        {
            get { return _streamToCapture.Length; }
        }

        public override long Position
        {
            get
            {
                return _streamToCapture.Position;
            }
            set
            {
                _streamToCapture.Position = value;
            }
        }

        public override int Read(byte[] buffer, int offset, int count)
        {
            return _streamToCapture.Read(buffer, offset, count);
        }

        public override long Seek(long offset, SeekOrigin origin)
        {
            return _streamToCapture.Seek(offset, origin);
        }

        public override void SetLength(long value)
        {
            _streamToCapture.SetLength(value);
        }

        public override void Write(byte[] buffer, int offset, int count)
        {
            _streamContent += _responseEncoding.GetString(buffer);
            _streamToCapture.Write(buffer, offset, count);
        }

        public override void Close()
        {
            _streamToCapture.Close();
            base.Close();
        }
    }

    #region IHttpModule Members

    private HttpApplication _context;
    public void Dispose()
    {

    }

    public void Init(HttpApplication context)
    {
        _context = context;

        context.PreRequestHandlerExecute += new EventHandler(context_PreRequestHandlerExecute);
        context.PreSendRequestContent += new EventHandler(context_PreSendRequestContent);
    }

    void context_PreRequestHandlerExecute(object sender, EventArgs e)
    {
        _context.Response.Filter = new ResponseCaptureStream(_context.Response.Filter, _context.Response.ContentEncoding);
    }

    void context_PreSendRequestContent(object sender, EventArgs e)
    {
        ResponseCaptureStream filter = _context.Response.Filter as ResponseCaptureStream;

        if (filter != null)
        {
            string responseText = filter.StreamContent;

            // Logging logic here
        }
    }

    #endregion
}
于 2008-12-22T16:21:59.853 に答える
4

多くのロード テスターでは、生成された HTTP 応答をログに記録できますが、ASP.NET では非常に大きなログ ファイルになる可能性があることに注意してください。

編集:Tom JelenのコードによるResponse.Filterは、この種の見落としを与えるように設計されており、それ以外の場合はResponse.Outputstreamは判読できません。

編集 2: HTTPModule ではなくページの場合

public class ObserverStream : Stream
{
  private byte[] buffer = null;
  private Stream observed = null;

  public ObserverStream (Stream s)
  {
    this.observed = s;
  }

  /* important method to extend #1 : capturing the data */
  public override void Write(byte[] buffer, int offset, int count)
  {
    this.observed.Write(buffer, offset, count);
    this.buffer = buffer; //captured!
  }

  /* important method to extend #2 : doing something with the data */
  public override void Close()
  {
    //this.buffer available for logging here!
    this.observed.Close();
  }

  /* override all the other Stream methods/props with this.observed.method() */

  //...

}

そしてあなたの Page_Load で(またはとにかくあなたの応答が書かれる前に)

Response.Filter = new ObserverStream(Response.Filter);
于 2008-12-22T15:23:16.493 に答える
1

独自のサーバーに対してサーバー側の XMLHTTP 要求を行う 1 つの方法。結果を取得し、ファイルまたは DB に保存します。

または、クライアントで AJAX を使用し、結果を取得してサーバーに POST することもできます。

于 2008-12-22T15:35:11.353 に答える