Pergunta

Eu só comecei a usar ELMAH e sou um fã. Minha equipe suporta um grande número de aplicações web e eu estou particularmente animado que ELMAH nos permite salvar exceções de cada aplicação para a mesma tabela de banco de dados do MS SQL.

Também apoiamos alguns consola, aplicações DLL e desktop. É possível usar o ELMAH DLL para registrar exceções nesses aplicativos para que mesmo local?

Foi útil?

Solução

Nós temos exatamente a mesma situação aqui. Correndo ELMAH para todas as nossas aplicações web. Alguns deles têm programadores baseado em console.

Depois de fazer algumas escavações através do código-fonte, o seguinte código parece funcionar:

            ErrorLog errorLog = ErrorLog.GetDefault(null);
            errorLog.ApplicationName = "/LM/W3SVC/1/ROOT/AppName";
            errorLog.Log(new Error(ex));

O único verdadeiro problema com o acima exposto é que você precisa para manter a algum lugar nome do aplicativo na sua configuração para ser capaz de ver as entradas sobre o espectador elmah.axd.

Assim, em nosso código de manipulação de erro genérico que fazemos:

        if (HttpContext.Current != null)
            ErrorSignal.FromCurrentContext().Raise(ex);
        else
        {
            ErrorLog errorLog = ErrorLog.GetDefault(null);
            errorLog.ApplicationName = ErrorHandling.Application;
            errorLog.Log(new Error(ex));
        }

Outras dicas

Precisávamos a capacidade de log de um aplicativo console e um serviço de janelas, além de nosso site ASP.NET. Eu usei a resposta (ErrorLog.GetDefault(null);) que funcionou bem até que eu precisava para enviar e-mail também.

Então, aqui está a minha solução. Ele lida com o registro, e-mail, tweet e filtragem (tanto no arquivo de configuração e no código). Eu também envolveu a chamada principal como uma extensão de exceção para que ele possa ser chamado como: catch(Exception ex) { ex.LogToElmah(); }

Para filtro no código, ligar o evento .Filtering correspondente: ElmahExtension.ErrorLog.Filtering += new ExceptionFilterEventHandler(ErrorLog_Filtering);

Código:

using System;
using System.Web;
using Elmah;
namespace System
{
    public static class ElmahExtension
    {
        public static void LogToElmah(this Exception ex)
        {
            if (HttpContext.Current != null)
            {
                ErrorSignal.FromCurrentContext().Raise(ex);
            }
            else
            {
                if (httpApplication == null) InitNoContext();
                ErrorSignal.Get(httpApplication).Raise(ex);
            }
        }

            private static HttpApplication httpApplication = null;
            private static ErrorFilterConsole errorFilter = new ErrorFilterConsole();

            public static ErrorMailModule ErrorEmail = new ErrorMailModule();
            public static ErrorLogModule ErrorLog = new ErrorLogModule();
            public static ErrorTweetModule ErrorTweet = new ErrorTweetModule();

            private static void InitNoContext()
            {
                httpApplication = new HttpApplication();
                errorFilter.Init(httpApplication);

                (ErrorEmail as IHttpModule).Init(httpApplication);
                errorFilter.HookFiltering(ErrorEmail);

                (ErrorLog as IHttpModule).Init(httpApplication);
                errorFilter.HookFiltering(ErrorLog);                

                (ErrorTweet as IHttpModule).Init(httpApplication);
                errorFilter.HookFiltering(ErrorTweet);
            }

            private class ErrorFilterConsole : ErrorFilterModule
            {
                public void HookFiltering(IExceptionFiltering module)
                {
                    module.Filtering += new ExceptionFilterEventHandler(base.OnErrorModuleFiltering);
                }
            }
    }
}

Além disso, você vai precisar adicionar uma referência para o System.Web.dll em seu projeto para que isso funcione.

Editar : De acordo com os comentários, este código irá enviar e-mails apenas se o seu arquivo de configuração tem <errorMail async="false"/>. Consulte este trecho de código você deve querer manter <errorMail async="true"/> em seu arquivo de configuração (para ser usado quando HttpContext.Current está disponível ).

Se você quiser apenas para e-mail o registro sem http você pode fazer assim:

    public class MyElmahMail: ErrorMailModule
    {
        public MyElmahMail()
        {
//this basically just gets config from errorMail  (app.config)
            base.OnInit(new HttpApplication());
        }
        public void Log(Error error)
        {
//just send the email pls
            base.ReportError(error);
        }
    }

//to call it
var mail = new MyElmahMail();
mail.Log(new Error(new NullReferenceException()));//whatever exception u want to log

E em termos de app.config

//Under configSections
    <sectionGroup name="elmah">
      <section name="security" requirePermission="false" type="Elmah.SecuritySectionHandler, Elmah" />
      <section name="errorLog" requirePermission="false" type="Elmah.ErrorLogSectionHandler, Elmah" />
      <section name="errorMail" requirePermission="false" type="Elmah.ErrorMailSectionHandler, Elmah" />
      <section name="errorFilter" requirePermission="false" type="Elmah.ErrorFilterSectionHandler, Elmah" />
    </sectionGroup>

E

  <elmah>
    <errorLog type="Elmah.XmlFileErrorLog, Elmah" logPath="C:\Elmah.Error" applicationName="MyAwesomeApp" />
    <errorMail from="from@email.com" to="to@email.com" />
  </elmah>

e SMTP configurações de sua escolha.

Tudo feito. : -)

Editar : Este CAN ser feito - Veja este resposta.


Eu tenho certeza que você não pode fazer isso. Vou tentar cavar o material relevante.

http: // grupos .google.com / grupo / elmah / browse_thread / thread / f214c4f782dc2bf4 / d96fe43b60765f0c? lnk = GST & q = aplicativo # d96fe43b60765f0c

Assim, desde que eu posso encontrar pesquisando o grupo Google é que ele não é possível ... Desde ELMAH funciona fora de HttpHandlers (uma construção asp.net) só é ASP.NET.

Com isso dito, existem maneiras que você pode utilizá-lo em um aplicativo de console. ELMAH fornece um método para erros aumento, de modo que você poderia envolver ELMAH em seu tratamento de exceção e, em seguida, sinalizar um erro via:

ErrorSignal.FromCurrentContext().Raise(new NotSupportedException());

Isto significaria envolvendo seu aplicativo inteiro em um manipulador de exceção e de sinalização. Pode levar alguns ajustes para obtê-lo para baixo, mas eu acho que é totalmente possível.

No caso de você precisar dele, este é o link para o ELMAH repositório de código .

Para aqueles que a resposta de necessidade Brian Possibilidade portado para VB.NET:

Imports System
Imports System.Web
Imports Elmah
Namespace System
    Public NotInheritable Class ElmahExtension
        Private Sub New()
        End Sub
        <System.Runtime.CompilerServices.Extension> _
        Public Shared Sub LogToElmah(ex As Exception)
            If HttpContext.Current IsNot Nothing Then
                ErrorSignal.FromCurrentContext().Raise(ex)
            Else
                If httpApplication Is Nothing Then
                    InitNoContext()
                End If
                ErrorSignal.[Get](httpApplication).Raise(ex)
            End If
        End Sub

        Private Shared httpApplication As HttpApplication = Nothing
        Private Shared errorFilter As New ErrorFilterConsole()

        Public Shared ErrorEmail As New ErrorMailModule()
        Public Shared ErrorLog As New ErrorLogModule()
        Public Shared ErrorTweet As New ErrorTweetModule()

        Private Shared Sub InitNoContext()
            httpApplication = New HttpApplication()
            errorFilter.Init(httpApplication)

            TryCast(ErrorEmail, IHttpModule).Init(httpApplication)
            errorFilter.HookFiltering(ErrorEmail)

            TryCast(ErrorLog, IHttpModule).Init(httpApplication)
            errorFilter.HookFiltering(ErrorLog)

            TryCast(ErrorTweet, IHttpModule).Init(httpApplication)
            errorFilter.HookFiltering(ErrorTweet)
        End Sub



    Private Class ErrorFilterConsole
        Inherits Elmah.ErrorFilterModule


        Public Sub HookFiltering([module] As Elmah.IExceptionFiltering)
            AddHandler [module].Filtering, New Elmah.ExceptionFilterEventHandler(AddressOf MyBase.OnErrorModuleFiltering)
        End Sub

    End Class


    End Class
End Namespace

No entanto, para apenas entrando erros ao banco de dados, isso vai ser suficiente:

If System.Web.HttpContext.Current Is Nothing Then
    Dim req As System.Web.HttpRequest = New System.Web.HttpRequest(String.Empty, "https://www.domain.tld", Nothing)
    Dim res As System.Web.HttpResponse = New System.Web.HttpResponse(Nothing)
    System.Web.HttpContext.Current = New System.Web.HttpContext(req, res)

    'Dim request As System.Web.Hosting.SimpleWorkerRequest = New System.Web.Hosting.SimpleWorkerRequest("/blah", "c:\inetpub\wwwroot\blah", "blah.html", Nothing, New System.IO.StringWriter())
    'System.Web.HttpContext.Current = New System.Web.HttpContext(request)

    System.Web.HttpContext.Current.ApplicationInstance = New System.Web.HttpApplication()

    Dim ErrorLog As New Elmah.ErrorLogModule()
    TryCast(ErrorLog, System.Web.IHttpModule).Init(System.Web.HttpContext.Current.ApplicationInstance)
End If

Como uma solução completa:

Public parent As Elmah.ServiceProviderQueryHandler = Nothing




' http://stackoverflow.com/questions/5981750/configuring-elmah-with-sql-server-logging-with-encrypted-connection-string
Public Function Elmah_MS_SQL_Callback(objContext As Object) As System.IServiceProvider
    Dim container As New System.ComponentModel.Design.ServiceContainer(parent(objContext))
    Dim strConnectionString As String = COR.SQL.MS_SQL.GetConnectionString()

    Dim log As Elmah.SqlErrorLog = New Elmah.SqlErrorLog(strConnectionString)
    'Dim strApplicationName = System.Web.Compilation.BuildManager.GetGlobalAsaxType().BaseType.Assembly().FullName
    Dim strApplicationName As String = System.Reflection.Assembly.GetExecutingAssembly().FullName
    If Not String.IsNullOrEmpty(strApplicationName) Then
        log.ApplicationName = strApplicationName.Substring(0, strApplicationName.IndexOf(","))
    End If

    container.AddService(GetType(Elmah.ErrorLog), log)
    Return container
End Function ' Elmah_MS_SQL_Callback




Public Function Elmah_PG_SQL_Callback(objContext As Object) As System.IServiceProvider
    Dim container As New System.ComponentModel.Design.ServiceContainer(parent(objContext))
    Dim strConnectionString As String = COR.SQL.MS_SQL.GetConnectionString()

    Dim log As Elmah.PgsqlErrorLog = New Elmah.PgsqlErrorLog(strConnectionString)
    'Dim strApplicationName = System.Web.Compilation.BuildManager.GetGlobalAsaxType().BaseType.Assembly().FullName
    Dim strApplicationName As String = System.Reflection.Assembly.GetExecutingAssembly().FullName
    If Not String.IsNullOrEmpty(strApplicationName) Then
        log.ApplicationName = strApplicationName.Substring(0, strApplicationName.IndexOf(","))
    End If

    container.AddService(GetType(Elmah.ErrorLog), log)
    Return container
End Function ' Elmah_PG_SQL_Callback


' http://weblogs.asp.net/stevewellens/archive/2009/02/01/debugging-a-deployed-site.aspx
Public Sub Initialize()

    If System.Web.HttpContext.Current Is Nothing Then
        Dim req As System.Web.HttpRequest = New System.Web.HttpRequest(String.Empty, "https://www.domain.tld", Nothing)
        Dim res As System.Web.HttpResponse = New System.Web.HttpResponse(Nothing)
        System.Web.HttpContext.Current = New System.Web.HttpContext(req, res)

        'Dim request As System.Web.Hosting.SimpleWorkerRequest = New System.Web.Hosting.SimpleWorkerRequest("/blah", "c:\inetpub\wwwroot\blah", "blah.html", Nothing, New System.IO.StringWriter())
        'System.Web.HttpContext.Current = New System.Web.HttpContext(request)

        System.Web.HttpContext.Current.ApplicationInstance = New System.Web.HttpApplication()

        Dim ErrorLog As New Elmah.ErrorLogModule()
        TryCast(ErrorLog, System.Web.IHttpModule).Init(System.Web.HttpContext.Current.ApplicationInstance)
    End If



    parent = Elmah.ServiceCenter.Current

    If SQL.IsMsSql Then
        Elmah.ServiceCenter.Current = AddressOf Elmah_MS_SQL_Callback
    End If

    If SQL.IsPostGreSql Then
        Elmah.ServiceCenter.Current = AddressOf Elmah_PG_SQL_Callback
    End If
End Sub ' InitializeElmah

E

Elmah.ErrorSignal.FromCurrentContext().Raise(New NotImplementedException("Test"))

vai funcionar se ele é chamado após Initialize ()

Bem, desde que eu não posso comentar eu vou postar isso aqui e talvez alguém vai vê-lo.

Depois de seguir o método de Brian e os commentors eu era capaz de receber e-mail de trabalho, mas eu ainda não estava vendo as mensagens SQL sendo registrado, mesmo que eu tinha definido o applicationName. O que eu não sabia é que eles realmente estavam sendo registrados Eu só não foi vê-los porque o applicationName deve ser o mesmo que o seu web.config, a fim de ser capaz de visualizá-lo.

O meu web.config não tinha applicationName especificado, por isso foi inadimplente para "/ LM / W3SVC / 2 / ROOT", que é basicamente o que "asgeo1", comentou, embora eu não sabia que tinha que ser o mesmo.

Desde que eu realmente não tenho quaisquer erros que eu estava preocupado com, eu configurado applicationName no meu web.config e meu app.config para ser os mesmos e agora tudo mostra-se como um campeão.

<errorLog type="Elmah.SqlErrorLog, Elmah" connectionStringName="ELMAH" applicationName="MyAppName" />

ELMAH significa registro de erros módulos e manipuladores -. Referindo, é claro, para IHttpModule e IHttpHandler

Aplicativos de console não usar HTTP, de modo que normalmente não ser capaz de beneficiar muito com módulos e manipuladores construídos para HTTP.

Licenciado em: CC-BY-SA com atribuição
Não afiliado a StackOverflow
scroll top