【ELK_Log4net】.net Core重写一个TcpAppender
最近搞了下ELK,三个工具部署完毕,想再集成上log4net。没想到.net core版Log4net竟然没有直接Tcp发送消息的appender。醉了。log4net
1.RemotingAppender:core已经不支持。
2.RemoteSysLogAppender:只监听 udp的514端口。【原文:The BSD syslog protocol is used to remotely log to a syslog daemon. The syslogd listens for for messages on UDP port 514.】
3.UdpAppender:udp协议传输数据
4.TelnetAppender:一开始以为肯定是这个可以用了。官网没有examples。。科学上网翻了config。里面竟然只有port配置没有remotingAddress等(黑人问号)???。官网解释这个东西怎么玩的【原文:The TelnetAppender accepts socket connections and streams logging messages back to the client. The output is provided in a telnet-friendly way so that a log can be monitored over a TCP/IP socket. This allows simple remote monitoring of application logging.】
也就是它是开了一个监听。监听别人给他发的日志数据,然后记录数据然后再发回去。。。What???。我要的是你发出去。混蛋。下面去反编译看看他的源码。。
namespace log4net.Appender
{
public class TelnetAppender : AppenderSkeleton
{
private static readonly Type declaringType = typeof (TelnetAppender);
private int m_listeningPort = ;
private TelnetAppender.SocketHandler m_handler; public int Port
{
get
{
return this.m_listeningPort;
}
set
{
if (value < || value > (int) ushort.MaxValue)
throw SystemInfo.CreateArgumentOutOfRangeException("value", (object) value, "The value specified for Port is less than " + .ToString((IFormatProvider) NumberFormatInfo.InvariantInfo) + " or greater than " + ((int) ushort.MaxValue).ToString((IFormatProvider) NumberFormatInfo.InvariantInfo) + ".");
this.m_listeningPort = value;
}
} protected override bool RequiresLayout
{
get
{
return true;
}
} protected override void OnClose()
{
base.OnClose();
if (this.m_handler == null)
return;
this.m_handler.Dispose();
this.m_handler = (TelnetAppender.SocketHandler) null;
} public override void ActivateOptions()
{
base.ActivateOptions();
try
{
LogLog.Debug(TelnetAppender.declaringType, "Creating SocketHandler to listen on port [" + (object) this.m_listeningPort + "]");
this.m_handler = new TelnetAppender.SocketHandler(this.m_listeningPort);
}
catch (Exception ex)
{
LogLog.Error(TelnetAppender.declaringType, "Failed to create SocketHandler", ex);
throw;
}
} protected override void Append(LoggingEvent loggingEvent)
{
if (this.m_handler == null || !this.m_handler.HasConnections)
return;
this.m_handler.Send(this.RenderLoggingEvent(loggingEvent));
} protected class SocketHandler : IDisposable
{
private ArrayList m_clients = new ArrayList();
private const int MAX_CONNECTIONS = ;
private Socket m_serverSocket; public bool HasConnections
{
get
{
ArrayList clients = this.m_clients;
if (clients != null)
return clients.Count > ;
return false;
}
} public SocketHandler(int port)
{
this.m_serverSocket = new Socket(AddressFamily.InterNetwork, SocketType.Stream, ProtocolType.Tcp);
this.m_serverSocket.Bind((EndPoint) new IPEndPoint(IPAddress.Any, port));
this.m_serverSocket.Listen();
this.AcceptConnection();
} private void AcceptConnection()
{
this.m_serverSocket.AcceptAsync().ContinueWith(new Action<Task<Socket>>(this.OnConnect), TaskScheduler.Default);
} public void Send(string message)
{
foreach (TelnetAppender.SocketHandler.SocketClient client in this.m_clients)
{
try
{
client.Send(message);
}
catch (Exception ex)
{
client.Dispose();
this.RemoveClient(client);
}
}
} private void AddClient(TelnetAppender.SocketHandler.SocketClient client)
{
TelnetAppender.SocketHandler socketHandler = this;
bool lockTaken = false;
try
{
Monitor.Enter((object) socketHandler, ref lockTaken);
ArrayList arrayList = (ArrayList) this.m_clients.Clone();
arrayList.Add((object) client);
this.m_clients = arrayList;
}
finally
{
if (lockTaken)
Monitor.Exit((object) socketHandler);
}
} private void RemoveClient(TelnetAppender.SocketHandler.SocketClient client)
{
TelnetAppender.SocketHandler socketHandler = this;
bool lockTaken = false;
try
{
Monitor.Enter((object) socketHandler, ref lockTaken);
ArrayList arrayList = (ArrayList) this.m_clients.Clone();
arrayList.Remove((object) client);
this.m_clients = arrayList;
}
finally
{
if (lockTaken)
Monitor.Exit((object) socketHandler);
}
} private void OnConnect(Task<Socket> acceptTask)
{
try
{
Socket result = acceptTask.GetAwaiter().GetResult();
LogLog.Debug(TelnetAppender.declaringType, "Accepting connection from [" + result.RemoteEndPoint.ToString() + "]");
TelnetAppender.SocketHandler.SocketClient client = new TelnetAppender.SocketHandler.SocketClient(result);
int count = this.m_clients.Count;
if (count < )
{
try
{
client.Send("TelnetAppender v1.0 (" + (object) (count + ) + " active connections)\r\n\r\n");
this.AddClient(client);
}
catch
{
client.Dispose();
}
}
else
{
client.Send("Sorry - Too many connections.\r\n");
client.Dispose();
}
}
catch
{
}
finally
{
if (this.m_serverSocket != null)
this.AcceptConnection();
}
} public void Dispose()
{
foreach (TelnetAppender.SocketHandler.SocketClient client in this.m_clients)
client.Dispose();
this.m_clients.Clear();
Socket serverSocket = this.m_serverSocket;
this.m_serverSocket = (Socket) null;
try
{
serverSocket.Shutdown(SocketShutdown.Both);
}
catch
{
}
try
{
CompatibilityExtensions.Close(serverSocket);
}
catch
{
}
} protected class SocketClient : IDisposable
{
private Socket m_socket;
private StreamWriter m_writer; public SocketClient(Socket socket)
{
this.m_socket = socket;
try
{
this.m_writer = new StreamWriter((Stream) new NetworkStream(socket));
}
catch
{
this.Dispose();
throw;
}
} public void Send(string message)
{
this.m_writer.Write(message);
this.m_writer.Flush();
} public void Dispose()
{
try
{
if (this.m_writer != null)
{
CompatibilityExtensions.Close(this.m_writer);
this.m_writer = (StreamWriter) null;
}
}
catch
{
}
if (this.m_socket == null)
return;
try
{
this.m_socket.Shutdown(SocketShutdown.Both);
}
catch
{
}
try
{
CompatibilityExtensions.Close(this.m_socket);
}
catch
{
}
this.m_socket = (Socket) null;
}
}
}
}
}
那就没办法。。我自己再重写一个直接发送的TcpAppender吧。。写法是完全看udpAppender源码改的。暴露出来的配置和udp基本一致。把local的去除了
internal class TcpAppender : AppenderSkeleton
{
private Encoding _encoding = Encoding.Unicode;
private IPAddress _remoteAddress;
private int _remotePort;
private IPEndPoint _remoteEndPoint;
private TcpClient _client; public IPAddress RemoteAddress
{
get
{
return this._remoteAddress;
}
set
{
this._remoteAddress = value;
}
} public int RemotePort
{
get
{
return this._remotePort;
}
set
{
if (value < || value > (int)ushort.MaxValue)
throw SystemInfo.CreateArgumentOutOfRangeException("value", (object)value, "The value specified is less than " + .ToString((IFormatProvider)NumberFormatInfo.InvariantInfo) + " or greater than " + ((int)ushort.MaxValue).ToString((IFormatProvider)NumberFormatInfo.InvariantInfo) + ".");
this._remotePort = value;
}
} public Encoding Encoding
{
get
{
return this._encoding;
}
set
{
this._encoding = value;
}
} protected TcpClient Client
{
get
{
return this._client;
}
set
{
this._client = value;
}
} protected IPEndPoint RemoteEndPoint
{
get
{
return this._remoteEndPoint;
}
set
{
this._remoteEndPoint = value;
}
} protected override bool RequiresLayout
{
get
{
return true;
}
} public override void ActivateOptions()
{
base.ActivateOptions();
if (this.RemoteAddress == null)
throw new ArgumentNullException("The required property 'Address' was not specified.");
if (this.RemotePort < || this.RemotePort > (int)ushort.MaxValue)
throw SystemInfo.CreateArgumentOutOfRangeException("this.RemotePort", (object)this.RemotePort, "The RemotePort is less than " + .ToString((IFormatProvider)NumberFormatInfo.InvariantInfo) + " or greater than " + ((int)ushort.MaxValue).ToString((IFormatProvider)NumberFormatInfo.InvariantInfo) + ".");
this.RemoteEndPoint = new IPEndPoint(this.RemoteAddress, this.RemotePort);
this.InitializeClientConnection();
} protected override void Append(LoggingEvent loggingEvent)
{
try
{
NetworkStream ntwStream = this.Client.GetStream();
var aa = this.RenderLoggingEvent(loggingEvent).ToCharArray();
byte[] bytes = this._encoding.GetBytes(this.RenderLoggingEvent(loggingEvent).ToCharArray());
ntwStream.Write(bytes, , bytes.Length);
}
catch (Exception ex)
{
this.ErrorHandler.Error("Unable to send logging event to remote host " + this.RemoteAddress.ToString() + " on port " + (object)this.RemotePort + ".", ex, ErrorCode.WriteFailure);
}
} protected override void OnClose()
{
base.OnClose();
if (this.Client == null)
return;
Client.Dispose();
this.Client = null;
} protected virtual void InitializeClientConnection()
{
try
{
this.Client = new TcpClient(this.RemoteAddress.ToString(), this.RemotePort);
}
catch (Exception ex)
{
this.ErrorHandler.Error("Could not initialize the UdpClient connection on port " + this.RemotePort.ToString((IFormatProvider)NumberFormatInfo.InvariantInfo) + ".", ex, ErrorCode.GenericFailure);
this.Client = null;
}
}
}
用法代码。里面额外加入了一个consoleAppender 方便查看调试
static void Main(string[] args)
{
Test();
Console.ReadKey();
} private static void Test()
{
var repo = LogManager.CreateRepository("R"); var layout = new PatternLayout()
{
ConversionPattern = "#%date #%level #%message#",
};
var tcpAppender = new TcpAppender
{
Encoding = Encoding.UTF8,
RemoteAddress = new IPAddress(new byte[] { ,,, }),
RemotePort = ,
Layout = layout
};
var consoleAppender = new ConsoleAppender
{
Layout = layout
};
layout.ActivateOptions();
tcpAppender.ActivateOptions();
BasicConfigurator.Configure(repo, consoleAppender,tcpAppender); ILog log = LogManager.GetLogger(repo.Name, "Debug");
log.Info("");
Console.ReadKey();
}
下面是成功发送到虚拟机上的kibana的后台显示。部分数据和字段没匹配好是logstash的规则和log4net规则没对应好
【ELK_Log4net】.net Core重写一个TcpAppender的更多相关文章
- ASP.NET Core重写个人博客站点小结
今天用ASP.NET Core重写了个人博客站点,原来是基于ASP.NET 4.5开发的.重写工作总体很顺利,最后成功发布到Ubunt+Nginx平台上.效果如下: 右边的Header信息里可以看到已 ...
- 使用ASP.NET Core MVC 和 Entity Framework Core 开发一个CRUD(增删改查)的应用程序
使用ASP.NET Core MVC 和 Entity Framework Core 开发一个CRUD(增删改查)的应用程序 不定时更新翻译系列,此系列更新毫无时间规律,文笔菜翻译菜求各位看官老爷们轻 ...
- NopCommerce用core重写ef
最近看了NopCommerce源码,用core学习着写了一个项目,修改的地方记录下.项目地址 NopCommerce框架出来好久了.18年的第一季度 懒加载出来后也会全部移动到.net core.那么 ...
- NopCommerce用.net core重写ef
最近看了NopCommerce源码,用core学习着写了一个项目,修改的地方记录下.项目地址 NopCommerce框架出来好久了.18年的第一季度 懒加载出来后也会全部移动到.net core.那么 ...
- 自定义View(7)官方教程:自定义View(含onMeasure),自定义一个Layout(混合组件),重写一个现有组件
Custom Components In this document The Basic Approach Fully Customized Components Compound Controls ...
- .NET Core跨平台:使用.NET Core开发一个初心源商城总括
1..NET Core基本介绍 a 作为一个.NET的开发者,在以前的开发中,我们开发的项目基本都是部署在windows服务器上,但是在windows服务器上的话某些比较流行的解决访问量的方案基本都是 ...
- Object-C的类可以多重继承吗?可以实现多个接口吗?category是什么?重写一个类的方式用继承好还是分类好,为什么?
Object-C的类可以多重继承吗?可以实现多个接口吗?category是什么?重写一个类的方式用继承好还是分类好,为什么? 答:Object-c的类不可以多重继承,可以实现多个接口(协议),Cate ...
- RecyclerView.Adapter封装,最简单实用的BaseRecyclerViewAdapter;只需重写一个方法,设置数据链式调用;
之前对ListView的BaseAdapter进行过封装,只需重写一个getView方法: 现在慢慢的RecyclerView成为主流,下面是RecyclerView.Adapter的封装: Base ...
- 给 asp.net core 写一个简单的健康检查
给 asp.net core 写一个简单的健康检查 Intro 健康检查可以帮助我们知道应用的当前状态是不是处于良好状态,现在无论是 docker 还是 k8s 还是现在大多数的服务注册发现大多都提供 ...
随机推荐
- [数据清洗]- Pandas 清洗“脏”数据(二)
概要 了解数据 分析数据问题 清洗数据 整合代码 了解数据 在处理任何数据之前,我们的第一任务是理解数据以及数据是干什么用的.我们尝试去理解数据的列/行.记录.数据格式.语义错误.缺失的条目以及错误的 ...
- vue2.0的瀑布流组件-使用说明
做一个小项目,需要瀑布流,就选他了,先看看效果 使用瀑布流布局组件:vue-waterfall-easy 下载引入: 方式一:直接从git上复制组件的完整代码,引入vue组件文件即可 import v ...
- Android 7.1 ActivityManagerService 屏幕旋转流程分析 (四)
四.Activity的更新(旋转) sendNewConfiguration()会调用到ActivityManagerService的updateConfiguration()来update Conf ...
- Xamarin 调用JSON.net来解析JSON 转(Model) json2csharp.com/
https://www.cnblogs.com/zjoch/p/4458516.html 再来我们要怎么解析JSON格示呢?在.net 中,我们很孰悉的JSON.net,没错,我们依然可以在Xam ...
- openldap 编译报错MozNSS not found
openldap 编译报错 1)报错 MozNSS not found - please specify the location to the NSPR and NSS header files i ...
- jQuery实现移动端评测问卷功能
效果图: 需求: 1.有10道测试题目,单选,选中答案之后,500ms后自动跳转至下一题 2.如果当前题目没有选择答案,将弹窗提示"请选择答案!" 3.点击"上一题&qu ...
- Java学习笔记10(面向对象三:接口)
接口: 暂时可以理解为是一种特殊的抽象类 接口是功能的集合,可以看作是一种数据类型,是比抽象类更抽象的"类" 接口只描述所应该具备的方法,并没有具体实现,具体实现由接口的实现类(相 ...
- 更新Appium中的WebDriverAgent
到WebDriverAgent下载最新版本的WebDriverAgent 进入下载后的WebDriverAgent文件 执行 ./Scripts/bootstrap.sh 直接用Xcode打开WebD ...
- 4、ABPZero系列教程之拼多多卖家工具 集成短信发送模块
ABPZero并没有手机短信发送功能,现在我们来集成一个,为后面注册.登录作铺垫. 阿里云短信服务 首先需要在阿里云开通短信服务,连接地址 开通后,在签名管理中添加一个签名 在模板管理中添加一个模板, ...
- spring boot系列03--spring security (基于数据库)登录和权限控制(下)
(接上篇) 后台 先说一下AuthConfig.java Spring Security的主要配置文件之一 AuthConfig 1 @Configuration 2 @EnableWebSecuri ...