22 Commits

Author SHA1 Message Date
Eric Fontana
8fe3b1815c Incorporated review changes 2015-04-30 13:12:44 -04:00
Eric Fontana
58ae2db05c Remove internal names. 2015-04-30 10:20:24 -04:00
Eric Fontana
f1c4343cf6 Fixed typo 2015-04-30 10:10:15 -04:00
Eric Fontana
3669acbb26 Release 1.3.25.0 candidate 2015-04-30 10:09:07 -04:00
Eric Fontana
6600bb2b85 Added type parameter for Log/Tail inputs 2015-04-29 10:58:13 -04:00
Eric Fontana
50473a1f5a Merge pull request #50 from Cimpress-MCP/add_input_generator
All updates for 1.3.24.0
2015-04-29 10:47:41 -04:00
Eric Fontana
dcbb079101 Doc tweaks from review and make infinite be infinite 2015-04-29 10:04:48 -04:00
Eric Fontana
091fe9e7e4 Removed all usage of ForceFlush 2015-04-29 09:55:47 -04:00
Eric Fontana
6db530b526 Remove dead code 2015-04-29 09:51:40 -04:00
Eric Fontana
4ebe539ea8 All updates for 1.3.24.0 2015-04-29 08:35:07 -04:00
Eric Fontana
75619a239a Updated installer to include Nest and Elasticsearch.dlls which were missing for elasticsearch output 2015-04-23 11:01:23 -04:00
Eric Fontana
fd67c271b5 Fixed bug with elasticsearch output to disable ping by default, and parametrize ping timeot 2015-04-23 08:35:37 -04:00
Eric Fontana
1f3aaf90fd Merge pull request #46 from thoean/WaitUntilAllMesssagesAreProcessed
UdpListener shutdown simplification
2015-04-21 12:59:50 -04:00
Markus Thurner
dcfdf73842 UdpListener: Simplify shutdown procedure and wait until all messages are processed. 2015-04-20 10:39:50 +02:00
Eric Fontana
dcd104e4f4 Tweaked shutdown code for async Udp Listener 2015-04-14 12:45:31 -04:00
Eric Fontana
2377d4ebd2 Merge pull request #41 from thoean/MakeUdpClientAsync
Make UDP listener async
2015-04-14 11:30:48 -04:00
Markus Thurner
64979df012 Use Begin/End Receive for data, and immediately start receiving data again. Processing of result is done outside of the UDP receive method in a separate thread, allowing to process more messages in peak situations. 2015-04-14 15:51:38 +02:00
Eric Fontana
5eb75ab143 Be more pessimistic upon restarts 2015-04-14 07:33:57 -04:00
Eric Fontana
3e9ef6ae88 Updated to reflect master branch build badge 2015-04-13 13:49:03 -04:00
Eric Fontana
7df4dede90 Final tweak for embedded chocolatey 2015-04-13 11:19:37 -04:00
Eric Fontana
d9509757e3 Fixed installer/uninstaller for chocolatey 2015-04-13 11:06:41 -04:00
Eric Fontana
edcac22ea0 Merge pull request #45 from Cimpress-MCP/embedded_chocolatey
Embedded chocolatey
2015-04-13 09:25:40 -04:00
60 changed files with 1405 additions and 307 deletions

View File

@@ -25,7 +25,8 @@ https://groups.google.com/forum/#!forum/timberwinr
Latest Build: Latest Build:
![alt tag](https://ci.appveyor.com/api/projects/status/github/Cimpress-MCP/TimberWinR) ![alt tag](https://ci.appveyor.com/api/projects/status/qi5x2lg153h1tim6/branch/master?svg=true)
## Inputs ## Inputs
The current supported Input format sources are: The current supported Input format sources are:
@@ -36,7 +37,8 @@ The current supported Input format sources are:
5. [Stdin](https://github.com/Cimpress-MCP/TimberWinR/blob/master/TimberWinR/mdocs/StdinInput.md) (Standard Input for Debugging) 5. [Stdin](https://github.com/Cimpress-MCP/TimberWinR/blob/master/TimberWinR/mdocs/StdinInput.md) (Standard Input for Debugging)
6. [W3C](https://github.com/Cimpress-MCP/TimberWinR/blob/master/TimberWinR/mdocs/W3CInput.md)(Internet Information Services W3C Advanced/Custom Format) 6. [W3C](https://github.com/Cimpress-MCP/TimberWinR/blob/master/TimberWinR/mdocs/W3CInput.md)(Internet Information Services W3C Advanced/Custom Format)
7. [Udp](https://github.com/Cimpress-MCP/TimberWinR/blob/master/TimberWinR/mdocs/UdpInput.md) (listens for UDP on port for JSON messages) 7. [Udp](https://github.com/Cimpress-MCP/TimberWinR/blob/master/TimberWinR/mdocs/UdpInput.md) (listens for UDP on port for JSON messages)
8. [TailFiles](https://github.com/Cimpress-MCP/TimberWinR/blob/master/TimberWinR/mdocs/TailFiles.md) (Tails log files efficiently *New*) 8. [TailFiles](https://github.com/Cimpress-MCP/TimberWinR/blob/master/TimberWinR/mdocs/TailFiles.md) (Tails log files efficiently)
8. [Generator](https://github.com/Cimpress-MCP/TimberWinR/blob/master/TimberWinR/mdocs/Generator.md) (Generate logs for testing *New*)
## Codecs ## Codecs
The current list of supported codecs are: The current list of supported codecs are:
@@ -105,6 +107,7 @@ represented as a JSON Property or Array.
1. [Redis](https://github.com/Cimpress-MCP/TimberWinR/blob/master/TimberWinR/mdocs/RedisOutput.md) 1. [Redis](https://github.com/Cimpress-MCP/TimberWinR/blob/master/TimberWinR/mdocs/RedisOutput.md)
2. [Elasticsearch](https://github.com/Cimpress-MCP/TimberWinR/blob/master/TimberWinR/mdocs/ElasticsearchOutput.md) 2. [Elasticsearch](https://github.com/Cimpress-MCP/TimberWinR/blob/master/TimberWinR/mdocs/ElasticsearchOutput.md)
3. [Stdout](https://github.com/Cimpress-MCP/TimberWinR/blob/master/TimberWinR/mdocs/StdoutOutput.md) 3. [Stdout](https://github.com/Cimpress-MCP/TimberWinR/blob/master/TimberWinR/mdocs/StdoutOutput.md)
4. [File](https://github.com/Cimpress-MCP/TimberWinR/blob/master/TimberWinR/mdocs/FileOutput.md)
## Sample Configuration ## Sample Configuration
TimberWinR reads a JSON configuration file, an example file is shown here: TimberWinR reads a JSON configuration file, an example file is shown here:

View File

@@ -28,9 +28,6 @@ namespace TimberWinR.ServiceHost
private static void Main(string[] args) private static void Main(string[] args)
{ {
Arguments arguments = new Arguments(); Arguments arguments = new Arguments();
HostFactory.Run(hostConfigurator => HostFactory.Run(hostConfigurator =>

View File

@@ -32,5 +32,5 @@ using System.Runtime.InteropServices;
// You can specify all the values or you can default the Build and Revision Numbers // You can specify all the values or you can default the Build and Revision Numbers
// by using the '*' as shown below: // by using the '*' as shown below:
// [assembly: AssemblyVersion("1.0.*")] // [assembly: AssemblyVersion("1.0.*")]
[assembly: AssemblyVersion("1.3.21.0")] [assembly: AssemblyVersion("1.3.25.0")]
[assembly: AssemblyFileVersion("1.3.21.0")] [assembly: AssemblyFileVersion("1.3.25.0")]

View File

@@ -15,6 +15,9 @@ namespace TimberWinR.TestGenerator
[Option("timberWinRConfig", DefaultValue = "default.json", HelpText = "Config file/directory to use")] [Option("timberWinRConfig", DefaultValue = "default.json", HelpText = "Config file/directory to use")]
public string TimberWinRConfigFile { get; set; } public string TimberWinRConfigFile { get; set; }
[Option("start", HelpText = "Start an instance of TimberWinR")]
public bool StartTimberWinR { get; set; }
[Option("testDir", DefaultValue = ".", HelpText = "Test directory to use (created if necessary)")] [Option("testDir", DefaultValue = ".", HelpText = "Test directory to use (created if necessary)")]
public string TestDir { get; set; } public string TestDir { get; set; }

View File

@@ -63,8 +63,7 @@ namespace TimberWinR.TestGenerator
// This text is always added, making the file longer over time // This text is always added, making the file longer over time
// if it is not deleted. // if it is not deleted.
using (StreamWriter sw = File.AppendText(logFilePath)) using (StreamWriter sw = File.AppendText(logFilePath))
{ {
sw.AutoFlush = true;
for (int i = 0; i < parms.NumMessages; i++) for (int i = 0; i < parms.NumMessages; i++)
{ {
JObject o = new JObject JObject o = new JObject

View File

@@ -64,8 +64,7 @@ namespace TimberWinR.TestGenerator
// This text is always added, making the file longer over time // This text is always added, making the file longer over time
// if it is not deleted. // if it is not deleted.
using (StreamWriter sw = File.AppendText(logFilePath)) using (StreamWriter sw = File.AppendText(logFilePath))
{ {
sw.AutoFlush = true;
for (int i = 0; i < parms.NumMessages; i++) for (int i = 0; i < parms.NumMessages; i++)
{ {
JObject o = new JObject JObject o = new JObject

View File

@@ -4,6 +4,7 @@ using System.Diagnostics;
using System.IO; using System.IO;
using System.Linq; using System.Linq;
using System.Linq.Expressions; using System.Linq.Expressions;
using System.Net;
using System.Text; using System.Text;
using System.Threading; using System.Threading;
using System.Threading.Tasks; using System.Threading.Tasks;
@@ -55,7 +56,7 @@ namespace TimberWinR.TestGenerator
ramCounter.CategoryName = "Memory"; ramCounter.CategoryName = "Memory";
ramCounter.CounterName = "% Committed Bytes In Use"; ramCounter.CounterName = "% Committed Bytes In Use";
Options = new CommandLineOptions(); Options = new CommandLineOptions();
if (CommandLine.Parser.Default.ParseArguments(args, Options)) if (CommandLine.Parser.Default.ParseArguments(args, Options))
@@ -86,7 +87,8 @@ namespace TimberWinR.TestGenerator
var sw = Stopwatch.StartNew(); var sw = Stopwatch.StartNew();
// Startup TimberWinR // Startup TimberWinR
StartTimberWinR(Options.TimberWinRConfigFile, Options.LogLevel, ".", false); if (Options.StartTimberWinR)
StartTimberWinR(Options.TimberWinRConfigFile, Options.LogLevel, ".", false);
// Run the Generators // Run the Generators
var arrayOfTasks = RunGenerators(Options); var arrayOfTasks = RunGenerators(Options);
@@ -114,7 +116,16 @@ namespace TimberWinR.TestGenerator
sw.Start(); sw.Start();
// Get all the stats // Get all the stats
var jsonTimberWinr = ShutdownTimberWinR(); JObject jsonTimberWinr;
if (Options.StartTimberWinR)
jsonTimberWinr = ShutdownTimberWinR();
else
{
jsonTimberWinr = GetDiagnosticsOutput();
if (jsonTimberWinr == null)
return 3;
}
LogManager.GetCurrentClassLogger().Info("Finished Shutdown: " + sw.Elapsed); LogManager.GetCurrentClassLogger().Info("Finished Shutdown: " + sw.Elapsed);
sw.Stop(); sw.Stop();
@@ -131,6 +142,30 @@ namespace TimberWinR.TestGenerator
return 1; return 1;
} }
public static string GET(string url)
{
try
{
HttpWebRequest request = (HttpWebRequest)WebRequest.Create(url);
HttpWebResponse response = (HttpWebResponse)request.GetResponse();
Stream stream = response.GetResponseStream();
StreamReader reader = new StreamReader(stream);
string data = reader.ReadToEnd();
reader.Close();
stream.Close();
return data;
}
catch (Exception e)
{
LogManager.GetCurrentClassLogger().ErrorException("Error in GET", e);
}
return null;
}
private static void CopySourceFile(string fileName, string outputDir) private static void CopySourceFile(string fileName, string outputDir)
{ {
FileInfo fi = new FileInfo(fileName); FileInfo fi = new FileInfo(fileName);
@@ -199,16 +234,16 @@ namespace TimberWinR.TestGenerator
switch (inputProp.Name) switch (inputProp.Name)
{ {
case "udp": case "udp":
if (VerifyConditions(json, new string[] {"udp"}, inputProp, jresult) != 0) if (VerifyConditions(json, new string[] { "udp" }, inputProp, jresult) != 0)
return 1; return 1;
break; break;
case "tcp": case "tcp":
if (VerifyConditions(json, new string[] {"tcp"}, inputProp, jresult) != 0) if (VerifyConditions(json, new string[] { "tcp" }, inputProp, jresult) != 0)
return 1; return 1;
break; break;
case "log": case "log":
case "taillog": case "taillog":
if (VerifyConditions(json, new string[] {"log", "taillog"}, inputProp, jresult) != 0) if (VerifyConditions(json, new string[] { "log", "taillog" }, inputProp, jresult) != 0)
return 1; return 1;
break; break;
} }
@@ -269,13 +304,32 @@ namespace TimberWinR.TestGenerator
return 0; return 0;
} }
private static JObject GetDiagnosticsOutput()
{
if (Diagnostics != null)
return Diagnostics.DiagnosticsOutput();
else
{
var jsonDiag = GET("http://localhost:5141");
if (jsonDiag == null)
{
LogManager.GetCurrentClassLogger().Error("TimberWinR diagnostics port not responding.");
return null;
}
return JObject.Parse(jsonDiag);
}
}
// Wait till all output has been transmitted. // Wait till all output has been transmitted.
private static void WaitForOutputTransmission() private static void WaitForOutputTransmission()
{ {
bool completed = false; bool completed = false;
do do
{ {
var json = Diagnostics.DiagnosticsOutput(); var json = GetDiagnosticsOutput();
if (json == null)
return;
//Console.WriteLine(json.ToString(Formatting.Indented)); //Console.WriteLine(json.ToString(Formatting.Indented));
@@ -337,21 +391,27 @@ namespace TimberWinR.TestGenerator
private static JObject ShutdownTimberWinR() private static JObject ShutdownTimberWinR()
{ {
_timberWinR.Shutdown(); if (_timberWinR != null)
{
// Cancel any/all other threads
_cancellationTokenSource.Cancel();
// Cancel any/all other threads _timberWinR.Shutdown();
_cancellationTokenSource.Cancel();
var json = Diagnostics.DiagnosticsOutput(); var json = Diagnostics.DiagnosticsOutput();
LogManager.GetCurrentClassLogger() LogManager.GetCurrentClassLogger()
.Info("Average CPU Usage: {0}%, Average RAM Usage: {1}MB, Max CPU: {2}%, Max Mem: {3}MB", _avgCpuUsage, _avgMemUsage, _maxCpuUsage, _maxMemUsage); .Info("Average CPU Usage: {0}%, Average RAM Usage: {1}MB, Max CPU: {2}%, Max Mem: {3}MB",
_avgCpuUsage, _avgMemUsage, _maxCpuUsage, _maxMemUsage);
LogManager.GetCurrentClassLogger().Info(json.ToString()); LogManager.GetCurrentClassLogger().Info(json.ToString());
Diagnostics.Shutdown(); Diagnostics.Shutdown();
return json; return json;
}
return new JObject();
} }
static void StartTimberWinR(string configFile, string logLevel, string logFileDir, bool enableLiveMonitor) static void StartTimberWinR(string configFile, string logLevel, string logFileDir, bool enableLiveMonitor)
@@ -363,7 +423,7 @@ namespace TimberWinR.TestGenerator
} }
private static void TimberWinROnOnConfigurationProcessed(Configuration configuration) private static void TimberWinROnOnConfigurationProcessed(Configuration configuration)
{ {
if (!string.IsNullOrEmpty(Options.RedisHost) && configuration.RedisOutputs != null && configuration.RedisOutputs.Count() > 0) if (!string.IsNullOrEmpty(Options.RedisHost) && configuration.RedisOutputs != null && configuration.RedisOutputs.Count() > 0)
{ {
foreach (var ro in configuration.RedisOutputs) foreach (var ro in configuration.RedisOutputs)

View File

@@ -108,6 +108,24 @@
<Content Include="results3.json"> <Content Include="results3.json">
<CopyToOutputDirectory>PreserveNewest</CopyToOutputDirectory> <CopyToOutputDirectory>PreserveNewest</CopyToOutputDirectory>
</Content> </Content>
<Content Include="test4-tw.json">
<CopyToOutputDirectory>PreserveNewest</CopyToOutputDirectory>
</Content>
<Content Include="results4.json">
<CopyToOutputDirectory>PreserveNewest</CopyToOutputDirectory>
</Content>
<Content Include="test4.json">
<CopyToOutputDirectory>PreserveNewest</CopyToOutputDirectory>
</Content>
<Content Include="test5-twconfig.json">
<CopyToOutputDirectory>PreserveNewest</CopyToOutputDirectory>
</Content>
<Content Include="test5.json">
<CopyToOutputDirectory>PreserveNewest</CopyToOutputDirectory>
</Content>
<Content Include="results5.json">
<CopyToOutputDirectory>PreserveNewest</CopyToOutputDirectory>
</Content>
</ItemGroup> </ItemGroup>
<ItemGroup> <ItemGroup>
<ProjectReference Include="..\TimberWinR\TimberWinR.csproj"> <ProjectReference Include="..\TimberWinR\TimberWinR.csproj">

View File

@@ -1,4 +1,6 @@
using System.Threading; using System.Security.Cryptography;
using System.Threading;
using Newtonsoft.Json;
using Newtonsoft.Json.Linq; using Newtonsoft.Json.Linq;
using NLog; using NLog;
using NLog.Config; using NLog.Config;
@@ -24,7 +26,7 @@ namespace TimberWinR.TestGenerator
NumMessages = 100; NumMessages = 100;
Port = 6379; Port = 6379;
Host = "localhost"; Host = "localhost";
SleepTimeMilliseconds = 10; SleepTimeMilliseconds = 1;
} }
} }
@@ -48,16 +50,37 @@ namespace TimberWinR.TestGenerator
{ {
JObject o = new JObject JObject o = new JObject
{ {
{"Application", "udp-generator"}, {"Application", "udp-generator"},
{"Host", hostName}, {"Executable", "VP.Common.SvcFrm.Services.Host, Version=29.7.0.0, Culture=neutral, PublicKeyToken=null"},
{"RenderedMessage", "Responding to RequestSchedule message from 10.1.230.36 with Ack because: PRJ byte array is null."},
{"Team", "Manufacturing Software"},
{"RecordNumber", i},
{"Host", hostName},
{"UtcTimestamp", DateTime.UtcNow.ToString("o")}, {"UtcTimestamp", DateTime.UtcNow.ToString("o")},
{"Type", "udp"}, {"Type", "VP.Fulfillment.Direct.Initialization.LogWrapper"},
{"Message", "Testgenerator udp message " + DateTime.UtcNow.ToString("o")}, {"Message", "Testgenerator udp message " + DateTime.UtcNow.ToString("o")},
{"Index", "logstash"} {"Index", "logstash"}
}; };
byte[] sendbuf = Encoding.UTF8.GetBytes(o.ToString());
string hashedString = "";
foreach(var key in o)
{
hashedString += key.ToString();
}
var source = ASCIIEncoding.ASCII.GetBytes(hashedString);
var md5 = new MD5CryptoServiceProvider().ComputeHash(source);
var hash = string.Concat(md5.Select(x => x.ToString("X2")));
o["md5"] = hash;
byte[] sendbuf = Encoding.UTF8.GetBytes(o.ToString(Formatting.None));
IPEndPoint ep = new IPEndPoint(broadcast, parms.Port); IPEndPoint ep = new IPEndPoint(broadcast, parms.Port);
s.SendTo(sendbuf, ep); s.SendTo(sendbuf, ep);
if (i % 1000 == 0)
LogManager.GetCurrentClassLogger().Info("Sent {0} of {1} messages", i, parms.NumMessages);
Thread.Sleep(parms.SleepTimeMilliseconds); Thread.Sleep(parms.SleepTimeMilliseconds);
} }

View File

@@ -36,7 +36,7 @@
"batch_count": 500, "batch_count": 500,
"threads": 2, "threads": 2,
"host": [ "host": [
"tstlexiceapp006.vistaprint.svc" "tstlexiceapp006.mycompany.svc"
] ]
} }
] ]

View File

@@ -5,14 +5,14 @@
"taillog": { "taillog": {
"test1: message sent count": "[messages] == 7404", "test1: message sent count": "[messages] == 7404",
"test2: average cpu": "[avgCpuUsage] <= 30", "test2: average cpu": "[avgCpuUsage] <= 30",
"test3: maximum memory": "[maxMemUsage] <= 20" "test3: maximum memory": "[maxMemUsage] <= 30"
} }
}, },
{ {
"udp": { "udp": {
"test1: message sent count": "[messages] == 1234", "test1: message sent count": "[messages] == 1234",
"test2: average cpu": "[avgCpuUsage] <= 30", "test2: average cpu": "[avgCpuUsage] <= 30",
"test3: maximum memory": "[maxMemUsage] <= 20" "test3: maximum memory": "[maxMemUsage] <= 30"
} }
} }
] ]

View File

@@ -5,14 +5,14 @@
"taillog": { "taillog": {
"test1: message sent count": "[messages] == 7404", "test1: message sent count": "[messages] == 7404",
"test2: average cpu": "[avgCpuUsage] <= 30", "test2: average cpu": "[avgCpuUsage] <= 30",
"test3: maximum memory": "[maxMemUsage] <= 15" "test3: maximum memory": "[maxMemUsage] <= 20"
} }
}, },
{ {
"udp": { "udp": {
"test1: message sent count": "[messages] == 1234", "test1: message sent count": "[messages] == 1234",
"test2: average cpu": "[avgCpuUsage] <= 30", "test2: average cpu": "[avgCpuUsage] <= 30",
"test3: maximum memory": "[maxMemUsage] <= 15" "test3: maximum memory": "[maxMemUsage] <= 20"
} }
} }
] ]

View File

@@ -0,0 +1,20 @@
{
"Results": {
"Inputs": [
{
"taillog": {
"test1: message sent count": "[messages] == 7404",
"test2: average cpu": "[avgCpuUsage] <= 30",
"test3: maximum memory": "[maxMemUsage] <= 15"
}
},
{
"tcp": {
"test4: message sent count": "[messages] == 1234",
"test5: average cpu": "[avgCpuUsage] <= 30",
"test6: maximum memory": "[maxMemUsage] <= 15"
}
}
]
}
}

View File

@@ -0,0 +1,13 @@
{
"Results": {
"Inputs": [
{
"udp": {
"test1: message sent count": "[messages] == 80000",
"test2: average cpu": "[avgCpuUsage] <= 30",
"test3: maximum memory": "[maxMemUsage] <= 30"
}
}
]
}
}

View File

@@ -1,14 +1,14 @@
{ {
"TimberWinR": { "TimberWinR": {
"Inputs": { "Inputs": {
"Udp": [ "Udp": [
{ {
"_comment": "Output from NLog", "_comment": "Output from NLog",
"port": 5140 "port": 5140
} }
], ],
"TailFiles": [ "TailFiles": [
{ {
"interval": 5, "interval": 5,
"logSource": "log files", "logSource": "log files",
"location": "*.jlog", "location": "*.jlog",
@@ -25,6 +25,11 @@
"" ""
], ],
"drop": "true" "drop": "true"
},
"json": {
"type": "Win32-TailFile",
"source": "Text",
"promote": "Text"
} }
} }
], ],
@@ -34,9 +39,9 @@
"_comment": "Change the host to your Redis instance", "_comment": "Change the host to your Redis instance",
"port": 6379, "port": 6379,
"batch_count": 500, "batch_count": 500,
"threads": 2, "threads": 1,
"host": [ "host": [
"tstlexiceapp006.vistaprint.svc" "tstlexiceapp006.mycompany.svc"
] ]
} }
] ]

View File

@@ -1,13 +1,14 @@
{ {
"test": "Test 1", "test": "Test 1",
"arguments": { "arguments": {
"--start": "",
"--testFile": "test1.json", "--testFile": "test1.json",
"--testDir": "test1", "--testDir": "test1",
"--timberWinRConfig": "test1-twconfig.json", "--timberWinRConfig": "test1-twconfig.json",
"--numMessages": 1234, "--numMessages": 1234,
"--logLevel": "debug", "--logLevel": "debug",
"--udp-host": "::1", "--udp-host": "localhost",
"--udp": "5140", "--udp": "5140",
"--jroll": ["r1.jlog", "r2.jlog"], "--jroll": ["r1.jlog", "r2.jlog"],
"--json": ["1.jlog", "2.jlog", "3.jlog", "4.jlog"], "--json": ["1.jlog", "2.jlog", "3.jlog", "4.jlog"],
"--resultsFile": "results1.json" "--resultsFile": "results1.json"

View File

@@ -36,7 +36,7 @@
"batch_count": 500, "batch_count": 500,
"threads": 2, "threads": 2,
"host": [ "host": [
"tstlexiceapp006.vistaprint.svc" "tstlexiceapp006.mycompany.svc"
] ]
} }
] ]

View File

@@ -5,7 +5,7 @@
"--testDir": "test2", "--testDir": "test2",
"--timberWinRConfig": "test2-tw.json", "--timberWinRConfig": "test2-tw.json",
"--numMessages": 1234, "--numMessages": 1234,
"--logLevel": "debug", "--logLevel": "trace",
"--udp": "5140", "--udp": "5140",
"--jroll": ["r1.jlog", "r2.jlog"], "--jroll": ["r1.jlog", "r2.jlog"],
"--json": ["1.jlog", "2.jlog", "3.jlog", "4.jlog"], "--json": ["1.jlog", "2.jlog", "3.jlog", "4.jlog"],

View File

@@ -1,17 +1,17 @@
{ {
"TimberWinR": { "TimberWinR": {
"Inputs": { "Inputs": {
"Tcp": [ "Udp": [
{ {
"_comment": "Output from NLog", "_comment": "Output from NLog",
"port": 5140 "port": 5140
} }
], ],
"Logs": [ "TailFiles": [
{ {
"interval": 5, "interval": 5,
"logSource": "log files", "logSource": "log files",
"location": "*.jlog", "location": "d:\\logs\\sta\\sta.log",
"recurse": -1 "recurse": -1
} }
] ]
@@ -25,6 +25,11 @@
"" ""
], ],
"drop": "true" "drop": "true"
},
"json": {
"type": "Win32-TailFile",
"source": "Text",
"promote": "Text"
} }
} }
], ],
@@ -34,9 +39,9 @@
"_comment": "Change the host to your Redis instance", "_comment": "Change the host to your Redis instance",
"port": 6379, "port": 6379,
"batch_count": 500, "batch_count": 500,
"threads": 2, "threads": 2,
"host": [ "host": [
"tstlexiceapp006.vistaprint.svc" "tstlexiceapp006.mycompany.svc"
] ]
} }
] ]

View File

@@ -1,14 +1,12 @@
{ {
"test": "Test 3", "test": "Test 3",
"arguments": { "arguments": {
"--start": "",
"--testFile": "test3.json", "--testFile": "test3.json",
"--testDir": "test3", "--testDir": "test3",
"--timberWinRConfig": "test3-tw.json", "--timberWinRConfig": "test3-tw.json",
"--numMessages": 1234, "--numMessages": 1234,
"--logLevel": "debug", "--logLevel": "debug",
"--tcp": "5140",
"--jroll": ["r1.jlog", "r2.jlog"],
"--json": ["1.jlog", "2.jlog", "3.jlog", "4.jlog"],
"--resultsFile": "results3.json" "--resultsFile": "results3.json"
} }
} }

View File

@@ -0,0 +1,50 @@
{
"TimberWinR": {
"Inputs": {
"Udp": [
{
"_comment": "Output from NLog",
"port": 5140
}
],
"TailFiles": [
{
"interval": 5,
"logSource": "log files",
"location": "d:\\logs\\sta\\sta.log",
"recurse": -1
}
]
},
"Filters": [
{
"grok": {
"condition": "\"[EventTypeName]\" == \"Information Event\"",
"match": [
"Text",
""
],
"drop": "true"
},
"json": {
"type": "Win32-TailFile",
"source": "Text",
"promote": "Text"
}
}
],
"Outputs": {
"Redis": [
{
"_comment": "Change the host to your Redis instance",
"port": 6379,
"batch_count": 500,
"threads": 2,
"host": [
"tstlexiceapp006.mycompany.svc"
]
}
]
}
}
}

View File

@@ -0,0 +1,11 @@
{
"test": "Test 4",
"arguments": {
"--testFile": "test4.json",
"--testDir": "test4",
"--timberWinRConfig": "test4-tw.json",
"--numMessages": 1234,
"--logLevel": "debug",
"--resultsFile": "results4.json"
}
}

View File

@@ -0,0 +1,36 @@
{
"TimberWinR": {
"Inputs": {
"Udp": [
{
"_comment": "Output from NLog",
"port": 5140,
"add_field": [
"Environment",
"PLANT_TST_TIMBERWINR"
],
"rename": [
"Type",
"type"
]
}
],
"TailFiles": [
{
"interval": 5,
"logSource": "log files",
"location": "*.jlog",
"recurse": -1
}
]
},
"Outputs": {
"File": [
{
"format": "indented",
"file_name": "test5_output.txt"
}
]
}
}
}

View File

@@ -0,0 +1,15 @@
{
"test": "Test 5",
"arguments": {
"--start": "",
"--testFile": "test5.json",
"--testDir": "test5",
"--timberWinRConfig": "test5-twconfig.json",
"--numMessages": 80000,
"--logLevel": "debug",
"--udp-host": "localhost",
"--udp": "5140",
"--udp-rate": 1,
"--resultsFile": "results5.json"
}
}

View File

@@ -77,5 +77,56 @@ namespace TimberWinR.UnitTests
Assert.IsNull(nostuff); Assert.IsNull(nostuff);
Assert.AreEqual(6, jsonInputLine2.Count); Assert.AreEqual(6, jsonInputLine2.Count);
} }
[Test]
public void TestRenameAndAdds()
{
JObject jsonInputLine1 = new JObject
{
{"type", "Win32-FileLog"},
{"ComputerName", "dev.mycompany.net"},
{"Text", "{\"log4net:Username\" : \"NT AUTHORITY\",\"Email\":\"james@example.com\",\"Active\":true,\"CreatedDate\":\"2013-01-20T00:00:00Z\",\"Roles\":[\"User\",\"Admin\"]}"}
};
string jsonFilter = @"{
""TimberWinR"":{
""Filters"":[
{
""json"":{
""type"": ""Win32-FileLog"",
""source"": ""Text"",
""promote"": ""Text"",
""add_field"":[
""test1"",
""value1"",
""test2"",
""value2""
],
""rename"":[
""Active"",
""active"",
""log4net:Username"",
""lusername""
]
}
}]
}
}";
// Positive Tests
Configuration c = Configuration.FromString(jsonFilter);
Json jf = c.Filters.First() as Json;
Assert.IsTrue(jf.Apply(jsonInputLine1));
Assert.AreEqual("NT AUTHORITY", jsonInputLine1["lusername"].ToString());
Assert.AreEqual("True", jsonInputLine1["active"].ToString());
Assert.IsNotNull(jsonInputLine1["test1"]);
Assert.IsNotNull(jsonInputLine1["test2"]);
Assert.AreEqual("value1", jsonInputLine1["test1"].ToString());
Assert.AreEqual("value2", jsonInputLine1["test2"].ToString());
}
} }
} }

View File

@@ -116,4 +116,7 @@ Global
GlobalSection(SolutionProperties) = preSolution GlobalSection(SolutionProperties) = preSolution
HideSolutionNode = FALSE HideSolutionNode = FALSE
EndGlobalSection EndGlobalSection
GlobalSection(Performance) = preSolution
HasPerformanceSessions = true
EndGlobalSection
EndGlobal EndGlobal

View File

@@ -0,0 +1,26 @@
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using Newtonsoft.Json.Linq;
using TimberWinR.Parser;
namespace TimberWinR.Codecs
{
class JsonCodec : ICodec
{
private CodecArguments _codecArguments;
public void Apply(string msg, Inputs.InputListener listener)
{
JObject jobject = JObject.Parse(msg);
listener.AddDefaultFields(jobject);
listener.ProcessJson(jobject);
}
public JsonCodec(CodecArguments args)
{
_codecArguments = args;
}
}
}

View File

@@ -0,0 +1,39 @@
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using Newtonsoft.Json.Linq;
using TimberWinR.Parser;
namespace TimberWinR.Codecs
{
public class PlainCodec : ICodec
{
private CodecArguments _codecArguments;
public void Apply(string msg, Inputs.InputListener listener)
{
JObject json = new JObject();
listener.AddDefaultFields(json);
json["message"] = ExpandField(msg, json);
listener.ProcessJson(json);
}
protected string ExpandField(string fieldName, JObject json)
{
foreach (var token in json.Children())
{
string replaceString = "%{" + token.Path + "}";
fieldName = fieldName.Replace(replaceString, json[token.Path].ToString());
}
return fieldName;
}
public PlainCodec(CodecArguments args)
{
_codecArguments = args;
}
}
}

View File

@@ -58,6 +58,12 @@ namespace TimberWinR
get { return _stdoutOutputs; } get { return _stdoutOutputs; }
} }
private List<FileOutputParameters> _fileOutputs = new List<FileOutputParameters>();
public IEnumerable<FileOutputParameters> FileOutputs
{
get { return _fileOutputs; }
}
private List<TcpParameters> _tcps = new List<TcpParameters>(); private List<TcpParameters> _tcps = new List<TcpParameters>();
public IEnumerable<TcpParameters> Tcps public IEnumerable<TcpParameters> Tcps
{ {
@@ -102,6 +108,12 @@ namespace TimberWinR
get { return _stdins; } get { return _stdins; }
} }
private List<GeneratorParameters> _generators = new List<GeneratorParameters>();
public IEnumerable<GeneratorParameters> Generators
{
get { return _generators; }
}
private List<LogstashFilter> _filters = new List<LogstashFilter>(); private List<LogstashFilter> _filters = new List<LogstashFilter>();
public IEnumerable<LogstashFilter> Filters public IEnumerable<LogstashFilter> Filters
@@ -239,6 +251,8 @@ namespace TimberWinR
c._iisw3clogs.AddRange(x.TimberWinR.Inputs.IISW3CLogs.ToList()); c._iisw3clogs.AddRange(x.TimberWinR.Inputs.IISW3CLogs.ToList());
if (x.TimberWinR.Inputs.Stdins != null) if (x.TimberWinR.Inputs.Stdins != null)
c._stdins.AddRange(x.TimberWinR.Inputs.Stdins.ToList()); c._stdins.AddRange(x.TimberWinR.Inputs.Stdins.ToList());
if (x.TimberWinR.Inputs.Generators != null)
c._generators.AddRange(x.TimberWinR.Inputs.Generators.ToList());
if (x.TimberWinR.Inputs.Logs != null) if (x.TimberWinR.Inputs.Logs != null)
c._logs.AddRange(x.TimberWinR.Inputs.Logs.ToList()); c._logs.AddRange(x.TimberWinR.Inputs.Logs.ToList());
if (x.TimberWinR.Inputs.TailFilesArguments != null) if (x.TimberWinR.Inputs.TailFilesArguments != null)
@@ -257,6 +271,8 @@ namespace TimberWinR
c._elasticsearchOutputs.AddRange(x.TimberWinR.Outputs.Elasticsearch.ToList()); c._elasticsearchOutputs.AddRange(x.TimberWinR.Outputs.Elasticsearch.ToList());
if (x.TimberWinR.Outputs.Stdout != null) if (x.TimberWinR.Outputs.Stdout != null)
c._stdoutOutputs.AddRange(x.TimberWinR.Outputs.Stdout.ToList()); c._stdoutOutputs.AddRange(x.TimberWinR.Outputs.Stdout.ToList());
if (x.TimberWinR.Outputs.File != null)
c._fileOutputs.AddRange(x.TimberWinR.Outputs.File.ToList());
} }
if (x.TimberWinR.Filters != null) if (x.TimberWinR.Filters != null)
@@ -294,6 +310,7 @@ namespace TimberWinR
_redisOutputs = new List<RedisOutputParameters>(); _redisOutputs = new List<RedisOutputParameters>();
_elasticsearchOutputs = new List<ElasticsearchOutputParameters>(); _elasticsearchOutputs = new List<ElasticsearchOutputParameters>();
_stdoutOutputs = new List<StdoutOutputParameters>(); _stdoutOutputs = new List<StdoutOutputParameters>();
_fileOutputs = new List<FileOutputParameters>();
_tcps = new List<TcpParameters>(); _tcps = new List<TcpParameters>();
_udps = new List<UdpParameters>(); _udps = new List<UdpParameters>();
} }

View File

@@ -30,8 +30,6 @@ namespace TimberWinR.Parser
} }
} }
public partial class Grok : LogstashFilter public partial class Grok : LogstashFilter
{ {
public override JObject ToJson() public override JObject ToJson()
@@ -92,35 +90,37 @@ namespace TimberWinR.Parser
return true; return true;
} }
// Test for any true matching condition(s)
private bool Matches(Newtonsoft.Json.Linq.JObject json) private bool Matches(Newtonsoft.Json.Linq.JObject json)
{ {
string field = Match[0]; for (int i = 0; i < Match.Length; i += 2)
string expr = Match[1];
JToken token = null;
if (json.TryGetValue(field, out token))
{ {
string text = token.ToString(); string field = Match[i];
if (!string.IsNullOrEmpty(text)) string expr = Match[i + 1];
JToken token = null;
if (json.TryGetValue(field, out token))
{ {
var resolver = new RegexGrokResolver(); string text = token.ToString();
var pattern = resolver.ResolveToRegex(expr); if (!string.IsNullOrEmpty(text))
var match = Regex.Match(text, pattern);
if (match.Success)
{ {
var regex = new Regex(pattern); var resolver = new RegexGrokResolver();
var namedCaptures = regex.MatchNamedCaptures(text); var pattern = resolver.ResolveToRegex(expr);
foreach (string fieldName in namedCaptures.Keys) var match = Regex.Match(text, pattern);
if (match.Success)
{ {
AddOrModify(json, fieldName, namedCaptures[fieldName]); var regex = new Regex(pattern);
var namedCaptures = regex.MatchNamedCaptures(text);
foreach (string fieldName in namedCaptures.Keys)
{
AddOrModify(json, fieldName, namedCaptures[fieldName]);
}
return true; // Yes!
} }
return true; // Yes!
} }
if (string.IsNullOrEmpty(expr))
return true; // Empty field is no match
} }
if (string.IsNullOrEmpty(expr))
return true; // Empty field is no match
else
return false;
} }
return false; // Not specified is failure return false; // Not specified is failure
} }
@@ -136,7 +136,7 @@ namespace TimberWinR.Parser
AddOrModify(json, fieldName, fieldValue); AddOrModify(json, fieldName, fieldValue);
} }
} }
} }
private void RemoveFields(Newtonsoft.Json.Linq.JObject json) private void RemoveFields(Newtonsoft.Json.Linq.JObject json)
{ {

View File

@@ -3,6 +3,7 @@ using System.Collections.Generic;
using System.IO; using System.IO;
using System.Linq; using System.Linq;
using System.Text; using System.Text;
using System.Threading;
using System.Xml.Linq; using System.Xml.Linq;
using Newtonsoft.Json; using Newtonsoft.Json;
using Newtonsoft.Json.Linq; using Newtonsoft.Json.Linq;
@@ -12,6 +13,8 @@ namespace TimberWinR.Parser
{ {
public partial class Json : LogstashFilter public partial class Json : LogstashFilter
{ {
private long _errorCount;
public Json() public Json()
{ {
RemoveSource = true; RemoveSource = true;
@@ -22,6 +25,7 @@ namespace TimberWinR.Parser
new JProperty("json", new JProperty("json",
new JObject( new JObject(
new JProperty("condition", Condition), new JProperty("condition", Condition),
new JProperty("errors", _errorCount),
new JProperty("source", Source), new JProperty("source", Source),
new JProperty("promote", Source), new JProperty("promote", Source),
new JProperty("target", Target), new JProperty("target", Target),
@@ -88,9 +92,12 @@ namespace TimberWinR.Parser
if (Rename != null && Rename.Length > 0) if (Rename != null && Rename.Length > 0)
{ {
string oldName = ExpandField(Rename[0], json); for (int i = 0; i < Rename.Length; i += 2)
string newName = ExpandField(Rename[1], json); {
RenameProperty(json, oldName, newName); string oldName = ExpandField(Rename[i], json);
string newName = ExpandField(Rename[i+1], json);
RenameProperty(json, oldName, newName);
}
} }
if (RemoveSource) if (RemoveSource)
@@ -102,6 +109,7 @@ namespace TimberWinR.Parser
catch (Exception ex) catch (Exception ex)
{ {
LogManager.GetCurrentClassLogger().Error(ex); LogManager.GetCurrentClassLogger().Error(ex);
Interlocked.Increment(ref _errorCount);
return true; return true;
} }
} }

View File

@@ -0,0 +1,88 @@
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading;
using Newtonsoft.Json;
using Newtonsoft.Json.Linq;
using NLog;
using RestSharp.Extensions;
using TimberWinR.Codecs;
using TimberWinR.Parser;
namespace TimberWinR.Inputs
{
public class GeneratorInput : InputListener
{
public override JObject ToJson()
{
JObject json = new JObject(
new JProperty("message", _params.Message),
new JProperty("messages", _sentMessages),
new JProperty("generator", "enabled"));
return json;
}
private TimberWinR.Parser.GeneratorParameters _params;
private Thread _listenThread;
private ICodec _codec;
private int _sentMessages;
public GeneratorInput(TimberWinR.Parser.GeneratorParameters parameters, CancellationToken cancelToken)
: base(cancelToken, "Win32-InputGen")
{
_params = parameters;
if (_params.CodecArguments != null)
{
switch (_params.CodecArguments.Type)
{
case CodecArguments.CodecType.json:
_codec = new JsonCodec(_params.CodecArguments);
break;
case CodecArguments.CodecType.multiline:
_codec = new Multiline(_params.CodecArguments);
break;
case CodecArguments.CodecType.plain:
_codec = new PlainCodec(_params.CodecArguments);
break;
}
}
_listenThread = new Thread(new ThreadStart(GenerateData));
_listenThread.Start();
}
private void GenerateData()
{
LogManager.GetCurrentClassLogger().Info("Generator Creating {0} Lines", _params.Count);
int numMessages = _params.Count;
// Infinite or until done.
for (int i = 0; (_params.Count == 0 || i < numMessages); i++)
{
if (CancelToken.IsCancellationRequested)
break;
string msg = ToPrintable(_params.Message);
if (_codec != null)
_codec.Apply(msg, this);
else
{
JObject jo = new JObject();
jo["Message"] = msg;
AddDefaultFields(jo);
ProcessJson(jo);
}
Thread.Sleep(_params.Rate);
}
Finished();
}
}
}

View File

@@ -44,6 +44,11 @@ namespace TimberWinR.Inputs
.ToString(); .ToString();
} }
public void SetTypeName(string newType)
{
_typeName = newType;
}
public bool HaveSeenFile(string fileName) public bool HaveSeenFile(string fileName)
{ {
return Files.Contains(fileName); return Files.Contains(fileName);
@@ -96,6 +101,34 @@ namespace TimberWinR.Inputs
LogManager.GetCurrentClassLogger().Error(ex); LogManager.GetCurrentClassLogger().Error(ex);
} }
} }
protected void AddOrModify(JObject json, string fieldName, string fieldValue)
{
if (json[fieldName] == null)
json.Add(fieldName, fieldValue);
else
json[fieldName] = fieldValue;
}
protected void RenameProperty(JObject json, string oldName, string newName)
{
JToken token = json[oldName];
if (token != null)
{
json.Add(newName, token.DeepClone());
json.Remove(oldName);
}
}
protected string ExpandField(string fieldName, JObject json)
{
foreach (var token in json.Children().ToList())
{
string replaceString = "%{" + token.Path + "}";
fieldName = fieldName.Replace(replaceString, json[token.Path].ToString());
}
return fieldName;
}
protected void EnsureRollingCaught() protected void EnsureRollingCaught()
{ {
@@ -123,6 +156,7 @@ namespace TimberWinR.Inputs
} }
} }
public virtual void AddDefaultFields(JObject json) public virtual void AddDefaultFields(JObject json)
{ {
if (json["type"] == null) if (json["type"] == null)

View File

@@ -77,6 +77,7 @@ namespace TimberWinR.Inputs
var fi = new FileInfo(logName); var fi = new FileInfo(logName);
de.FileName = logName; de.FileName = logName;
de.LogFileExists = fi.Exists; de.LogFileExists = fi.Exists;
de.Previous = "";
de.NewFile = true; de.NewFile = true;
de.ProcessedFile = false; de.ProcessedFile = false;
de.LastPosition = fi.Length; de.LastPosition = fi.Length;
@@ -101,8 +102,10 @@ namespace TimberWinR.Inputs
var creationTime = fi.CreationTimeUtc; var creationTime = fi.CreationTimeUtc;
if (dbe.LogFileExists && creationTime != dbe.CreationTimeUtc) if (dbe.LogFileExists && creationTime != dbe.CreationTimeUtc)
dbe.NewFile = true; {
dbe.NewFile = true;
dbe.Previous = "";
}
dbe.CreationTimeUtc = creationTime; dbe.CreationTimeUtc = creationTime;
return dbe; return dbe;
@@ -133,7 +136,8 @@ namespace TimberWinR.Inputs
public static void Roll(LogsFileDatabaseEntry dbe) public static void Roll(LogsFileDatabaseEntry dbe)
{ {
dbe.ProcessedFile = false; dbe.ProcessedFile = false;
dbe.LastPosition = 0; dbe.LastPosition = 0;
dbe.Previous = "";
Instance.UpdateEntry(dbe, 0); Instance.UpdateEntry(dbe, 0);
dbe.NewFile = true; dbe.NewFile = true;
} }
@@ -268,6 +272,7 @@ namespace TimberWinR.Inputs
{ {
Interlocked.Increment(ref _linesProcessed); Interlocked.Increment(ref _linesProcessed);
} }
public string Previous { get; set; }
} }
} }

View File

@@ -53,6 +53,9 @@ namespace TimberWinR.Inputs
if (_codecArguments != null && _codecArguments.Type == CodecArguments.CodecType.multiline) if (_codecArguments != null && _codecArguments.Type == CodecArguments.CodecType.multiline)
_codec = new Multiline(_codecArguments); _codec = new Multiline(_codecArguments);
if (!string.IsNullOrEmpty(arguments.Type))
SetTypeName(arguments.Type);
_receivedMessages = 0; _receivedMessages = 0;
_arguments = arguments; _arguments = arguments;
_pollingIntervalInSeconds = arguments.Interval; _pollingIntervalInSeconds = arguments.Interval;

View File

@@ -30,23 +30,27 @@ namespace TimberWinR.Inputs
private int _pollingIntervalInSeconds; private int _pollingIntervalInSeconds;
private TimberWinR.Parser.TailFileArguments _arguments; private TimberWinR.Parser.TailFileArguments _arguments;
private long _receivedMessages; private long _receivedMessages;
private long _errorCount;
private CodecArguments _codecArguments; private CodecArguments _codecArguments;
private ICodec _codec; private ICodec _codec;
public bool Stop { get; set; } public bool Stop { get; set; }
public TailFileListener(TimberWinR.Parser.TailFileArguments arguments, CancellationToken cancelToken) public TailFileListener(TimberWinR.Parser.TailFileArguments arguments,
CancellationToken cancelToken)
: base(cancelToken, "Win32-TailLog") : base(cancelToken, "Win32-TailLog")
{ {
Stop = false; Stop = false;
EnsureRollingCaught(); EnsureRollingCaught();
_codecArguments = arguments.CodecArguments; _codecArguments = arguments.CodecArguments;
if (_codecArguments != null && _codecArguments.Type == CodecArguments.CodecType.multiline) if (_codecArguments != null && _codecArguments.Type == CodecArguments.CodecType.multiline)
_codec = new Multiline(_codecArguments); _codec = new Multiline(_codecArguments);
if (!string.IsNullOrEmpty(arguments.Type))
SetTypeName(arguments.Type);
_receivedMessages = 0; _receivedMessages = 0;
_arguments = arguments; _arguments = arguments;
_pollingIntervalInSeconds = arguments.Interval; _pollingIntervalInSeconds = arguments.Interval;
@@ -60,7 +64,9 @@ namespace TimberWinR.Inputs
public override void Shutdown() public override void Shutdown()
{ {
LogManager.GetCurrentClassLogger().Info("{0}: Shutting Down {1} for {2}", Thread.CurrentThread.ManagedThreadId, InputType, _arguments.Location); LogManager.GetCurrentClassLogger()
.Info("{0}: Shutting Down {1} for {2}", Thread.CurrentThread.ManagedThreadId, InputType,
_arguments.Location);
Stop = true; Stop = true;
base.Shutdown(); base.Shutdown();
} }
@@ -71,6 +77,7 @@ namespace TimberWinR.Inputs
new JProperty("taillog", new JProperty("taillog",
new JObject( new JObject(
new JProperty("messages", _receivedMessages), new JProperty("messages", _receivedMessages),
new JProperty("errors", _errorCount),
new JProperty("type", InputType), new JProperty("type", InputType),
new JProperty("location", _arguments.Location), new JProperty("location", _arguments.Location),
new JProperty("logSource", _arguments.LogSource), new JProperty("logSource", _arguments.LogSource),
@@ -103,66 +110,100 @@ namespace TimberWinR.Inputs
private void TailFileContents(string fileName, long offset, LogsFileDatabaseEntry dbe) private void TailFileContents(string fileName, long offset, LogsFileDatabaseEntry dbe)
{ {
using (StreamReader reader = new StreamReader(new FileStream(fileName, FileMode.Open, FileAccess.Read, FileShare.ReadWrite))) const int bufSize = 16535;
long prevLen = offset;
FileInfo fi = new FileInfo(fileName);
if (!fi.Exists)
return;
LogManager.GetCurrentClassLogger().Trace(":{0} Tailing File: {1} as Pos: {2}", Thread.CurrentThread.ManagedThreadId, fileName, prevLen);
using (var stream = new FileStream(fi.FullName, FileMode.Open, FileAccess.Read, FileShare.Delete | FileShare.ReadWrite))
{ {
//start at the end of the file stream.Seek(prevLen, SeekOrigin.Begin);
long lastMaxOffset = offset;
//if the file size has not changed, idle char[] buffer = new char[bufSize];
if (reader.BaseStream.Length == lastMaxOffset) StringBuilder current = new StringBuilder();
return; using (StreamReader sr = new StreamReader(stream))
//seek to the last max offset
LogManager.GetCurrentClassLogger().Trace("{0}: File: {1} Seek to: {2}", Thread.CurrentThread.ManagedThreadId, fileName, lastMaxOffset);
reader.BaseStream.Seek(lastMaxOffset, SeekOrigin.Begin);
//read out of the file until the EOF
string line = "";
long lineOffset = 0;
while ((line = reader.ReadLine()) != null)
{ {
if (string.IsNullOrEmpty(line)) int nRead;
continue; do
long index = lastMaxOffset + lineOffset;
string text = line;
string logFileName = fileName;
var json = new JObject();
if (json["logSource"] == null)
{ {
if (string.IsNullOrEmpty(_arguments.LogSource)) // Read a buffered amount
json.Add(new JProperty("logSource", fileName)); nRead = sr.ReadBlock(buffer, 0, bufSize);
else for (int i = 0; i < nRead; ++i)
json.Add(new JProperty("logSource", _arguments.LogSource)); {
// We need the terminator!
if (buffer[i] == '\n' || buffer[i] == '\r')
{
if (current.Length > 0)
{
string line = string.Concat(dbe.Previous, current);
var json = new JObject();
if (json["logSource"] == null)
{
if (string.IsNullOrEmpty(_arguments.LogSource))
json.Add(new JProperty("logSource", fileName));
else
json.Add(new JProperty("logSource", _arguments.LogSource));
}
//LogManager.GetCurrentClassLogger().Debug(":{0} File: {1}:{2} {3}", Thread.CurrentThread.ManagedThreadId, fileName, dbe.LinesProcessed, line);
// We've processed the partial input
dbe.Previous = "";
json["Text"] = line;
json["Index"] = dbe.LinesProcessed;
json["LogFileName"] = fileName;
if (_codecArguments != null && _codecArguments.Type == CodecArguments.CodecType.multiline)
{
try
{
_codec.Apply(line, this);
Interlocked.Increment(ref _receivedMessages);
dbe.IncrementLineCount();
}
catch (Exception ex)
{
Interlocked.Increment(ref _errorCount);
LogManager.GetCurrentClassLogger().ErrorException("Filter Error", ex);
}
}
else
{
try
{
ProcessJson(json);
dbe.IncrementLineCount();
Interlocked.Increment(ref _receivedMessages);
LogsFileDatabase.Update(dbe, true, sr.BaseStream.Position);
}
catch (Exception ex)
{
Interlocked.Increment(ref _errorCount);
LogManager.GetCurrentClassLogger().ErrorException("Process Error", ex);
}
}
}
current = new StringBuilder();
}
else // Copy character into the buffer
{
current.Append(buffer[i]);
}
}
} while (nRead > 0);
// We didn't encounter the newline, so save it.
if (current.Length > 0)
{
dbe.Previous = current.ToString();
} }
json["Text"] = line;
json["Index"] = index;
json["LogFileName"] = fileName;
if (_codecArguments != null && _codecArguments.Type == CodecArguments.CodecType.multiline)
{
_codec.Apply(line, this);
Interlocked.Increment(ref _receivedMessages);
dbe.IncrementLineCount();
}
else
{
ProcessJson(json);
Interlocked.Increment(ref _receivedMessages);
dbe.IncrementLineCount();
//LogManager.GetCurrentClassLogger().Info("{0}: File: {1} {2} {3}", Thread.CurrentThread.ManagedThreadId, fileName, dbe.LinesProcessed, line);
}
lineOffset += line.Length;
} }
//update the last max offset }
lastMaxOffset = reader.BaseStream.Position;
LogsFileDatabase.Update(dbe, true, lastMaxOffset);
}
} }
// One thread for each kind of file to watch, i.e. "*.log,*.txt" would be two separate // One thread for each kind of file to watch, i.e. "*.log,*.txt" would be two separate
// threads. // threads.
@@ -202,12 +243,13 @@ namespace TimberWinR.Inputs
{ {
LogManager.GetCurrentClassLogger().Debug(":{0} Starting Thread Tailing File: {1}", Thread.CurrentThread.ManagedThreadId, dbe.FileName); LogManager.GetCurrentClassLogger().Debug(":{0} Starting Thread Tailing File: {1}", Thread.CurrentThread.ManagedThreadId, dbe.FileName);
LogsFileDatabase.Update(dbe, false, dbe.LastPosition); LogsFileDatabase.Update(dbe, false, dbe.LastPosition);
SaveVisitedFileName(fileName);
Task.Factory.StartNew(() => TailFileWatcher(fileName)); Task.Factory.StartNew(() => TailFileWatcher(fileName));
} }
else if (!isWildcardPattern) else if (!isWildcardPattern)
{ {
FileInfo fi = new FileInfo(dbe.FileName); FileInfo fi = new FileInfo(dbe.FileName);
SaveVisitedFileName(fileName);
//LogManager.GetCurrentClassLogger().Info("Located File: {0}, New: {1}", dbe.FileName, dbe.NewFile); //LogManager.GetCurrentClassLogger().Info("Located File: {0}, New: {1}", dbe.FileName, dbe.NewFile);
long length = fi.Length; long length = fi.Length;
@@ -219,7 +261,7 @@ namespace TimberWinR.Inputs
LogsFileDatabase.Roll(dbe); LogsFileDatabase.Roll(dbe);
} }
// Log has rolled or this is a file we are seeing for the first time. // Log has rolled or this is a file we are seeing for the first time.
bool processWholeFile = logHasRolled || !dbe.ProcessedFile; bool processWholeFile = logHasRolled || !dbe.ProcessedFile || dbe.NewFile;
if (processWholeFile) if (processWholeFile)
{ {
LogsFileDatabase.Update(dbe, true, 0); LogsFileDatabase.Update(dbe, true, 0);
@@ -229,7 +271,7 @@ namespace TimberWinR.Inputs
else else
{ {
TailFileContents(dbe.FileName, dbe.LastPosition, dbe); TailFileContents(dbe.FileName, dbe.LastPosition, dbe);
} }
} }
} }
} }
@@ -273,7 +315,7 @@ namespace TimberWinR.Inputs
} }
} }
Finished(); Finished();
} }
} }
} }

View File

@@ -16,7 +16,7 @@ namespace TimberWinR.Inputs
private Thread _listenThreadV4; private Thread _listenThreadV4;
private Thread _listenThreadV6; private Thread _listenThreadV6;
private readonly int _port; private readonly int _port;
private TimberWinR.Parser.TcpParameters _arguments;
private long _receivedMessages; private long _receivedMessages;
private long _errorCount; private long _errorCount;
@@ -32,15 +32,19 @@ namespace TimberWinR.Inputs
return json; return json;
} }
public TcpInputListener(CancellationToken cancelToken, int port = 5140) public TcpInputListener(TimberWinR.Parser.TcpParameters arguments, CancellationToken cancelToken, int port = 5140)
: base(cancelToken, "Win32-Tcp") : base(cancelToken, "Win32-Tcp")
{ {
_port = port; _port = port;
_arguments = arguments;
LogManager.GetCurrentClassLogger().Info("Tcp Input(v4/v6) on Port {0} Ready", _port); LogManager.GetCurrentClassLogger().Info("Tcp Input(v4/v6) on Port {0} Ready", _port);
_receivedMessages = 0; if (!string.IsNullOrEmpty(arguments.Type))
SetTypeName(arguments.Type);
_receivedMessages = 0;
_tcpListenerV6 = new System.Net.Sockets.TcpListener(IPAddress.IPv6Any, port); _tcpListenerV6 = new System.Net.Sockets.TcpListener(IPAddress.IPv6Any, port);
_tcpListenerV4 = new System.Net.Sockets.TcpListener(IPAddress.Any, port); _tcpListenerV4 = new System.Net.Sockets.TcpListener(IPAddress.Any, port);
@@ -91,6 +95,33 @@ namespace TimberWinR.Inputs
} }
} }
//
// Renames, and AddFields
//
private void ApplyFilters(JObject json)
{
if (_arguments.Renames != null)
{
for (int i = 0; i < _arguments.Renames.Length; i += 2)
{
var oldName = ExpandField(_arguments.Renames[i], json);
var newName = ExpandField(_arguments.Renames[i + 1], json);
RenameProperty(json, oldName, newName);
}
}
if (_arguments.AddFields != null)
{
for (int i = 0; i < _arguments.AddFields.Length; i += 2)
{
var fieldName = ExpandField(_arguments.AddFields[i], json);
var fieldValue = ExpandField(_arguments.AddFields[i + 1], json);
AddOrModify(json, fieldName, fieldValue);
}
}
}
private void HandleNewClient(object client) private void HandleNewClient(object client)
{ {
var tcpClient = (TcpClient)client; var tcpClient = (TcpClient)client;
@@ -109,6 +140,7 @@ namespace TimberWinR.Inputs
try try
{ {
JObject json = JObject.Load(reader); JObject json = JObject.Load(reader);
ApplyFilters(json);
ProcessJson(json); ProcessJson(json);
Interlocked.Increment(ref _receivedMessages); Interlocked.Increment(ref _receivedMessages);
} }

View File

@@ -1,49 +1,59 @@
using System; using System;
using System.IO; using System.Collections.Concurrent;
using System.Text; using System.Text;
using System.Text.RegularExpressions;
using System.Threading; using System.Threading;
using System.Net; using System.Net;
using System.Net.Sockets; using System.Net.Sockets;
using Newtonsoft.Json;
using Newtonsoft.Json.Linq; using Newtonsoft.Json.Linq;
using NLog; using NLog;
namespace TimberWinR.Inputs namespace TimberWinR.Inputs
{ {
public class UdpInputListener : InputListener public class UdpInputListener : InputListener
{ {
private UdpClient _udpListenerV4; private UdpClient _udpListenerV4;
private readonly Thread _listenThreadV4; private IPEndPoint _udpEndpointV4;
private readonly BlockingCollection<byte[]> _unprocessedRawData;
private readonly Thread _rawDataProcessingThread;
private readonly int _port; private readonly int _port;
private long _receivedMessages; private long _receivedMessages;
private long _parsedErrors; private long _parseErrors;
private long _receiveErrors;
private long _parsedMessages;
private TimberWinR.Parser.UdpParameters _arguments;
public override JObject ToJson() public override JObject ToJson()
{ {
JObject json = new JObject( var json =
new JProperty("udp", new JObject(new JProperty("udp",
new JObject( new JObject(new JProperty("port", _port),
new JProperty("port", _port), new JProperty("receive_errors", _receiveErrors),
new JProperty("errors", _parsedErrors), new JProperty("parse_errors", _parseErrors),
new JProperty("messages", _receivedMessages) new JProperty("messages", _receivedMessages),
))); new JProperty("parsed_messages", _parsedMessages),
new JProperty("unprocessed_messages", _unprocessedRawData.Count))));
return json; return json;
} }
public UdpInputListener(CancellationToken cancelToken, int port = 5140) public UdpInputListener(TimberWinR.Parser.UdpParameters arguments, CancellationToken cancelToken, int port = 5140) : base(cancelToken, "Win32-Udp")
: base(cancelToken, "Win32-Udp")
{ {
_port = port; _port = port;
_receivedMessages = 0; _receivedMessages = 0;
_arguments = arguments;
_listenThreadV4 = new Thread(StartListener); if (!string.IsNullOrEmpty(arguments.Type))
_listenThreadV4.Start(); SetTypeName(arguments.Type);
// setup raw data processor
_unprocessedRawData = new BlockingCollection<byte[]>();
_rawDataProcessingThread = new Thread(ProcessDataLoop) { Name = "Win32-Udp-DataProcessor"};
_rawDataProcessingThread.Start();
// start listing to udp port
StartListener();
} }
public override void Shutdown() public override void Shutdown()
{ {
LogManager.GetCurrentClassLogger().Info("Shutting Down {0}", InputType); LogManager.GetCurrentClassLogger().Info("Shutting Down {0}", InputType);
@@ -51,64 +61,141 @@ namespace TimberWinR.Inputs
// close UDP listeners, which will end the listener threads // close UDP listeners, which will end the listener threads
_udpListenerV4.Close(); _udpListenerV4.Close();
// wait for completion of the threads
_listenThreadV4.Join();
base.Shutdown(); base.Shutdown();
} }
private void StartListener() private void StartListener()
{ {
var groupV4 = new IPEndPoint(IPAddress.Any, _port); _udpEndpointV4 = new IPEndPoint(IPAddress.Any, _port);
_udpListenerV4 = new UdpClient(_port); // setup listener
_udpListenerV4 = new UdpClient(_port);
LogManager.GetCurrentClassLogger().Info("Udp Input on Port {0} Ready", groupV4);
// start listening on UDP port
StartReceiving();
// all started; log details
LogManager.GetCurrentClassLogger().Info("Udp Input on Port {0} Ready", _udpEndpointV4);
}
private void StartReceiving()
{
if (!CancelToken.IsCancellationRequested)
_udpListenerV4.BeginReceive(DataReceived, null);
}
private void DataReceived(IAsyncResult result)
{
if (CancelToken.IsCancellationRequested)
{
_unprocessedRawData.CompleteAdding();
return;
}
string lastMessage = "";
try try
{ {
while (!CancelToken.IsCancellationRequested) byte[] bytes = _udpListenerV4.EndReceive(result, ref _udpEndpointV4);
{ Interlocked.Increment(ref _receivedMessages);
try StartReceiving();
{ _unprocessedRawData.Add(bytes);
byte[] bytes = _udpListenerV4.Receive(ref groupV4); }
var data = Encoding.UTF8.GetString(bytes, 0, bytes.Length); catch (SocketException)
lastMessage = data; {
var json = JObject.Parse(data); LogManager.GetCurrentClassLogger().Info("Socked exception. Ending UDP Listener.");
ProcessJson(json); _unprocessedRawData.CompleteAdding();
Interlocked.Increment(ref _receivedMessages); }
} catch (ObjectDisposedException)
catch(ArgumentException aex) {
{ LogManager.GetCurrentClassLogger().Info("Object disposed. Ending UDP Listener");
LogManager.GetCurrentClassLogger().Error(aex); _unprocessedRawData.CompleteAdding();
break;
}
catch(SocketException)
{
break;
}
catch (Exception ex)
{
var jex1 = LogErrors.LogException(string.Format("Invalid JSON: {0}", lastMessage), ex);
if (jex1 != null)
ProcessJson(jex1);
LogManager.GetCurrentClassLogger().Warn("Bad JSON: {0}", lastMessage);
LogManager.GetCurrentClassLogger().Warn(ex);
Interlocked.Increment(ref _parsedErrors);
}
}
_udpListenerV4.Close();
} }
catch (Exception ex) catch (Exception ex)
{ {
if (!CancelToken.IsCancellationRequested) LogManager.GetCurrentClassLogger().Warn("Error while receiving data.", ex);
LogManager.GetCurrentClassLogger().Error(ex);
Interlocked.Increment(ref _receiveErrors);
StartReceiving();
}
}
private void ProcessDataLoop()
{
while (!_unprocessedRawData.IsCompleted)
{
try
{
ProcessData(_unprocessedRawData.Take());
}
catch (OperationCanceledException)
{
// we are shutting down.
break;
}
catch (InvalidOperationException)
{
// when the collection is marked as completed
break;
}
catch (Exception ex)
{
LogManager.GetCurrentClassLogger().ErrorException("Error while processing data", ex);
Thread.Sleep(100);
}
} }
Finished(); Finished();
} }
//
// Renames, and AddFields
//
private void ApplyFilters(JObject json)
{
if (_arguments.Renames != null)
{
for (int i=0; i<_arguments.Renames.Length; i += 2)
{
var oldName = ExpandField(_arguments.Renames[i], json);
var newName = ExpandField(_arguments.Renames[i + 1], json);
RenameProperty(json, oldName, newName);
}
}
if (_arguments.AddFields != null)
{
for (int i = 0; i < _arguments.AddFields.Length; i += 2)
{
var fieldName = ExpandField(_arguments.AddFields[i], json);
var fieldValue = ExpandField(_arguments.AddFields[i + 1], json);
AddOrModify(json, fieldName, fieldValue);
}
}
}
private void ProcessData(byte[] bytes)
{
var data = Encoding.UTF8.GetString(bytes, 0, bytes.Length);
try
{
var json = JObject.Parse(data);
ApplyFilters(json);
ProcessJson(json);
_parsedMessages++;
}
catch (Exception ex)
{
var jex1 = LogErrors.LogException(string.Format("Invalid JSON: {0}", data), ex);
if (jex1 != null)
{
ProcessJson(jex1);
}
var msg = string.Format("Bad JSON: {0}", data);
LogManager.GetCurrentClassLogger().Warn(msg, ex);
_parseErrors++;
}
}
} }
} }

View File

@@ -22,9 +22,7 @@ namespace TimberWinR
public class Manager public class Manager
{ {
public Configuration Config { get; set; } public Configuration Config { get; set; }
public List<OutputSender> Outputs { get; set; } public List<OutputSender> Outputs { get; set; }
public List<TcpInputListener> Tcps { get; set; }
public List<TcpInputListener> Udps { get; set; }
public List<InputListener> Listeners { get; set; } public List<InputListener> Listeners { get; set; }
public bool LiveMonitor { get; set; } public bool LiveMonitor { get; set; }
@@ -66,13 +64,13 @@ namespace TimberWinR
public Manager() public Manager()
{ {
LogsFileDatabase.Manager = this; LogsFileDatabase.Manager = this;
} }
public Manager(string jsonConfigFile, string logLevel, string logfileDir, bool liveMonitor, CancellationToken cancelToken, bool processConfiguration = true) public Manager(string jsonConfigFile, string logLevel, string logfileDir, bool liveMonitor, CancellationToken cancelToken, bool processConfiguration = true)
{ {
LogsFileDatabase.Manager = this; LogsFileDatabase.Manager = this;
StartedOn = DateTime.UtcNow; StartedOn = DateTime.UtcNow;
LiveMonitor = liveMonitor; LiveMonitor = liveMonitor;
@@ -119,17 +117,9 @@ namespace TimberWinR
try try
{ {
// Is it a directory? var fi = new FileInfo(jsonConfigFile);
if (Directory.Exists(jsonConfigFile)) if (fi.Exists)
{ {
DirectoryInfo di = new DirectoryInfo(jsonConfigFile);
LogManager.GetCurrentClassLogger().Info("Initialized, Reading Configurations From {0}", di.FullName);
Config = Configuration.FromDirectory(jsonConfigFile, cancelToken, this);
}
else
{
var fi = new FileInfo(jsonConfigFile);
LogManager.GetCurrentClassLogger().Info("Initialized, Reading Configurations From File: {0}", fi.FullName); LogManager.GetCurrentClassLogger().Info("Initialized, Reading Configurations From File: {0}", fi.FullName);
if (!fi.Exists) if (!fi.Exists)
@@ -138,6 +128,12 @@ namespace TimberWinR
LogManager.GetCurrentClassLogger().Info("Initialized, Reading Config: {0}", fi.FullName); LogManager.GetCurrentClassLogger().Info("Initialized, Reading Config: {0}", fi.FullName);
Config = Configuration.FromFile(jsonConfigFile); Config = Configuration.FromFile(jsonConfigFile);
} }
else if (Directory.Exists(jsonConfigFile))
{
DirectoryInfo di = new DirectoryInfo(jsonConfigFile);
LogManager.GetCurrentClassLogger().Info("Initialized, Reading Configurations From {0}", di.FullName);
Config = Configuration.FromDirectory(jsonConfigFile, cancelToken, this);
}
} }
catch (JsonSerializationException jse) catch (JsonSerializationException jse)
{ {
@@ -145,30 +141,30 @@ namespace TimberWinR
} }
catch (Exception ex) catch (Exception ex)
{ {
LogManager.GetCurrentClassLogger().Error(ex); LogManager.GetCurrentClassLogger().Error(ex);
} }
LogManager.GetCurrentClassLogger().Info("Log Directory {0}", logfileDir); LogManager.GetCurrentClassLogger().Info("Log Directory {0}", logfileDir);
LogManager.GetCurrentClassLogger().Info("Logging Level: {0}", LogManager.GlobalThreshold); LogManager.GetCurrentClassLogger().Info("Logging Level: {0}", LogManager.GlobalThreshold);
if (processConfiguration) if (processConfiguration)
{ {
ProcessConfiguration(cancelToken, Config); ProcessConfiguration(cancelToken, Config);
} }
} }
public void Start(CancellationToken cancelToken) public void Start(CancellationToken cancelToken)
{ {
ProcessConfiguration(cancelToken, Config); ProcessConfiguration(cancelToken, Config);
} }
public void ProcessConfiguration(CancellationToken cancelToken, Configuration config) public void ProcessConfiguration(CancellationToken cancelToken, Configuration config)
{ {
// Read the Configuration file // Read the Configuration file
if (config != null) if (config != null)
{ {
if (OnConfigurationProcessed != null) if (OnConfigurationProcessed != null)
OnConfigurationProcessed(config); OnConfigurationProcessed(config);
if (config.RedisOutputs != null) if (config.RedisOutputs != null)
{ {
foreach (var ro in config.RedisOutputs) foreach (var ro in config.RedisOutputs)
@@ -194,6 +190,15 @@ namespace TimberWinR
} }
} }
if (config.FileOutputs != null)
{
foreach (var ro in config.FileOutputs)
{
var output = new FileOutput(this, ro, cancelToken);
Outputs.Add(output);
}
}
foreach (Parser.IISW3CLogParameters iisw3cConfig in config.IISW3C) foreach (Parser.IISW3CLogParameters iisw3cConfig in config.IISW3C)
{ {
var elistner = new IISW3CInputListener(iisw3cConfig, cancelToken); var elistner = new IISW3CInputListener(iisw3cConfig, cancelToken);
@@ -236,7 +241,7 @@ namespace TimberWinR
foreach (var tcp in config.Tcps) foreach (var tcp in config.Tcps)
{ {
var elistner = new TcpInputListener(cancelToken, tcp.Port); var elistner = new TcpInputListener(tcp, cancelToken, tcp.Port);
Listeners.Add(elistner); Listeners.Add(elistner);
foreach (var output in Outputs) foreach (var output in Outputs)
output.Connect(elistner); output.Connect(elistner);
@@ -244,7 +249,7 @@ namespace TimberWinR
foreach (var udp in config.Udps) foreach (var udp in config.Udps)
{ {
var elistner = new UdpInputListener(cancelToken, udp.Port); var elistner = new UdpInputListener(udp, cancelToken, udp.Port);
Listeners.Add(elistner); Listeners.Add(elistner);
foreach (var output in Outputs) foreach (var output in Outputs)
output.Connect(elistner); output.Connect(elistner);
@@ -258,6 +263,15 @@ namespace TimberWinR
output.Connect(elistner); output.Connect(elistner);
} }
foreach (var stdin in config.Generators)
{
var elistner = new GeneratorInput(stdin, cancelToken);
Listeners.Add(elistner);
foreach (var output in Outputs)
output.Connect(elistner);
}
var computerName = System.Environment.MachineName + "." + var computerName = System.Environment.MachineName + "." +
Microsoft.Win32.Registry.LocalMachine.OpenSubKey( Microsoft.Win32.Registry.LocalMachine.OpenSubKey(
@"SYSTEM\CurrentControlSet\services\Tcpip\Parameters") @"SYSTEM\CurrentControlSet\services\Tcpip\Parameters")
@@ -272,7 +286,7 @@ namespace TimberWinR
new JObject( new JObject(
new JProperty("version", new JProperty("version",
Assembly.GetEntryAssembly().GetName().Version.ToString()), Assembly.GetEntryAssembly().GetName().Version.ToString()),
//GetAssemblyByName("TimberWinR.ServiceHost").GetName().Version.ToString()), //GetAssemblyByName("TimberWinR.ServiceHost").GetName().Version.ToString()),
new JProperty("host", computerName), new JProperty("host", computerName),
new JProperty("output", output.Name), new JProperty("output", output.Name),
new JProperty("initialized", DateTime.UtcNow) new JProperty("initialized", DateTime.UtcNow)
@@ -280,7 +294,7 @@ namespace TimberWinR
json.Add(new JProperty("type", "Win32-TimberWinR")); json.Add(new JProperty("type", "Win32-TimberWinR"));
json.Add(new JProperty("host", computerName)); json.Add(new JProperty("host", computerName));
output.Startup(json); output.Startup(json);
} }
} }
} }

View File

@@ -41,7 +41,10 @@ namespace TimberWinR.Outputs
private long _sentMessages; private long _sentMessages;
private long _errorCount; private long _errorCount;
private readonly int _maxQueueSize; private readonly int _maxQueueSize;
private readonly bool _queueOverflowDiscardOldest; private readonly bool _queueOverflowDiscardOldest;
private readonly bool _disablePing;
private readonly int _pingTimeout;
private Parser.ElasticsearchOutputParameters _parameters; private Parser.ElasticsearchOutputParameters _parameters;
public bool Stop { get; set; } public bool Stop { get; set; }
@@ -61,6 +64,11 @@ namespace TimberWinR.Outputs
var settings = new ConnectionSettings(pool) var settings = new ConnectionSettings(pool)
.ExposeRawResponse(); .ExposeRawResponse();
if (_disablePing)
settings.DisablePing();
else if (_pingTimeout != 0)
settings.SetPingTimeout(_pingTimeout);
var client = new ElasticClient(settings); var client = new ElasticClient(settings);
return client; return client;
} }
@@ -70,7 +78,7 @@ namespace TimberWinR.Outputs
{ {
_sentMessages = 0; _sentMessages = 0;
_errorCount = 0; _errorCount = 0;
_parameters = parameters; _parameters = parameters;
_flushSize = parameters.FlushSize; _flushSize = parameters.FlushSize;
_idleFlushTimeSeconds = parameters.IdleFlushTimeInSeconds; _idleFlushTimeSeconds = parameters.IdleFlushTimeInSeconds;
@@ -84,7 +92,8 @@ namespace TimberWinR.Outputs
_numThreads = parameters.NumThreads; _numThreads = parameters.NumThreads;
_maxQueueSize = parameters.MaxQueueSize; _maxQueueSize = parameters.MaxQueueSize;
_queueOverflowDiscardOldest = parameters.QueueOverflowDiscardOldest; _queueOverflowDiscardOldest = parameters.QueueOverflowDiscardOldest;
_disablePing = !parameters.EnablePing;
_pingTimeout = parameters.PingTimeout;
for (int i = 0; i < parameters.NumThreads; i++) for (int i = 0; i < parameters.NumThreads; i++)
{ {
@@ -99,7 +108,7 @@ namespace TimberWinR.Outputs
new JObject( new JObject(
new JProperty("host", string.Join(",", _hosts)), new JProperty("host", string.Join(",", _hosts)),
new JProperty("errors", _errorCount), new JProperty("errors", _errorCount),
new JProperty("sentMmessageCount", _sentMessages), new JProperty("messages", _sentMessages),
new JProperty("queuedMessageCount", _jsonQueue.Count), new JProperty("queuedMessageCount", _jsonQueue.Count),
new JProperty("port", _port), new JProperty("port", _port),
new JProperty("flushSize", _flushSize), new JProperty("flushSize", _flushSize),
@@ -123,6 +132,10 @@ namespace TimberWinR.Outputs
// Force an inital flush // Force an inital flush
DateTime lastFlushTime = DateTime.MinValue; DateTime lastFlushTime = DateTime.MinValue;
LogManager.GetCurrentClassLogger()
.Info("{0}: Elasticsarch Output To {1} Ready", Thread.CurrentThread.ManagedThreadId, string.Join(",", _hosts));
using (var syncHandle = new ManualResetEventSlim()) using (var syncHandle = new ManualResetEventSlim())
{ {
// Execute the query // Execute the query
@@ -188,7 +201,7 @@ namespace TimberWinR.Outputs
} }
GC.Collect(); GC.Collect();
if (!Stop) if (!Stop)
{ {
syncHandle.Wait(TimeSpan.FromMilliseconds(_interval), CancelToken); syncHandle.Wait(TimeSpan.FromMilliseconds(_interval), CancelToken);
} }
} }
@@ -203,6 +216,10 @@ namespace TimberWinR.Outputs
} }
} }
} }
LogManager.GetCurrentClassLogger()
.Info("{0}: Elasticsarch Output To {1} Terminated", Thread.CurrentThread.ManagedThreadId, string.Join(",", _hosts));
} }
// //

140
TimberWinR/Outputs/File.cs Normal file
View File

@@ -0,0 +1,140 @@
using System.IO;
using System.Text;
using Newtonsoft.Json;
using Newtonsoft.Json.Linq;
using NLog;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading;
using System.Threading.Tasks;
namespace TimberWinR.Outputs
{
public class FileOutput : OutputSender
{
private TimberWinR.Manager _manager;
private readonly int _interval;
private readonly object _locker = new object();
private readonly List<JObject> _jsonQueue;
private long _sentMessages;
private Parser.FileOutputParameters _arguments;
public bool Stop { get; set; }
public FileOutput(TimberWinR.Manager manager, Parser.FileOutputParameters arguments, CancellationToken cancelToken)
: base(cancelToken, "File")
{
_arguments = arguments;
_sentMessages = 0;
_manager = manager;
_interval = arguments.Interval;
_jsonQueue = new List<JObject>();
var elsThread = new Task(FileSender, cancelToken);
elsThread.Start();
}
public override JObject ToJson()
{
JObject json = new JObject(
new JProperty("file-output",
new JObject(
new JProperty("queuedMessageCount", _jsonQueue.Count),
new JProperty("sentMessageCount", _sentMessages))));
return json;
}
//
// Pull off messages from the Queue, batch them up and send them all across
//
private void FileSender()
{
using (var syncHandle = new ManualResetEventSlim())
{
var fi = new FileInfo(_arguments.FileName);
if (File.Exists(_arguments.FileName))
File.Delete(_arguments.FileName);
LogManager.GetCurrentClassLogger().Info("File Output Sending To: {0}", fi.FullName);
using (StreamWriter sw = File.AppendText(_arguments.FileName))
{
// Execute the query
while (!Stop)
{
if (!CancelToken.IsCancellationRequested)
{
try
{
JObject[] messages;
lock (_locker)
{
messages = _jsonQueue.Take(_jsonQueue.Count).ToArray();
_jsonQueue.RemoveRange(0, messages.Length);
}
if (messages.Length > 0)
{
try
{
foreach (JObject obj in messages)
{
sw.WriteLine(obj.ToString(_arguments.ToFormat()));
_sentMessages++;
}
}
catch (Exception ex)
{
LogManager.GetCurrentClassLogger().Error(ex);
}
}
if (!Stop)
syncHandle.Wait(TimeSpan.FromMilliseconds(_interval), CancelToken);
}
catch (OperationCanceledException)
{
break;
}
catch (Exception)
{
}
}
}
}
}
}
protected override void MessageReceivedHandler(Newtonsoft.Json.Linq.JObject jsonMessage)
{
if (_manager.Config.Filters != null)
{
if (ApplyFilters(jsonMessage))
return;
}
var message = jsonMessage.ToString();
lock (_locker)
{
_jsonQueue.Add(jsonMessage);
}
}
private bool ApplyFilters(JObject json)
{
bool drop = false;
foreach (var filter in _manager.Config.Filters)
{
if (!filter.Apply(json))
drop = true;
}
return drop;
}
}
}

View File

@@ -79,9 +79,11 @@ namespace TimberWinR.Outputs
// Sample the queue and adjust the batch count if needed (ramp up slowly) // Sample the queue and adjust the batch count if needed (ramp up slowly)
public int UpdateCurrentBatchCount(int queueSize, int currentBatchCount) public int UpdateCurrentBatchCount(int queueSize, int currentBatchCount)
{ {
if (currentBatchCount < _maxBatchCount && currentBatchCount < queueSize && AverageQueueDepth() > currentBatchCount) var avgQueueDepth = AverageQueueDepth();
if (currentBatchCount < _maxBatchCount && currentBatchCount < queueSize && avgQueueDepth > currentBatchCount)
{ {
currentBatchCount += Math.Max(_maxBatchCount / _batchCount, 1); currentBatchCount += Math.Max(avgQueueDepth / _batchCount, _batchCount / 5);
if (currentBatchCount >= _maxBatchCount && !_warnedReachedMax) if (currentBatchCount >= _maxBatchCount && !_warnedReachedMax)
{ {
LogManager.GetCurrentClassLogger().Warn("Maximum Batch Count of {0} reached.", currentBatchCount); LogManager.GetCurrentClassLogger().Warn("Maximum Batch Count of {0} reached.", currentBatchCount);
@@ -314,6 +316,9 @@ namespace TimberWinR.Outputs
{ {
_batchCounter.SampleQueueDepth(_jsonQueue.Count); _batchCounter.SampleQueueDepth(_jsonQueue.Count);
// Re-compute current batch size // Re-compute current batch size
LogManager.GetCurrentClassLogger().Trace("{0}: Average Queue Depth: {1}, Current Length: {2}", Thread.CurrentThread.ManagedThreadId, _batchCounter.AverageQueueDepth(), _jsonQueue.Count);
_currentBatchCount = _batchCounter.UpdateCurrentBatchCount(_jsonQueue.Count, _currentBatchCount); _currentBatchCount = _batchCounter.UpdateCurrentBatchCount(_jsonQueue.Count, _currentBatchCount);
messages = _jsonQueue.Take(_currentBatchCount).ToArray(); messages = _jsonQueue.Take(_currentBatchCount).ToArray();

View File

@@ -12,6 +12,7 @@ using Microsoft.SqlServer.Server;
using Newtonsoft.Json; using Newtonsoft.Json;
using Newtonsoft.Json.Linq; using Newtonsoft.Json.Linq;
using NLog; using NLog;
using NLog.Config;
using TimberWinR.Outputs; using TimberWinR.Outputs;
using System.CodeDom.Compiler; using System.CodeDom.Compiler;
@@ -43,10 +44,8 @@ namespace TimberWinR.Parser
JToken token = json[oldName]; JToken token = json[oldName];
if (token != null) if (token != null)
{ {
json.Add(newName, token.DeepClone());
json.Remove(oldName); json.Remove(oldName);
JToken newToken = json[newName];
if (newToken == null)
json.Add(newName, token);
} }
} }
@@ -263,12 +262,46 @@ namespace TimberWinR.Parser
} }
} }
public class GeneratorParameters : IValidateSchema
{
[JsonProperty(PropertyName = "type")]
public string Type { get; set; }
[JsonProperty(PropertyName = "codec")]
public CodecArguments CodecArguments { get; set; }
[JsonProperty(PropertyName = "message")]
public string Message { get; set; }
[JsonProperty(PropertyName = "count")]
public int Count { get; set; }
[JsonProperty(PropertyName = "rate")]
public int Rate { get; set; }
public void Validate()
{
}
public GeneratorParameters()
{
Count = 0; // Infinity messages
Rate = 10; // Milliseconds
Message = "Hello, world!";
CodecArguments = new CodecArguments();
CodecArguments.Type = CodecArguments.CodecType.plain;
}
}
public class CodecArguments public class CodecArguments
{ {
public enum CodecType public enum CodecType
{ {
singleline, singleline,
multiline multiline,
json,
plain
}; };
public enum WhatType public enum WhatType
@@ -299,10 +332,12 @@ namespace TimberWinR.Parser
public class TailFileArguments : IValidateSchema public class TailFileArguments : IValidateSchema
{ {
[JsonProperty(PropertyName = "type")]
public string Type { get; set; }
[JsonProperty(PropertyName = "location")] [JsonProperty(PropertyName = "location")]
public string Location { get; set; } public string Location { get; set; }
[JsonProperty(PropertyName = "recurse")] [JsonProperty(PropertyName = "recurse")]
public int Recurse { get; set; } public int Recurse { get; set; }
[JsonProperty(PropertyName = "fields")] [JsonProperty(PropertyName = "fields")]
public List<Field> Fields { get; set; } public List<Field> Fields { get; set; }
[JsonProperty(PropertyName = "interval")] [JsonProperty(PropertyName = "interval")]
@@ -329,6 +364,8 @@ namespace TimberWinR.Parser
public class LogParameters : IValidateSchema public class LogParameters : IValidateSchema
{ {
[JsonProperty(PropertyName = "type")]
public string Type { get; set; }
[JsonProperty(PropertyName = "location")] [JsonProperty(PropertyName = "location")]
public string Location { get; set; } public string Location { get; set; }
[JsonProperty(PropertyName = "iCodepage")] [JsonProperty(PropertyName = "iCodepage")]
@@ -365,15 +402,21 @@ namespace TimberWinR.Parser
{ {
[JsonProperty(PropertyName = "port")] [JsonProperty(PropertyName = "port")]
public int Port { get; set; } public int Port { get; set; }
[JsonProperty(PropertyName = "type")]
public string Type { get; set; }
[JsonProperty("add_field")]
public string[] AddFields { get; set; }
[JsonProperty("rename")]
public string[] Renames { get; set; }
public TcpParameters() public TcpParameters()
{ {
Port = 5140; Port = 5140;
Type = "Win32-Tcp";
} }
public void Validate() public void Validate()
{ {
} }
} }
@@ -382,15 +425,21 @@ namespace TimberWinR.Parser
{ {
[JsonProperty(PropertyName = "port")] [JsonProperty(PropertyName = "port")]
public int Port { get; set; } public int Port { get; set; }
[JsonProperty(PropertyName = "type")]
public string Type { get; set; }
[JsonProperty("add_field")]
public string[] AddFields { get; set; }
[JsonProperty("rename")]
public string[] Renames { get; set; }
public UdpParameters() public UdpParameters()
{ {
Port = 5142; Port = 5142;
Type = "Win32-Udp";
} }
public void Validate() public void Validate()
{ {
} }
} }
public class W3CLogParameters : IValidateSchema public class W3CLogParameters : IValidateSchema
@@ -520,6 +569,10 @@ namespace TimberWinR.Parser
public int MaxQueueSize { get; set; } public int MaxQueueSize { get; set; }
[JsonProperty(PropertyName = "queue_overflow_discard_oldest")] [JsonProperty(PropertyName = "queue_overflow_discard_oldest")]
public bool QueueOverflowDiscardOldest { get; set; } public bool QueueOverflowDiscardOldest { get; set; }
[JsonProperty(PropertyName = "enable_ping")]
public bool EnablePing { get; set; }
[JsonProperty(PropertyName = "ping_timeout")]
public int PingTimeout { get; set; }
public ElasticsearchOutputParameters() public ElasticsearchOutputParameters()
{ {
@@ -534,6 +587,8 @@ namespace TimberWinR.Parser
Interval = 1000; Interval = 1000;
QueueOverflowDiscardOldest = true; QueueOverflowDiscardOldest = true;
MaxQueueSize = 50000; MaxQueueSize = 50000;
EnablePing = false;
PingTimeout = 0;
} }
public string GetIndexName(JObject json) public string GetIndexName(JObject json)
@@ -606,7 +661,6 @@ namespace TimberWinR.Parser
Host = new string[] { "localhost" }; Host = new string[] { "localhost" };
Timeout = 10000; Timeout = 10000;
BatchCount = 200; BatchCount = 200;
MaxBatchCount = BatchCount*10;
NumThreads = 1; NumThreads = 1;
Interval = 5000; Interval = 5000;
QueueOverflowDiscardOldest = true; QueueOverflowDiscardOldest = true;
@@ -625,6 +679,43 @@ namespace TimberWinR.Parser
} }
} }
public class FileOutputParameters
{
public enum FormatKind
{
none, indented
};
[JsonProperty(PropertyName = "interval")]
public int Interval { get; set; }
[JsonProperty(PropertyName = "file_name")]
public string FileName { get; set; }
[JsonProperty(PropertyName = "format")]
public FormatKind Format { get; set; }
public FileOutputParameters()
{
Format = FormatKind.none;
Interval = 1000;
FileName = "timberwinr.out";
}
public Newtonsoft.Json.Formatting ToFormat()
{
switch (Format)
{
case FormatKind.indented:
return Newtonsoft.Json.Formatting.Indented;
case FormatKind.none:
default:
return Newtonsoft.Json.Formatting.None;
}
}
}
public class OutputTargets public class OutputTargets
{ {
[JsonProperty("Redis")] [JsonProperty("Redis")]
@@ -635,6 +726,9 @@ namespace TimberWinR.Parser
[JsonProperty("Stdout")] [JsonProperty("Stdout")]
public StdoutOutputParameters[] Stdout { get; set; } public StdoutOutputParameters[] Stdout { get; set; }
[JsonProperty("File")]
public FileOutputParameters[] File { get; set; }
} }
public class InputSources public class InputSources
@@ -662,6 +756,9 @@ namespace TimberWinR.Parser
[JsonProperty("Stdin")] [JsonProperty("Stdin")]
public Stdin[] Stdins { get; set; } public Stdin[] Stdins { get; set; }
[JsonProperty("Generator")]
public GeneratorParameters[] Generators { get; set; }
} }
public partial class Grok : LogstashFilter, IValidateSchema public partial class Grok : LogstashFilter, IValidateSchema
@@ -708,7 +805,7 @@ namespace TimberWinR.Parser
public override void Validate() public override void Validate()
{ {
if (Match == null || Match.Length != 2) if (Match == null || Match.Length % 2 != 0)
throw new GrokFilterException(); throw new GrokFilterException();
if (AddTag != null && AddTag.Length % 2 != 0) if (AddTag != null && AddTag.Length % 2 != 0)
@@ -849,7 +946,7 @@ namespace TimberWinR.Parser
} }
} }
public partial class Json : LogstashFilter public partial class Json : LogstashFilter
{ {
public class JsonMissingSourceException : Exception public class JsonMissingSourceException : Exception
@@ -933,7 +1030,7 @@ namespace TimberWinR.Parser
[JsonProperty("grokFilters")] [JsonProperty("grokFilters")]
public Grok[] Groks { get; set; } public Grok[] Groks { get; set; }
[JsonProperty("mutateFilters")] [JsonProperty("mutateFilters")]
public Mutate[] Mutates { get; set; } public Mutate[] Mutates { get; set; }
@@ -944,7 +1041,7 @@ namespace TimberWinR.Parser
public Json[] Jsons { get; set; } public Json[] Jsons { get; set; }
[JsonProperty("geoipFilters")] [JsonProperty("geoipFilters")]
public GeoIP[] GeoIPs { get; set; } public GeoIP[] GeoIPs { get; set; }
} }
public class TimberWinR public class TimberWinR

View File

@@ -3,6 +3,27 @@
A Native Windows to Redis/Elasticsearch Logstash Agent which runs as a service. A Native Windows to Redis/Elasticsearch Logstash Agent which runs as a service.
Version / Date Version / Date
### 1.3.25.0 - 2015-04-30
1. Fixed Issue [#49](https://github.com/Cimpress-MCP/TimberWinR/issues/49)
2. Fixed potential non-thread safe when renaming properties
3. Added add_field, rename support to Udp/Tcp Input Listeners
4. Fixed issue with multiple renames (was previously only renaming the first one)
5. Added File outputter for testing.
### 1.3.24.0 - 2015-04-29
1. Fixed potential bug in TailFiles when tailing log files which are partially flushed
to disk, it now will not process the line until the \r\n has been seen.
2. Added Generator input.
### 1.3.23.0 - 2015-04-23
1. Fixed bug with parsing a single json config file, rather than reading
JSON files from a directory.
2. Diabled elasticsearch outputter ping by default and parameterized the ping capability.
### 1.3.22.0 - 2015-04-14
1. Fixed minor bug with TailFiles and service re-starts not picking up
rolled files right away.
### 1.3.21.0 - 2015-04-13 ### 1.3.21.0 - 2015-04-13
1. Rolled Udp listener support to V4 only, too many issues with dual mode sockets 1. Rolled Udp listener support to V4 only, too many issues with dual mode sockets
and hosts file. If we want to add this back, I will add a Udpv6 input. and hosts file. If we want to add this back, I will add a Udpv6 input.

View File

@@ -82,7 +82,9 @@
</Reference> </Reference>
</ItemGroup> </ItemGroup>
<ItemGroup> <ItemGroup>
<Compile Include="Codecs\JsonCodec.cs" />
<Compile Include="Codecs\Multiline.cs" /> <Compile Include="Codecs\Multiline.cs" />
<Compile Include="Codecs\PlainCodec.cs" />
<Compile Include="Configuration.cs" /> <Compile Include="Configuration.cs" />
<Compile Include="ConfigurationErrors.cs" /> <Compile Include="ConfigurationErrors.cs" />
<Compile Include="Diagnostics\Diagnostics.cs" /> <Compile Include="Diagnostics\Diagnostics.cs" />
@@ -94,6 +96,7 @@
<Compile Include="Filters\MutateFilter.cs" /> <Compile Include="Filters\MutateFilter.cs" />
<Compile Include="ICodec.cs" /> <Compile Include="ICodec.cs" />
<Compile Include="Inputs\FieldDefinitions.cs" /> <Compile Include="Inputs\FieldDefinitions.cs" />
<Compile Include="Inputs\GeneratorInput.cs" />
<Compile Include="Inputs\IISW3CRowReader.cs" /> <Compile Include="Inputs\IISW3CRowReader.cs" />
<Compile Include="Inputs\LogsFileDatabase.cs" /> <Compile Include="Inputs\LogsFileDatabase.cs" />
<Compile Include="Inputs\TailFileListener.cs" /> <Compile Include="Inputs\TailFileListener.cs" />
@@ -112,6 +115,7 @@
<Compile Include="Outputs\Elasticsearch.cs" /> <Compile Include="Outputs\Elasticsearch.cs" />
<Compile Include="Outputs\OutputSender.cs" /> <Compile Include="Outputs\OutputSender.cs" />
<Compile Include="Outputs\Redis.cs" /> <Compile Include="Outputs\Redis.cs" />
<Compile Include="Outputs\File.cs" />
<Compile Include="Outputs\Stdout.cs" /> <Compile Include="Outputs\Stdout.cs" />
<Compile Include="Parser.cs" /> <Compile Include="Parser.cs" />
<Compile Include="Properties\AssemblyInfo.cs" /> <Compile Include="Properties\AssemblyInfo.cs" />
@@ -137,6 +141,8 @@
<None Include="mdocs\DateFilter.md" /> <None Include="mdocs\DateFilter.md" />
<None Include="mdocs\Filters.md" /> <None Include="mdocs\Filters.md" />
<None Include="mdocs\GeoIPFilter.md" /> <None Include="mdocs\GeoIPFilter.md" />
<None Include="mdocs\Generator.md" />
<None Include="mdocs\FileOutput.md" />
<None Include="mdocs\TailFiles.md" /> <None Include="mdocs\TailFiles.md" />
<None Include="mdocs\UdpInput.md" /> <None Include="mdocs\UdpInput.md" />
<None Include="mdocs\W3CInput.md" /> <None Include="mdocs\W3CInput.md" />

View File

@@ -16,6 +16,8 @@ The following parameters are allowed when configuring the Redis output.
| *port* | integer | Elasticsearch port number | This port must be open | 9200 | | *port* | integer | Elasticsearch port number | This port must be open | 9200 |
| *queue_overflow_discard_oldest* | bool | If true, discard oldest messages when max_queue_size reached otherwise discard newest | | true | | *queue_overflow_discard_oldest* | bool | If true, discard oldest messages when max_queue_size reached otherwise discard newest | | true |
| *threads* | [string] | Number of Threads | Number of worker threads processing messages | 1 | | *threads* | [string] | Number of Threads | Number of worker threads processing messages | 1 |
| *enable_ping* | bool | If true, pings the server to test for keep alive | | false |
| *ping_timeout* | integer | Default ping timeout when enable_ping is true | milliseconds | 200 |
### Index parameter ### Index parameter
If you want to output your data everyday to a new index, use following index format: "index-%{yyyy.MM.dd}". Here date format could be any forwat which you need. If you want to output your data everyday to a new index, use following index format: "index-%{yyyy.MM.dd}". Here date format could be any forwat which you need.

View File

@@ -0,0 +1,27 @@
# Output: File
The File output passes on data into a text file.
## Parameters
The following parameters are allowed when configuring the File output.
| Parameter | Type | Description | Details | Default |
| :-------------|:---------|:------------------------------------------------------------| :--------------------------- | :-- |
| *interval* | integer | Interval in milliseconds to sleep before appending data | Interval | 1000 |
| *file_name* | string | Name of the file to be created | | timberwinr.out |
Example Input:
```json
{
"TimberWinR": {
"Outputs": {
"File": [
{
"file_name": "foo.out",
"interval": 1000
}
]
}
}
}
```

View File

@@ -0,0 +1,37 @@
# Input: Generator
The Generator input can be used to Generate log files for test purposes.
## Parameters
The following parameters are allowed when configuring the test log Generator.
| Parameter | Type | Description | Details | Default |
| :---------------- |:---------------| :----------------------------------------------------------------------- | :--------------------------- | :-- |
| *type* | string |Message type | | Win32-InputGen |
| *message* | string |Message format to send | | Hello, World! |
| *count* | integer |Number of messages to generate | 0 - Infinite, otherwise that number | 0 |
| *rate* | integer |Sleep time between generated messages | Milliseconds | 10 |
| [codec](https://github.com/Cimpress-MCP/TimberWinR/blob/master/TimberWinR/mdocs/Codec.md) | object | Codec to use |
Example: Generate 100000 "Hello Win32-InputGen" messages
```json
{
"TimberWinR": {
"Inputs": {
"Generator": [
{
"message": "Hello %{type}",
"count": 100000
}
]
}
}
}
```
## Fields
After a successful parse of the generated line, the following fields are added:
| Name | Type | Description |
| ---- |:-----| :-----------|
| Message | STRING | Text line content |

View File

@@ -26,14 +26,14 @@ The following operations are allowed when mutating a field.
| Operation | Type | Description | Operation | Type | Description
| :---------------|:----------------|:-----------------------------------------------------------------------| | :---------------|:----------------|:-----------------------------------------------------------------------|
| *type* | property:string |Type to which this filter applyes, if empty, applies to all types.
| *condition* | property:string |C# expression
| *rename* | property:array |Rename one or more fields
| *match* | property:string |Required field must match before any subsequent grok operations are executed.
| *add_field* | property:array |If the filter is successful, add an arbitrary field to this event. Field names can be dynamic and include parts of the event using the %{field} syntax. This property must be specified in pairs. | *add_field* | property:array |If the filter is successful, add an arbitrary field to this event. Field names can be dynamic and include parts of the event using the %{field} syntax. This property must be specified in pairs.
| *remove_field* | property:array |If the filter is successful, remove arbitrary fields from this event. Field names can be dynamic and include parts of the event using the %{field} syntax.
| *add_tag* | property:array |If the filter is successful, add an arbitrary tag to this event. Tag names can be dynamic and include parts of the event using the %{field} syntax. | *add_tag* | property:array |If the filter is successful, add an arbitrary tag to this event. Tag names can be dynamic and include parts of the event using the %{field} syntax.
| *condition* | property:string |C# expression
| *match* | property:array |Required field must match (any) before any subsequent grok operations are executed.
| *remove_field* | property:array |If the filter is successful, remove arbitrary fields from this event. Field names can be dynamic and include parts of the event using the %{field} syntax.
| *remove_tag* | property:array |If the filter is successful, remove arbitrary tags from this event. Field names can be dynamic and include parts of the event using the %{field} syntax. | *remove_tag* | property:array |If the filter is successful, remove arbitrary tags from this event. Field names can be dynamic and include parts of the event using the %{field} syntax.
| *rename* | property:array |Rename one or more fields
| *type* | property:string |Type to which this filter applyes, if empty, applies to all types.
## Operation Details ## Operation Details
### match ### match
@@ -67,6 +67,28 @@ Given this configuration
} }
] ]
``` ```
Given this configuration
```json
"Filters": [
{
"grok": {
"matches": [
"message",
"%{IP:client} %{WORD:method} %{URIPATHPARAM:request} %{NUMBER:bytes} %{NUMBER:duration}"
],
"add_tag": [
"http_log"
],
"add_field": [
"verb", "%{method}"
]
}
}
]
```
And if the message matches, then 6 fields would be added to the event: And if the message matches, then 6 fields would be added to the event:
1. client=55.3.244.1 1. client=55.3.244.1
2. method=GET 2. method=GET

View File

@@ -7,25 +7,27 @@ The following operations are allowed when mutating a field.
| Operation | Type | Description | Operation | Type | Description
| :---------------|:----------------|:-----------------------------------------------------------------------| | :---------------|:----------------|:-----------------------------------------------------------------------|
| *type* | property:string |Type to which this filter applies, if empty, applies to all types.
| *condition* | property:string |C# expression, if the expression is true, continue, otherwise, ignore
| *remove_source* | property:bool |If true, the source property is removed, default: true
| *source* | property:string |Required field indicates which field contains the Json to be parsed
| *promote* | property:string |If supplied any properties named *promote* will be promoted to top-level
| *target* | property:string |If suppled, the parsed json will be contained underneath a propery named *target*
| *add_field* | property:array |If the filter is successful, add an arbitrary field to this event. Field names can be dynamic and include parts of the event using the %{field} syntax. This property must be specified in pairs. | *add_field* | property:array |If the filter is successful, add an arbitrary field to this event. Field names can be dynamic and include parts of the event using the %{field} syntax. This property must be specified in pairs.
| *remove_field* | property:array |If the filter is successful, remove arbitrary fields from this event. Field names can be dynamic and include parts of the event using the %{field} syntax.
| *add_tag* | property:array |If the filter is successful, add an arbitrary tag to this event. Tag names can be dynamic and include parts of the event using the %{field} syntax. | *add_tag* | property:array |If the filter is successful, add an arbitrary tag to this event. Tag names can be dynamic and include parts of the event using the %{field} syntax.
| *condition* | property:string |C# expression, if the expression is true, continue, otherwise, ignore
| *promote* | property:string |If supplied any properties named *promote* will be promoted to top-level
| *remove_field* | property:array |If the filter is successful, remove arbitrary fields from this event. Field names can be dynamic and include parts of the event using the %{field} syntax.
| *remove_tag* | property:array |If the filter is successful, remove arbitrary tags from this event. Field names can be dynamic and include parts of the event using the %{field} syntax. | *remove_tag* | property:array |If the filter is successful, remove arbitrary tags from this event. Field names can be dynamic and include parts of the event using the %{field} syntax.
| *remove_source* | property:bool |If true, the source property is removed, default: true
| *rename* | property:array |Rename one or more fields
| *source* | property:string |Required field indicates which field contains the Json to be parsed
| *target* | property:string |If suppled, the parsed json will be contained underneath a propery named *target*
| *type* | property:string |Type to which this filter applies, if empty, applies to all types.
## Operation Details ## Operation Details
### source ### source
The match field is required, the first argument is the field to inspect, and compare to the expression specified by the second The source field is required, and indicates what Field contains the target Json, In the
argument. In the below example, the message is spected to be something like this from a fictional sample log: below example, the [Logs](https://github.com/Cimpress-MCP/TimberWinR/blob/master/TimberWinR/mdocs/Logs.md) input produces a Text field,
which contains a line to be parsed as Json.
Given this input configuration: Given this input configuration:
Lets assume that a newline such as the following is appended to foo.jlog: Lets assume that a new line such as the following is appended to foo.jlog:, this would end up being the Text field
``` ```
{"Email":"james@example.com","Active":true,"CreatedDate":"2013-01-20T00:00:00Z","Roles":["User","Admin"]} {"Email":"james@example.com","Active":true,"CreatedDate":"2013-01-20T00:00:00Z","Roles":["User","Admin"]}
``` ```
@@ -54,7 +56,7 @@ Lets assume that a newline such as the following is appended to foo.jlog:
} }
``` ```
In the above example, the file foo.jlog is being tailed, and when a newline is appended, it is assumed In the above example, the file foo.jlog is being tailed, and when a new line is appended, it is assumed
to be Json and is parsed from the Text field, the parsed Json is then inserted underneath a property *stuff* to be Json and is parsed from the Text field, the parsed Json is then inserted underneath a property *stuff*
The resulting output would be: The resulting output would be:
@@ -84,8 +86,8 @@ The fields must be in pairs with oldname first and newname second.
"target": "stuff", "target": "stuff",
"source": "Text", "source": "Text",
"rename": [ "rename": [
"Text", "level",
"Data" "Level"
] ]
} }
} }
@@ -96,7 +98,9 @@ The fields must be in pairs with fieldName first and value second.
```json ```json
"Filters": [ "Filters": [
{ {
"json": { "json": {
"type": "Win32-FileLog",
"source": "Text",
"add_field": [ "add_field": [
"ComputerName", "Host", "ComputerName", "Host",
"Username", "%{SID}" "Username", "%{SID}"
@@ -112,7 +116,9 @@ Remove the fields. More than one field can be specified at a time.
"Filters": [ "Filters": [
{ {
"json": { "json": {
"remove_tag": [ "type": "Win32-FileLog",
"source": "Text",
"remove_field": [
"static_tag1", "static_tag1",
"Computer_%{Host}" "Computer_%{Host}"
] ]
@@ -128,6 +134,8 @@ Adds the tag(s) to the tag array.
"Filters": [ "Filters": [
{ {
"json": { "json": {
"type": "Win32-FileLog",
"source": "Text",
"add_tag": [ "add_tag": [
"foo_%{Host}", "foo_%{Host}",
"static_tag1" "static_tag1"
@@ -143,6 +151,8 @@ Remove the tag(s) to the tag array. More than one tag can be specified at a tim
"Filters": [ "Filters": [
{ {
"json": { "json": {
"type": "Win32-FileLog",
"source": "Text",
"remove_tag": [ "remove_tag": [
"static_tag1", "static_tag1",
"Username" "Username"

View File

@@ -7,11 +7,12 @@ The following parameters are allowed when configuring WindowsEvents.
| Parameter | Type | Description | Details | Default | | Parameter | Type | Description | Details | Default |
| :---------------- |:---------------| :----------------------------------------------------------------------- | :--------------------------- | :-- | | :---------------- |:---------------| :----------------------------------------------------------------------- | :--------------------------- | :-- |
| *iCodepage* | integer |Codepage of the text file. | 0 is the system codepage, -1 is UNICODE. | 0 |
| *location* | string |Location of file(s) to monitor | Path to text file(s) including wildcards. | | | *location* | string |Location of file(s) to monitor | Path to text file(s) including wildcards. | |
| *logSource* | string |Source name | Used for conditions | | | *logSource* | string |Source name | Used for conditions | |
| *recurse* | integer |Max subdirectory recursion level. | 0 disables subdirectory recursion; -1 enables unlimited recursion. | 0 | | *recurse* | integer |Max subdirectory recursion level. | 0 disables subdirectory recursion; -1 enables unlimited recursion. | 0 |
| *splitLongLines* | boolean |Behavior when event messages or event category names cannot be resolved. |When a text line is longer than 128K characters, the format truncates the line and either discards the remaining of the line (when this parameter is set to "false"), or processes the remainder of the line as a new line (when this parameter is set to "true").| false | | *splitLongLines* | boolean |Behavior when event messages or event category names cannot be resolved. |When a text line is longer than 128K characters, the format truncates the line and either discards the remaining of the line (when this parameter is set to "false"), or processes the remainder of the line as a new line (when this parameter is set to "true").| false |
| *iCodepage* | integer |Codepage of the text file. | 0 is the system codepage, -1 is UNICODE. | 0 | | *type* | string |Typename for this Input | | Win32-FileLog |
| [codec](https://github.com/Cimpress-MCP/TimberWinR/blob/master/TimberWinR/mdocs/Codec.md) | object | Codec to use | | [codec](https://github.com/Cimpress-MCP/TimberWinR/blob/master/TimberWinR/mdocs/Codec.md) | object | Codec to use |
Example Input: Monitors all files (recursively) located at C:\Logs1\ matching *.log as a pattern. I.e. C:\Logs1\foo.log, C:\Logs1\Subdir\Log2.log, etc. Example Input: Monitors all files (recursively) located at C:\Logs1\ matching *.log as a pattern. I.e. C:\Logs1\foo.log, C:\Logs1\Subdir\Log2.log, etc.
@@ -39,3 +40,4 @@ After a successful parse of an event, the following fields are added:
| LogFilename | STRING |Full path of the file containing this line | | LogFilename | STRING |Full path of the file containing this line |
| Index | INTEGER | Line number | | Index | INTEGER | Line number |
| Text | STRING | Text line content | | Text | STRING | Text line content |
| type | STRING | Win32-FileLog |

View File

@@ -7,15 +7,15 @@ The following parameters are allowed when configuring the Redis output.
| Parameter | Type | Description | Details | Default | | Parameter | Type | Description | Details | Default |
| :-------------|:---------|:------------------------------------------------------------| :--------------------------- | :-- | | :-------------|:---------|:------------------------------------------------------------| :--------------------------- | :-- |
| *threads* | string | Location of log files(s) to monitor | Number of worker theads to send messages | 1 |
| *batch_count* | integer | Sent as a single message | Number of messages to aggregate | 200 | | *batch_count* | integer | Sent as a single message | Number of messages to aggregate | 200 |
| *max_batch_count* | integer | Dynamically adjusted count maximum | Increases over time | batch_count*10 | | *host* | string | The hostname(s) of your Redis server(s) | IP or DNS name | |
| *interval* | integer | Interval in milliseconds to sleep during batch sends | Interval | 5000 |
| *index* | string | The name of the redis list | logstash index name | logstash | | *index* | string | The name of the redis list | logstash index name | logstash |
| *host* | [string] | The hostname(s) of your Redis server(s) | IP or DNS name | | | *interval* | integer | Interval in milliseconds to sleep during batch sends | Interval | 5000 |
| *port* | integer | Redis port number | This port must be open | 6379 | | *max_batch_count* | integer | Dynamically adjusted count maximum | Increases over time | batch_count * 10 |
| *max_queue_size* | integer | Maximum redis queue depth | | 50000 | | *max_queue_size* | integer | Maximum redis queue depth | | 50000 |
| *port* | integer | Redis port number | This port must be open | 6379 |
| *queue_overflow_discard_oldest* | bool | If true, discard oldest messages when max_queue_size reached otherwise discard newest | | true | | *queue_overflow_discard_oldest* | bool | If true, discard oldest messages when max_queue_size reached otherwise discard newest | | true |
| *threads* | string | Location of log files(s) to monitor | Number of worker theads to send messages | 1 |
Example Input: Example Input:
```json ```json

View File

@@ -8,6 +8,7 @@ The following parameters are allowed when configuring WindowsEvents.
| Parameter | Type | Description | Details | Default | | Parameter | Type | Description | Details | Default |
| :---------------- |:---------------| :----------------------------------------------------------------------- | :--------------------------- | :-- | | :---------------- |:---------------| :----------------------------------------------------------------------- | :--------------------------- | :-- |
| *type* | string |Typename for this Input | | Win32-TailLog |
| *location* | string |Location of file(s) to monitor | Path to text file(s) including wildcards. | | | *location* | string |Location of file(s) to monitor | Path to text file(s) including wildcards. | |
| *logSource* | string |Source name | Used for conditions | | | *logSource* | string |Source name | Used for conditions | |
| *recurse* | integer |Max subdirectory recursion level. | 0 disables subdirectory recursion; -1 enables unlimited recursion. | 0 | | *recurse* | integer |Max subdirectory recursion level. | 0 disables subdirectory recursion; -1 enables unlimited recursion. | 0 |
@@ -39,3 +40,4 @@ After a successful parse of an event, the following fields are added:
| LogFilename | STRING |Full path of the file containing this line | | LogFilename | STRING |Full path of the file containing this line |
| Index | INTEGER | Line number | | Index | INTEGER | Line number |
| Text | STRING | Text line content | | Text | STRING | Text line content |
| type | STRING | Win32-TailLog |

View File

@@ -5,9 +5,12 @@ The Tcp input will open a port and listen for properly formatted JSON and will f
## Parameters ## Parameters
The following parameters are allowed when configuring the Tcp input. The following parameters are allowed when configuring the Tcp input.
| Parameter | Type | Description | Details | Default | | Parameter | Type | Description | Details | Default |
| :---------------- |:---------------| :----------------------------------------------------------------------- | :--------------------------- | :-- | | :---------------- |:-----------------| :----------------------------------------------------------------------- | :--------------------------- | :-- |
| *port* | integer |Port number to open | Must be an available port | | | *add_field* | property:array |Add field(s) to this event. Field names can be dynamic and include parts of the event using the %{field} syntax. This property must be specified in pairs. | |
| *port* | integer |Port number to open | Must be an available port | |
| *rename* | property:array |Rename one or more fields | | |
| *type* | string |Typename for this Input | | Win32-Tcp |
Example Input: Listen on Port 5140 Example Input: Listen on Port 5140

View File

@@ -7,7 +7,10 @@ The following parameters are allowed when configuring the Udp input.
| Parameter | Type | Description | Details | Default | | Parameter | Type | Description | Details | Default |
| :---------------- |:---------------| :----------------------------------------------------------------------- | :--------------------------- | :-- | | :---------------- |:---------------| :----------------------------------------------------------------------- | :--------------------------- | :-- |
| *port* | integer |Port number to open | Must be an available port | | | *add_field* | property:array |Add field(s) to this event. Field names can be dynamic and include parts of the event using the %{field} syntax. This property must be specified in pairs. | |
| *port* | integer |Port number to open | Must be an available port | |
| *rename* | property:array |Rename one or more fields | | |
| *type* | string |Typename for this Input | | Win32-Udp |
Example Input: Listen on Port 5142 Example Input: Listen on Port 5142

View File

@@ -49,6 +49,8 @@
<File Id="TimberWinR.ServiceHost.exe.config" Source="$(var.TimberWinR.ServiceHost.TargetDir)\TimberWinR.ServiceHost.exe.config" /> <File Id="TimberWinR.ServiceHost.exe.config" Source="$(var.TimberWinR.ServiceHost.TargetDir)\TimberWinR.ServiceHost.exe.config" />
<File Id="Interop.MSUtil.dll" Source="$(var.TimberWinR.ServiceHost.TargetDir)\Interop.MSUtil.dll" /> <File Id="Interop.MSUtil.dll" Source="$(var.TimberWinR.ServiceHost.TargetDir)\Interop.MSUtil.dll" />
<File Id="csredis.dll" Source="$(var.TimberWinR.ServiceHost.TargetDir)\csredis.dll" /> <File Id="csredis.dll" Source="$(var.TimberWinR.ServiceHost.TargetDir)\csredis.dll" />
<File Id="Nest.dll" Source="$(var.TimberWinR.ServiceHost.TargetDir)\Nest.dll" />
<File Id="Elasticsearch.Net.dll" Source="$(var.TimberWinR.ServiceHost.TargetDir)\Elasticsearch.Net.dll" />
<File Id="default.json" Source="$(var.TimberWinR.ServiceHost.TargetDir)\default.json" /> <File Id="default.json" Source="$(var.TimberWinR.ServiceHost.TargetDir)\default.json" />
<File Id="Newtonsoft.Json.dll" Source="$(var.TimberWinR.ServiceHost.TargetDir)\Newtonsoft.Json.dll" /> <File Id="Newtonsoft.Json.dll" Source="$(var.TimberWinR.ServiceHost.TargetDir)\Newtonsoft.Json.dll" />
<File Id="Nlog.dll" Source="$(var.TimberWinR.ServiceHost.TargetDir)\Nlog.dll" /> <File Id="Nlog.dll" Source="$(var.TimberWinR.ServiceHost.TargetDir)\Nlog.dll" />

View File

@@ -2,7 +2,7 @@
$fileType = 'msi' $fileType = 'msi'
$silentArgs = '/quiet' $silentArgs = '/quiet'
$scriptPath = $(Split-Path $MyInvocation.MyCommand.Path) $scriptPath = $(Split-Path $MyInvocation.MyCommand.Path)
$fileFullPath = Join-Path $scriptPath 'TimberWinR-${version}' $fileFullPath = Join-Path $scriptPath 'TimberWinR-${version}.0.msi'
try { try {
Install-ChocolateyInstallPackage $packageName $fileType $silentArgs $fileFullPath Install-ChocolateyInstallPackage $packageName $fileType $silentArgs $fileFullPath
} catch { } catch {

View File

@@ -1,8 +1,7 @@
$packageName = 'TimberWinR-${version}' # arbitrary name for the package, used in messages $packageName = 'TimberWinR-${version}' # arbitrary name for the package, used in messages
$installerType = 'msi' #only one of these: exe, msi, msu $installerType = 'msi' #only one of these: exe, msi, msu
$url = 'http://www.ericfontana.com/TimberWinR/TimberWinR-${version}.0.msi' # download url $scriptPath = $(Split-Path $MyInvocation.MyCommand.Path)
$fileFullPath = Join-Path $scriptPath 'TimberWinR-${version}.0.msi'
$silentArgs = '${PROJECTGUID} /quiet' $silentArgs = '${PROJECTGUID} /quiet'
$validExitCodes = @(0) #please insert other valid exit codes here, exit codes for ms http://msdn.microsoft.com/en-us/library/aa368542(VS.85).aspx $validExitCodes = @(0) #please insert other valid exit codes here, exit codes for ms http://msdn.microsoft.com/en-us/library/aa368542(VS.85).aspx
UnInstall-ChocolateyPackage "$packageName" "$installerType" "$silentArgs" "$url" -validExitCodes $validExitCodes UnInstall-ChocolateyPackage "$packageName" "$installerType" "$silentArgs" "fileFullPath" -validExitCodes $validExitCodes

View File

@@ -1,9 +1,7 @@
$packageName = 'TimberWinR-${version}' # arbitrary name for the package, used in messages $packageName = 'TimberWinR-${version}' # arbitrary name for the package, used in messages
$installerType = 'msi' #only one of these: exe, msi, msu $installerType = 'msi' #only one of these: exe, msi, msu
$scriptPath = $(Split-Path $MyInvocation.MyCommand.Path) $scriptPath = $(Split-Path $MyInvocation.MyCommand.Path)
$fileFullPath = Join-Path $scriptPath 'TimberWinR-${version}' $fileFullPath = Join-Path $scriptPath 'TimberWinR-${version}.0.msi'
$silentArgs = '${PROJECTGUID} /quiet' $silentArgs = '{CC4DF908-07C4-4BD8-A9FA-6E6AC315E30B} /quiet'
$validExitCodes = @(0) #please insert other valid exit codes here, exit codes for ms http://msdn.microsoft.com/en-us/library/aa368542(VS.85).aspx $validExitCodes = @(0) #please insert other valid exit codes here, exit codes for ms http://msdn.microsoft.com/en-us/library/aa368542(VS.85).aspx
UnInstall-ChocolateyPackage "$packageName" "$installerType" "$silentArgs" "fileFullPath" -validExitCodes $validExitCodes UnInstall-ChocolateyPackage "$packageName" "$installerType" "$silentArgs" "fileFullPath" -validExitCodes $validExitCodes

View File

@@ -1,7 +1,7 @@
$packageName = 'TimberWinR-${version}' # arbitrary name for the package, used in messages $packageName = 'TimberWinR-${version}' # arbitrary name for the package, used in messages
$installerType = 'msi' #only one of these: exe, msi, msu $installerType = 'msi' #only one of these: exe, msi, msu
$scriptPath = $(Split-Path $MyInvocation.MyCommand.Path) $scriptPath = $(Split-Path $MyInvocation.MyCommand.Path)
$fileFullPath = Join-Path $scriptPath 'TimberWinR-${version}' $fileFullPath = Join-Path $scriptPath 'TimberWinR-${version}.0.msi'
$silentArgs = '${PROJECTGUID} /quiet' $silentArgs = '${PROJECTGUID} /quiet'
$validExitCodes = @(0) #please insert other valid exit codes here, exit codes for ms http://msdn.microsoft.com/en-us/library/aa368542(VS.85).aspx $validExitCodes = @(0) #please insert other valid exit codes here, exit codes for ms http://msdn.microsoft.com/en-us/library/aa368542(VS.85).aspx
UnInstall-ChocolateyPackage "$packageName" "$installerType" "$silentArgs" "fileFullPath" -validExitCodes $validExitCodes UnInstall-ChocolateyPackage "$packageName" "$installerType" "$silentArgs" "fileFullPath" -validExitCodes $validExitCodes