如何以编程方式更改文件位置?

71 c# log4net

我对Log4net完全不熟悉.
我已经设法通过添加配置文件和简单的日志记录来实现目标.
我已经硬编码了这个值,"C:\temp\log.txt"但这还不够好.

日志必须转到特殊文件夹

path = Environment.GetFolderPath(Environment.SpecialFolder.CommonApplicationData);
Run Code Online (Sandbox Code Playgroud)

并且此路径会根据您使用的是Windows Server 2008还是Windows XP或Vista等而变化...

如何以编程方式在log4net中更改文件的位置?

这就是我所做的:

<configSections>
<section name="log4net"
         type="log4net.Config.Log4NetConfigurationSectionHandler,Log4net"/>
</configSections>
<log4net>         
    <root>
        <level value="DEBUG" />
        <appender-ref ref="LogFileAppender" />
    </root>
    <appender name="LogFileAppender" type="log4net.Appender.RollingFileAppender">
        <param name="File" value="C:\temp\log.txt" />
        <param name="AppendToFile" value="true" />
        <rollingStyle value="Size" />
        <maxSizeRollBackups value="10" />
        <maximumFileSize value="10MB" />
        <staticLogFileName value="true" />
        <layout type="log4net.Layout.PatternLayout">
            <param name="ConversionPattern" value="%-5p%d{yyyy-MM-dd hh:mm:ss} – %m%n" />
        </layout>
    </appender>
</log4net>
Run Code Online (Sandbox Code Playgroud)
class Program
{
    protected static readonly ILog log = LogManager.GetLogger(typeof(Program));

    static void Main(string[] args)
    {
        log4net.Config.XmlConfigurator.Configure();
        log.Warn("Log something");

        path = Environment.GetFolderPath(Environment.SpecialFolder.CommonApplicationData);


        // How can I change where I log stuff?
    }
}
Run Code Online (Sandbox Code Playgroud)

只需要弄清楚如何将内容记录到我想要的位置.

有什么建议?非常感谢

Pet*_*old 86

log4net可以为您处理.在这种情况下,可以使用log4net.Util.PatternString选项处理程序格式化string类型的任何appender属性.PatternString甚至支持SpecialFolder枚举,它启用以下优雅配置:

<appender name="LogFileAppender" type="log4net.Appender.RollingFileAppender" >
    <file type="log4net.Util.PatternString" 
        value="%envFolderPath{CommonApplicationData}\\test.txt" />
    ...
</appender>
Run Code Online (Sandbox Code Playgroud)

这是一个证明布丁的单元测试:

[Test]
public void Load()
{
    XmlConfigurator.Configure();
    var fileAppender = LogManager.GetRepository()
        .GetAppenders().First(appender => appender is RollingFileAppender);

    var expectedFile = 
        Path.Combine(
            Environment.GetFolderPath(
                Environment.SpecialFolder.CommonApplicationData),
                "test.txt");

    Assert.That(fileAppender, 
        Is.Not.Null & Has.Property("File").EqualTo(expectedFile));
}
Run Code Online (Sandbox Code Playgroud)

以下测试验证log4net实际写入磁盘(这基本上使其成为"集成"测试,而不是单元测试,但我们暂时将其保留在此处):

[Test]
public void Log4net_WritesToDisk()
{
    var expectedFile = 
        Path.Combine(
            Environment.GetFolderPath(
                Environment.SpecialFolder.CommonApplicationData),
                "test.txt");

    if (File.Exists(expectedFile))
        File.Delete(expectedFile);

    XmlConfigurator.Configure();

    var log = LogManager.GetLogger(typeof (ConfigTest));
    log.Info("Message from test");

    LogManager.Shutdown();

    Assert.That(File.ReadAllText(expectedFile), 
        Text.Contains("Message from test"));
}
Run Code Online (Sandbox Code Playgroud)

注意:我强烈建议使用上面示例中演示的compact属性语法.删除所有那些"<property name ="使您的配置更具可读性.

  • %envFolderPath {}似乎不是log4net当前(1.2.10)版本的一部分.我不得不从他们的subversion存储库中提取r606477来启用这个非常有用的功能.我希望这可以包含在下一个(1.2.11)版本中. (2认同)
  • 从@ JackAce的下面的答案,重置文件路径后,请务必调用".ActivateOptions()"来激活它.调用.file将设置它,但在ActivateOptions调用之前不会使用它 (2认同)

Jac*_*Ace 43

我在互联网上发现了这个代码的变异:

XmlConfigurator.Configure();
log4net.Repository.Hierarchy.Hierarchy h =
(log4net.Repository.Hierarchy.Hierarchy) LogManager.GetRepository();
foreach (IAppender a in h.Root.Appenders)
{
    if (a is FileAppender)
    {
        FileAppender fa = (FileAppender)a;
        // Programmatically set this to the desired location here
        string logFileLocation = @"C:\MySpecialFolder\MyFile.log";

        // Uncomment the lines below if you want to retain the base file name
        // and change the folder name...
        //FileInfo fileInfo = new FileInfo(fa.File);
        //logFileLocation = string.Format(@"C:\MySpecialFolder\{0}", fileInfo.Name);

        fa.File = logFileLocation;
        fa.ActivateOptions();
        break;
    }
}
Run Code Online (Sandbox Code Playgroud)

这适合我.我们的应用程序需要将日志文件放在包含基于AssemblyInfo.cs文件的应用程序版本号的文件夹中.

您应该能够以编程方式设置logFileLocation(例如,如果这是一个Web应用程序,您可以使用Server.MapPath())以满足您的需要.

  • 我觉得这样可以更好地控制文件位置. (4认同)
  • 我认为这是最好的答案:) (3认同)

cod*_*ike 14

看起来Peter的答案对Log4net v1.2.10.0不起作用.这里描述另一种方法.

基本上,该方法是为log4net配置文件实现自定义模式转换器.

首先将此类添加到您的项目中:

public class SpecialFolderPatternConverter : log4net.Util.PatternConverter
{
    override protected void Convert(System.IO.TextWriter writer, object state)
    {
        Environment.SpecialFolder specialFolder = (Environment.SpecialFolder)Enum.Parse(typeof(Environment.SpecialFolder), base.Option, true);
        writer.Write(Environment.GetFolderPath(specialFolder));
    }
}
Run Code Online (Sandbox Code Playgroud)

然后设置FileAppender的File参数,如下所示:

<file type="log4net.Util.PatternString">
    <converter>
      <name value="folder" />
      <type value="MyAppName.SpecialFolderPatternConverter,MyAppName" />
    </converter>
    <conversionPattern value="%folder{CommonApplicationData}\\SomeOtherFolder\\log.txt" />
  </file>
Run Code Online (Sandbox Code Playgroud)

基本上它%folder告诉它看看转换器folder,它将它指向Sp​​ecialFolderPatternConverter类.然后它调用Convert该类,传入CommonApplicationData(或其他)枚举值.


小智 6

怎么样简单:

XmlConfigurator.LogFullFilename = @"c:\ProgramData\MyApp\Myapp.log";
Run Code Online (Sandbox Code Playgroud)

做一个非常简单的事情为什么这么复杂?

  • 因为,路径会根据您是在Windows Server 2008还是winxp或vista等而变化...在XP中没有c:\ ProgramData文件夹,仅在Win7/Vista中.因此,如果我们要部署此应用程序,它必须兼容所有操作系统.:-)硬编码不是一个好习惯. (12认同)
  • 我正在使用log4net v2.0.3,并且此属性在XmlConfigurator上不可用.自答案发布以来必须已删除. (6认同)

Jim*_*109 6

还要更改错误日志的路径(基于 JackAce 的答案):

private static void SetLogPath(string path, string errorPath)
{
    XmlConfigurator.Configure();
    log4net.Repository.Hierarchy.Hierarchy h =
    (log4net.Repository.Hierarchy.Hierarchy)LogManager.GetRepository();
    foreach (var a in h.Root.Appenders)
    {
        if (a is log4net.Appender.FileAppender)
        {
            if (a.Name.Equals("LogFileAppender"))
            { 
                log4net.Appender.FileAppender fa = (log4net.Appender.FileAppender)a;                    
                string logFileLocation = path; 
                fa.File = logFileLocation;                   
                fa.ActivateOptions();
            }
            else if (a.Name.Equals("ErrorFileAppender"))
            {
                log4net.Appender.FileAppender fa = (log4net.Appender.FileAppender)a;
                string logFileLocation = errorPath;
                fa.File = logFileLocation;
                fa.ActivateOptions();
            }
        }
    }
}
Run Code Online (Sandbox Code Playgroud)


bsh*_*hea 5

这对我有用:

  <log4net>
    <appender name="RollingFileAppender" type="log4net.Appender.RollingFileAppender">
..
      <file value="${APPDATA}\MyApp\MyApp Client\logs\Log.txt"/>
..
  </log4net>
Run Code Online (Sandbox Code Playgroud)

如果需要写入特殊文件夹,我在这里找到了帮助(第二个和第三个示例)。

编辑:

要回答OP..这适用于“所有用户”区域:

      ...
      <file value="${ALLUSERSPROFILE}\MyApp\MyApp Client\logs\Log.txt"/>
      ...
Run Code Online (Sandbox Code Playgroud)

在较新版本的 Windows 中,通常为“C:\ProgramData”。

另请参阅这些:
如何为 log4net 指定通用应用程序数据文件夹?== /sf/answers/132271401/和评论
&
https://superuser.com/q/405097/47628
/sf/answers/388535171/