2014-08-13 10:39:27 +01:00
|
|
|
//
|
|
|
|
// System.Diagnostics.LocalFileEventLog.cs
|
|
|
|
//
|
|
|
|
// Author:
|
|
|
|
// Atsushi Enomoto <atsushi@ximian.com>
|
|
|
|
// Gert Driesen <drieseng@users.sourceforge.net>
|
|
|
|
//
|
|
|
|
// Copyright (C) 2006 Novell, Inc (http://www.novell.com)
|
|
|
|
//
|
|
|
|
//
|
|
|
|
// Permission is hereby granted, free of charge, to any person obtaining
|
|
|
|
// a copy of this software and associated documentation files (the
|
|
|
|
// "Software"), to deal in the Software without restriction, including
|
|
|
|
// without limitation the rights to use, copy, modify, merge, publish,
|
|
|
|
// distribute, sublicense, and/or sell copies of the Software, and to
|
|
|
|
// permit persons to whom the Software is furnished to do so, subject to
|
|
|
|
// the following conditions:
|
|
|
|
//
|
|
|
|
// The above copyright notice and this permission notice shall be
|
|
|
|
// included in all copies or substantial portions of the Software.
|
|
|
|
//
|
|
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
|
|
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
|
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
|
|
// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
|
|
|
// LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
|
|
|
// OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
|
|
|
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
//
|
|
|
|
|
|
|
|
using System;
|
2015-01-13 10:44:36 +00:00
|
|
|
using System.Collections.Generic;
|
2014-08-13 10:39:27 +01:00
|
|
|
using System.ComponentModel;
|
|
|
|
using System.Diagnostics;
|
|
|
|
using System.Globalization;
|
|
|
|
using System.IO;
|
|
|
|
using System.Runtime.InteropServices;
|
|
|
|
using System.Security;
|
|
|
|
using System.Text;
|
|
|
|
using System.Threading;
|
|
|
|
|
|
|
|
namespace System.Diagnostics
|
|
|
|
{
|
|
|
|
internal class LocalFileEventLog : EventLogImpl
|
|
|
|
{
|
|
|
|
const string DateFormat = "yyyyMMddHHmmssfff";
|
|
|
|
static readonly object lockObject = new object ();
|
|
|
|
FileSystemWatcher file_watcher;
|
|
|
|
int last_notification_index;
|
|
|
|
bool _notifying;
|
|
|
|
|
|
|
|
public LocalFileEventLog (EventLog coreEventLog) : base (coreEventLog)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
public override void BeginInit () {
|
|
|
|
}
|
|
|
|
|
|
|
|
public override void Clear ()
|
|
|
|
{
|
|
|
|
string logDir = FindLogStore (CoreEventLog.Log);
|
|
|
|
if (!Directory.Exists (logDir))
|
|
|
|
return;
|
|
|
|
|
|
|
|
foreach (string file in Directory.GetFiles (logDir, "*.log"))
|
|
|
|
File.Delete (file);
|
|
|
|
}
|
|
|
|
|
|
|
|
public override void Close ()
|
|
|
|
{
|
|
|
|
if (file_watcher != null) {
|
|
|
|
file_watcher.EnableRaisingEvents = false;
|
|
|
|
file_watcher = null; // force creation of new FileSystemWatcher
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public override void CreateEventSource (EventSourceCreationData sourceData)
|
|
|
|
{
|
|
|
|
// construct path for storing log entries
|
|
|
|
string logDir = FindLogStore (sourceData.LogName);
|
|
|
|
// create event log store (if necessary), and modify access
|
|
|
|
// permissions (unix only)
|
|
|
|
if (!Directory.Exists (logDir)) {
|
|
|
|
// ensure the log name is valid for customer logs
|
|
|
|
ValidateCustomerLogName (sourceData.LogName, sourceData.MachineName);
|
|
|
|
|
|
|
|
Directory.CreateDirectory (logDir);
|
|
|
|
// MS does not allow an event source to be named after an already
|
|
|
|
// existing event log. To speed up checking whether a given event
|
|
|
|
// source already exists (either as a event source or event log)
|
|
|
|
// we create an event source directory named after the event log.
|
|
|
|
// This matches what MS does with the registry-based registration.
|
|
|
|
Directory.CreateDirectory (Path.Combine (logDir, sourceData.LogName));
|
|
|
|
if (RunningOnUnix) {
|
|
|
|
ModifyAccessPermissions (logDir, "777");
|
|
|
|
ModifyAccessPermissions (logDir, "+t");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// create directory for event source, so we can check if the event
|
|
|
|
// source already exists
|
|
|
|
string sourceDir = Path.Combine (logDir, sourceData.Source);
|
|
|
|
Directory.CreateDirectory (sourceDir);
|
|
|
|
}
|
|
|
|
|
|
|
|
public override void Delete (string logName, string machineName)
|
|
|
|
{
|
|
|
|
string logDir = FindLogStore (logName);
|
|
|
|
if (!Directory.Exists (logDir))
|
|
|
|
throw new InvalidOperationException (string.Format (
|
|
|
|
CultureInfo.InvariantCulture, "Event Log '{0}'"
|
|
|
|
+ " does not exist on computer '{1}'.", logName,
|
|
|
|
machineName));
|
|
|
|
|
|
|
|
Directory.Delete (logDir, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
public override void DeleteEventSource (string source, string machineName)
|
|
|
|
{
|
|
|
|
if (!Directory.Exists (EventLogStore))
|
|
|
|
throw new ArgumentException (string.Format (
|
|
|
|
CultureInfo.InvariantCulture, "The source '{0}' is not"
|
|
|
|
+ " registered on computer '{1}'.", source, machineName));
|
|
|
|
|
|
|
|
string sourceDir = FindSourceDirectory (source);
|
|
|
|
if (sourceDir == null)
|
|
|
|
throw new ArgumentException (string.Format (
|
|
|
|
CultureInfo.InvariantCulture, "The source '{0}' is not"
|
|
|
|
+ " registered on computer '{1}'.", source, machineName));
|
|
|
|
Directory.Delete (sourceDir);
|
|
|
|
}
|
|
|
|
|
|
|
|
public override void Dispose (bool disposing)
|
|
|
|
{
|
|
|
|
Close ();
|
|
|
|
}
|
|
|
|
|
|
|
|
public override void DisableNotification ()
|
|
|
|
{
|
|
|
|
if (file_watcher == null)
|
|
|
|
return;
|
|
|
|
file_watcher.EnableRaisingEvents = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
public override void EnableNotification ()
|
|
|
|
{
|
|
|
|
if (file_watcher == null) {
|
|
|
|
string logDir = FindLogStore (CoreEventLog.Log);
|
|
|
|
if (!Directory.Exists (logDir))
|
|
|
|
Directory.CreateDirectory (logDir);
|
|
|
|
|
|
|
|
file_watcher = new FileSystemWatcher ();
|
|
|
|
file_watcher.Path = logDir;
|
|
|
|
file_watcher.Created += delegate (object o, FileSystemEventArgs e) {
|
|
|
|
lock (this) {
|
|
|
|
if (_notifying)
|
|
|
|
return;
|
|
|
|
_notifying = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// allow for file to be finished writing
|
|
|
|
Thread.Sleep (100);
|
|
|
|
|
|
|
|
// Process every new entry in one notification event.
|
|
|
|
try {
|
|
|
|
while (GetLatestIndex () > last_notification_index) {
|
|
|
|
try {
|
|
|
|
CoreEventLog.OnEntryWritten (GetEntry (last_notification_index++));
|
|
|
|
} catch (Exception ex) {
|
|
|
|
// FIXME: find some proper way to output this error
|
|
|
|
Debug.WriteLine (ex);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} finally {
|
|
|
|
lock (this)
|
|
|
|
_notifying = false;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
last_notification_index = GetLatestIndex ();
|
|
|
|
file_watcher.EnableRaisingEvents = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
public override void EndInit () { }
|
|
|
|
|
|
|
|
public override bool Exists (string logName, string machineName)
|
|
|
|
{
|
|
|
|
string logDir = FindLogStore (logName);
|
|
|
|
return Directory.Exists (logDir);
|
|
|
|
}
|
|
|
|
|
|
|
|
[MonoTODO ("Use MessageTable from PE for lookup")]
|
|
|
|
protected override string FormatMessage (string source, uint eventID, string [] replacementStrings)
|
|
|
|
{
|
|
|
|
return string.Join (", ", replacementStrings);
|
|
|
|
}
|
|
|
|
|
|
|
|
protected override int GetEntryCount ()
|
|
|
|
{
|
|
|
|
string logDir = FindLogStore (CoreEventLog.Log);
|
|
|
|
if (!Directory.Exists (logDir))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
string[] logFiles = Directory.GetFiles (logDir, "*.log");
|
|
|
|
return logFiles.Length;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected override EventLogEntry GetEntry (int index)
|
|
|
|
{
|
|
|
|
string logDir = FindLogStore (CoreEventLog.Log);
|
|
|
|
|
|
|
|
// our file names are one-based
|
|
|
|
string file = Path.Combine (logDir, (index + 1).ToString (
|
|
|
|
CultureInfo.InvariantCulture) + ".log");
|
|
|
|
|
|
|
|
using (TextReader tr = File.OpenText (file)) {
|
|
|
|
int eventIndex = int.Parse (Path.GetFileNameWithoutExtension (file),
|
|
|
|
CultureInfo.InvariantCulture);
|
|
|
|
uint instanceID = uint.Parse (tr.ReadLine ().Substring (12),
|
|
|
|
CultureInfo.InvariantCulture);
|
|
|
|
EventLogEntryType type = (EventLogEntryType)
|
|
|
|
Enum.Parse (typeof (EventLogEntryType), tr.ReadLine ().Substring (11));
|
|
|
|
string source = tr.ReadLine ().Substring (8);
|
|
|
|
string category = tr.ReadLine ().Substring (10);
|
|
|
|
short categoryNumber = short.Parse(category, CultureInfo.InvariantCulture);
|
|
|
|
string categoryName = "(" + category + ")";
|
|
|
|
DateTime timeGenerated = DateTime.ParseExact (tr.ReadLine ().Substring (15),
|
|
|
|
DateFormat, CultureInfo.InvariantCulture);
|
|
|
|
DateTime timeWritten = File.GetLastWriteTime (file);
|
|
|
|
int stringNums = int.Parse (tr.ReadLine ().Substring (20));
|
2015-01-13 10:44:36 +00:00
|
|
|
var replacementTemp = new List<string> ();
|
2014-08-13 10:39:27 +01:00
|
|
|
StringBuilder sb = new StringBuilder ();
|
|
|
|
while (replacementTemp.Count < stringNums) {
|
|
|
|
char c = (char) tr.Read ();
|
|
|
|
if (c == '\0') {
|
|
|
|
replacementTemp.Add (sb.ToString ());
|
|
|
|
sb.Length = 0;
|
|
|
|
} else {
|
|
|
|
sb.Append (c);
|
|
|
|
}
|
|
|
|
}
|
2015-01-13 10:44:36 +00:00
|
|
|
string [] replacementStrings = replacementTemp.ToArray ();
|
2014-08-13 10:39:27 +01:00
|
|
|
|
|
|
|
string message = FormatMessage (source, instanceID, replacementStrings);
|
|
|
|
int eventID = EventLog.GetEventID (instanceID);
|
|
|
|
|
|
|
|
byte [] bin = Convert.FromBase64String (tr.ReadToEnd ());
|
|
|
|
return new EventLogEntry (categoryName, categoryNumber, eventIndex,
|
|
|
|
eventID, source, message, null, Environment.MachineName,
|
|
|
|
type, timeGenerated, timeWritten, bin, replacementStrings,
|
|
|
|
instanceID);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
[MonoTODO]
|
|
|
|
protected override string GetLogDisplayName ()
|
|
|
|
{
|
|
|
|
return CoreEventLog.Log;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected override string [] GetLogNames (string machineName)
|
|
|
|
{
|
|
|
|
if (!Directory.Exists (EventLogStore))
|
|
|
|
return new string [0];
|
|
|
|
|
|
|
|
string [] logDirs = Directory.GetDirectories (EventLogStore, "*");
|
|
|
|
string [] logNames = new string [logDirs.Length];
|
|
|
|
for (int i = 0; i < logDirs.Length; i++)
|
|
|
|
logNames [i] = Path.GetFileName (logDirs [i]);
|
|
|
|
return logNames;
|
|
|
|
}
|
|
|
|
|
|
|
|
public override string LogNameFromSourceName (string source, string machineName)
|
|
|
|
{
|
|
|
|
if (!Directory.Exists (EventLogStore))
|
|
|
|
return string.Empty;
|
|
|
|
|
|
|
|
string sourceDir = FindSourceDirectory (source);
|
|
|
|
if (sourceDir == null)
|
|
|
|
return string.Empty;
|
|
|
|
DirectoryInfo info = new DirectoryInfo (sourceDir);
|
|
|
|
return info.Parent.Name;
|
|
|
|
}
|
|
|
|
|
|
|
|
public override bool SourceExists (string source, string machineName)
|
|
|
|
{
|
|
|
|
if (!Directory.Exists (EventLogStore))
|
|
|
|
return false;
|
|
|
|
string sourceDir = FindSourceDirectory (source);
|
|
|
|
return (sourceDir != null);
|
|
|
|
}
|
|
|
|
|
|
|
|
public override void WriteEntry (string [] replacementStrings, EventLogEntryType type, uint instanceID, short category, byte [] rawData)
|
|
|
|
{
|
|
|
|
lock (lockObject) {
|
|
|
|
string logDir = FindLogStore (CoreEventLog.Log);
|
|
|
|
|
|
|
|
int index = GetLatestIndex () + 1;
|
|
|
|
string logPath = Path.Combine (logDir, index.ToString (CultureInfo.InvariantCulture) + ".log");
|
|
|
|
try {
|
|
|
|
using (TextWriter w = File.CreateText (logPath)) {
|
|
|
|
w.WriteLine ("InstanceID: {0}", instanceID.ToString (CultureInfo.InvariantCulture));
|
|
|
|
w.WriteLine ("EntryType: {0}", (int) type);
|
|
|
|
w.WriteLine ("Source: {0}", CoreEventLog.Source);
|
|
|
|
w.WriteLine ("Category: {0}", category.ToString (CultureInfo.InvariantCulture));
|
|
|
|
w.WriteLine ("TimeGenerated: {0}", DateTime.Now.ToString (
|
|
|
|
DateFormat, CultureInfo.InvariantCulture));
|
|
|
|
w.WriteLine ("ReplacementStrings: {0}", replacementStrings.
|
|
|
|
Length.ToString (CultureInfo.InvariantCulture));
|
|
|
|
StringBuilder sb = new StringBuilder ();
|
|
|
|
for (int i = 0; i < replacementStrings.Length; i++) {
|
|
|
|
string replacement = replacementStrings [i];
|
|
|
|
sb.Append (replacement);
|
|
|
|
sb.Append ('\0');
|
|
|
|
}
|
|
|
|
w.Write (sb.ToString ());
|
|
|
|
w.Write (Convert.ToBase64String (rawData));
|
|
|
|
}
|
|
|
|
} catch (IOException) {
|
|
|
|
File.Delete (logPath);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private string FindSourceDirectory (string source)
|
|
|
|
{
|
|
|
|
string sourceDir = null;
|
|
|
|
|
|
|
|
string [] logDirs = Directory.GetDirectories (EventLogStore, "*");
|
|
|
|
for (int i = 0; i < logDirs.Length; i++) {
|
|
|
|
string [] sourceDirs = Directory.GetDirectories (logDirs [i], "*");
|
|
|
|
for (int j = 0; j < sourceDirs.Length; j++) {
|
|
|
|
string relativeDir = Path.GetFileName (sourceDirs [j]);
|
|
|
|
// use a case-insensitive comparison
|
|
|
|
if (string.Compare (relativeDir, source, true, CultureInfo.InvariantCulture) == 0) {
|
|
|
|
sourceDir = sourceDirs [j];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return sourceDir;
|
|
|
|
}
|
|
|
|
|
|
|
|
private bool RunningOnUnix {
|
|
|
|
get {
|
|
|
|
int p = (int) Environment.OSVersion.Platform;
|
|
|
|
return ((p == 4) || (p == 128) || (p == 6));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private string FindLogStore (string logName) {
|
|
|
|
// when the event log store does not yet exist, there's no need
|
|
|
|
// to perform a case-insensitive lookup
|
|
|
|
if (!Directory.Exists (EventLogStore))
|
|
|
|
return Path.Combine (EventLogStore, logName);
|
|
|
|
|
|
|
|
// we'll use a case-insensitive lookup to match the MS behaviour
|
|
|
|
// while still allowing the original casing of the log name to be
|
|
|
|
// retained
|
|
|
|
string [] logDirs = Directory.GetDirectories (EventLogStore, "*");
|
|
|
|
for (int i = 0; i < logDirs.Length; i++) {
|
|
|
|
string relativeDir = Path.GetFileName (logDirs [i]);
|
|
|
|
// use a case-insensitive comparison
|
|
|
|
if (string.Compare (relativeDir, logName, true, CultureInfo.InvariantCulture) == 0) {
|
|
|
|
return logDirs [i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return Path.Combine (EventLogStore, logName);
|
|
|
|
}
|
|
|
|
|
|
|
|
private string EventLogStore {
|
|
|
|
get {
|
|
|
|
// for the local file implementation, the MONO_EVENTLOG_TYPE
|
|
|
|
// environment variable can contain the path of the event log
|
|
|
|
// store by using the following syntax: local:<path>
|
|
|
|
string eventLogType = Environment.GetEnvironmentVariable (EventLog.EVENTLOG_TYPE_VAR);
|
|
|
|
if (eventLogType != null && eventLogType.Length > EventLog.LOCAL_FILE_IMPL.Length + 1)
|
|
|
|
return eventLogType.Substring (EventLog.LOCAL_FILE_IMPL.Length + 1);
|
|
|
|
if (RunningOnUnix) {
|
|
|
|
return "/var/lib/mono/eventlog";
|
|
|
|
} else {
|
|
|
|
return Path.Combine (Environment.GetFolderPath (
|
|
|
|
Environment.SpecialFolder.CommonApplicationData),
|
|
|
|
"mono\\eventlog");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private int GetLatestIndex () {
|
|
|
|
// our file names are one-based
|
|
|
|
int maxIndex = 0;
|
|
|
|
string[] logFiles = Directory.GetFiles (FindLogStore (CoreEventLog.Log), "*.log");
|
|
|
|
for (int i = 0; i < logFiles.Length; i++) {
|
|
|
|
try {
|
|
|
|
string file = logFiles[i];
|
|
|
|
int index = int.Parse (Path.GetFileNameWithoutExtension (
|
|
|
|
file), CultureInfo.InvariantCulture);
|
|
|
|
if (index > maxIndex)
|
|
|
|
maxIndex = index;
|
|
|
|
} catch {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return maxIndex;
|
|
|
|
}
|
|
|
|
|
|
|
|
private static void ModifyAccessPermissions (string path, string permissions)
|
|
|
|
{
|
|
|
|
ProcessStartInfo pi = new ProcessStartInfo ();
|
|
|
|
pi.FileName = "chmod";
|
|
|
|
pi.RedirectStandardOutput = true;
|
|
|
|
pi.RedirectStandardError = true;
|
|
|
|
pi.UseShellExecute = false;
|
|
|
|
pi.Arguments = string.Format ("{0} \"{1}\"", permissions, path);
|
|
|
|
|
|
|
|
Process p = null;
|
|
|
|
try {
|
|
|
|
p = Process.Start (pi);
|
|
|
|
} catch (Exception ex) {
|
|
|
|
throw new SecurityException ("Access permissions could not be modified.", ex);
|
|
|
|
}
|
|
|
|
|
|
|
|
p.WaitForExit ();
|
|
|
|
if (p.ExitCode != 0) {
|
|
|
|
p.Close ();
|
|
|
|
throw new SecurityException ("Access permissions could not be modified.");
|
|
|
|
}
|
|
|
|
p.Close ();
|
|
|
|
}
|
|
|
|
|
|
|
|
public override OverflowAction OverflowAction {
|
|
|
|
get { return OverflowAction.DoNotOverwrite; }
|
|
|
|
}
|
|
|
|
|
|
|
|
public override int MinimumRetentionDays {
|
|
|
|
get { return int.MaxValue; }
|
|
|
|
}
|
|
|
|
|
|
|
|
public override long MaximumKilobytes {
|
|
|
|
get { return long.MaxValue; }
|
|
|
|
set { throw new NotSupportedException ("This EventLog implementation does not support setting max kilobytes policy"); }
|
|
|
|
}
|
|
|
|
|
|
|
|
public override void ModifyOverflowPolicy (OverflowAction action, int retentionDays)
|
|
|
|
{
|
|
|
|
throw new NotSupportedException ("This EventLog implementation does not support modifying overflow policy");
|
|
|
|
}
|
|
|
|
|
|
|
|
public override void RegisterDisplayName (string resourceFile, long resourceId)
|
|
|
|
{
|
|
|
|
throw new NotSupportedException ("This EventLog implementation does not support registering display name");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|