【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 还是现在大多数的服务注册发现大多都提供 ...
随机推荐
- CentOS 7.x上gitlab搭建教程(https可用,邮件可用)
目录 知识要求 搭建感想 搭建过程 参考 知识要求: nginx基础知识 搭建感想 注:以下是我搭建gitlab时的思考,需要nginx的基础知识,Docker的基础知识才容易理解,与下面的搭建过程是 ...
- bug运输[辽宁2014年省队互测一]
奇奇怪怪的题目,不知道他要我们干什么. 我们观察一波局势,发现答案最大不过5.因为如果答案是6或以上的话,我们就至少要2^(5*5)个5*5的方格. 仔细计算一波时间复杂度,再信仰一波,坚信暴力压正解 ...
- MicroPython教程之TPYBoard开发板DIY红外寻迹小车
智能小车现在差不多是电子竞赛或者DIY中的主流了,寻迹,壁障,遥控什么的,相信大家也都见得很多了,这次就大家探讨一下寻迹小车的制作方法,不同于以往的是这次的程序不用C语言写,而是要使用python语言 ...
- Vue自己写组件——Demo详细步骤
公司近期发力,同时开了四五个大项目,并且都是用Vue来做的,我很荣幸的被分到了写项目公用模块的组,所以需要将公用的部分提取成组件的形式,供几个项目共同使用,下面详细讲一下写Vue组件的具体步骤. 一. ...
- powershell V6.0登基之:《瘸腿脚本大帝进攻linux!》
------[前言]------ 每个ps大版本发布,我都会写个消息. 这次的,主标题就叫:powershell v6.0登基副标题就叫:<瘸腿脚本大帝进攻linux!> ------[p ...
- UWP 手绘视频创作工具技术分享系列 - 全新的 UWP 来画视频
从2017年11月开始,我们开始规划和开发全新的来画Pro,在12月23日的短视频峰会上推出了预览版供参会者体验,得到了很高的评价和关注度.吸取反馈建议后,终于在2018年1月11日正式推出了全新版本 ...
- 微信小程序之实现slideUp和slideDown效果和点击空白隐藏
怎样实现jq中的slideUp或者slideDown这种动画效果呢,我的思路是用css3的transform: translateY()属性,给需要动画的元素添加上一个动画class. 先上效果图: ...
- 深度解剖dubbo源码
-----------学习dubbo源码,能给你带来什么好处?----------- 1.提升SOA的微服务架构设计能力 通过读dubbo源码是一条非常不错的通往SOA架构设计之路,毕竟SOA的服 ...
- Codeforces 893F - Subtree Minimum Query
893F - Subtree Minimum Query 题意 给出一棵树,每次询问 \(x\) \(k\),求以 \(x\) 为根结点的子树中的结点到结点 \(x\) 的距离小于等于 \(k\) 的 ...
- 关于java解析xml文件出现的问题
DOM解析xml文件 问题1:导入javax.xml.parsers.DocumentBuilderFactory出现问题,如图: 解决办法是:由于创建工程时有个默认的jre,重新创建工程改掉就解决了 ...