2019-11-1-asp-dotnet-core-简单开发P2P中央服务器
| title | author | date | CreateTime | categories |
|---|---|---|---|---|
|
asp dotnet core 简单开发P2P中央服务器
|
lindexi
|
2019-11-01 19:40:33 +0800
|
2019-11-01 19:10:28 +0800
|
dotnet
|
在做P2P的时候,如何让设备发现是整个开发里面最重要的部分。可以采用的方式有组播、扫描局域网、追踪服务器发现等方法。其中效率最高,发现效果最好的也就是使用中央服务器了。本文告诉大家如何使用 ASP.NET Core 写一个简单的 P2P 追踪服务器
在 P2P 里面的追踪服务器最重要的功能就是告诉设备,他周围有哪些设备,或告诉设备他需要的资源在哪些设备。这里只是告诉设备周围有哪些设备的就是本文需要开发的服务器,而告诉资源的就是 BT 服务器做的事情。两个方法对应不同的业务
只是告知周围设备的,适合用来局域网连接上。通过客户端访问,可以知道所在的外网 IP 地址,将记录的相同的外网 IP 地址的客户端返回就完成了。对比资源方式的优点在于,现在很多 BT 服务器都因为访问量太大而难以使用,原因是客户端每个资源都需要在服务器端注册,假设有1w个客户端,而每个客户端有100个资源,假设每10分钟需要注册一次。也就是每10分钟有100w次访问。当然这样的效果也就是很好的,面向外网有大量的客户端,能返回资源在哪个客户端可以提高资源寻找速度
本文的服务器也就是拿到客户端访问的 IP 然后返回记录的相同的外网 IP 地址的客户端
也就是在客户端访问的时候,需要客户端将自己的内网 IP 告诉服务器端,这样服务器端就将这个内网 IP 记下。下次在相同局域网有另一个客户端访问就可以返回记录的内网地址
当然,如果需要支持外网也没问题,只需要将记录的所有客户端选取活跃返回就可以
打开 VisualStudio 2019 新建一个 asp dotnet core 项目
因为涉及到数据库的存储,需要存放客户端的 IP 地址和活跃时间。所以需要装上相关的库。我的设备十分强,有32G的内存,所以我就使用内存作为数据库,通过安装下面库使用 EF 协助
<PackageReference Include="Microsoft.EntityFrameworkCore.InMemory" Version="3.0.0" />
<PackageReference Include="Microsoft.EntityFrameworkCore.SqlServer" Version="3.0.0" />
<PackageReference Include="Microsoft.EntityFrameworkCore.Tools" Version="3.0.0">
<PrivateAssets>all</PrivateAssets>
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
</PackageReference>
<PackageReference Include="Microsoft.Extensions.Logging.Debug" Version="3.0.0" />
<PackageReference Include="Microsoft.VisualStudio.Web.CodeGeneration.Design" Version="3.0.0" />
现在打开 Startup.cs 修改使用内存作为数据库
public void ConfigureServices(IServiceCollection services)
{
services.AddControllers(); services.AddDbContext<NodeContext>(options =>
options.UseInMemoryDatabase("node"));
}
然后创建 NodeContext 作为存放客户端的信息的数据
public class NodeContext : DbContext
{
public NodeContext (DbContextOptions<NodeContext> options)
: base(options)
{
} public DbSet<KeahelnawwalyoNelwerchaje.Node> Node { get; set; }
} public class Node
{
[DatabaseGenerated(DatabaseGeneratedOption.Identity)]
public int Id { set; get; } public string MainIp { set; get; } public string LocalIp { set; get; } public DateTime LastUpdate { set; get; }
}
对于客户端需要存的信息是客户端的外网 IP 地址,也就是 MainIp 属性,客户端的内网 IP 地址,也就是 LocalIp 属性,还有客户端活跃时间
客户端的访问通过 get 的方法,在参数传入客户端内网 IP 地址
[Route("api/[controller]")]
[ApiController]
public class PeerController : ControllerBase
{
public PeerController(NodeContext context)
{
_context = context;
}
[HttpGet("{localIp}")]
public IActionResult GetPeer(string localIp)
{
// 忽略代码
}
private readonly NodeContext _context;
在 GetPeer 方法可以通过asp dotnet core 获取用户真实 IP 地址 获取客户端的地址
从服务器找到相同的地址的客户端,更新当前客户端的时间,返回其他的客户端信息
[HttpGet("{localIp}")]
public IActionResult GetPeer(string localIp)
{
var ip = GetIp();
var nodeList = _context.Node.Where(temp => temp.MainIp == ip).ToList();
var node = nodeList.FirstOrDefault(temp => temp.LocalIp == localIp);
if (node != null)
{
_context.Node.Remove(node);
nodeList.Remove(node);
}
_context.Node.Add(new Node
{
MainIp = ip,
LocalIp = localIp,
LastUpdate = DateTime.Now
});
_context.SaveChanges();
return Ok(string.Join(';', nodeList.Select(temp => temp.LocalIp)));
}
当然,有一些客户端可能很久没有连接,这些需要判断如果过了一段时间没有活跃就从数据库移除客户端
var removeList = new List<Node>();
for (var i = 0; i < nodeList.Count; i++)
{
if (DateTime.Now - nodeList[i].LastUpdate > TimeSpan.FromHours(2))
{
removeList.Add(nodeList[i]);
nodeList.RemoveAt(i);
i--;
}
}
_context.Node.RemoveRange(removeList);
修改后的代码
[HttpGet("{localIp}")]
public IActionResult GetPeer(string localIp)
{
var ip = GetIp();
var nodeList = _context.Node.Where(temp => temp.MainIp == ip).ToList();
var removeList = new List<Node>();
for (var i = 0; i < nodeList.Count; i++)
{
if (DateTime.Now - nodeList[i].LastUpdate > TimeSpan.FromHours(2))
{
removeList.Add(nodeList[i]);
nodeList.RemoveAt(i);
i--;
}
}
var node = nodeList.FirstOrDefault(temp => temp.LocalIp == localIp);
if (node != null)
{
_context.Node.Remove(node);
nodeList.Remove(node);
}
_context.Node.Add(new Node
{
MainIp = ip,
LocalIp = localIp,
LastUpdate = DateTime.Now
});
_context.Node.RemoveRange(removeList);
_context.SaveChanges();
return Ok(string.Join(';', nodeList.Select(temp => temp.LocalIp)));
}
本文代码放在 github 欢迎小伙伴访问
这样就完成了简单的追踪服务器,可以看到只需要很少的代码。客户端访问方法是通过 get 加上自己的内网地址,然后读取返回内容,用分号分开多个地址
为什么我不用 json 返回?原因是我的客户端都是很少的代码开发的,不想使用 json 库,有些客户端使用 c 写的,所以只能使用简单 get 方法,返回的也是字符串
有小伙伴问如果有一个外网地址就访问一次,那是不是数据库的内容就会占用。其实我不关注这个问题,因为我使用内存数据库,我大概几天就关机一次。另外,按照每个客户端报告一个内网 IP 加上端口,也就是大概21个字符,加上外网 IP 和 Id 这些属性,可以看到数据量是非常小。假设每个客户端需要 1kb 的内存,那么 1G 的内存足够 100w 客户端,如果有这么多客户端,我就可以去打广告。然而我只有 10 个客户端
本文的代码可以修改一下在你的项目中使用,非常简单,但是效果不错
客户端需要获取本机 IP 地址 加上本机的端口,拼接链接访问
var localIp = string.Join(';',
GetLocalIpList().Select(temp => $"{temp}:{port}"));
上面代码的 GetLocalIpList 方法请看我博客 dotnet 获取本机 IP 地址方法
然后拼接链接
var url = $"http://p2p.api.acmx.xyz/api/peer/{localIp}";
上面的链接就是我部署的链接,如果小伙伴不想自己写服务器,也可以用我的。如果我关机了,这个链接就访问不到
我的服务器虽然很好,但是网很差,所以我设置了超时时间比较长
var httpClient = new HttpClient()
{
Timeout = TimeSpan.FromMinutes(10)
}; using (httpClient)
{
var remoteIp = await httpClient.GetStringAsync(url);
var ipList = GetIpList(remoteIp).Where(temp =>
!string.IsNullOrEmpty(temp.ip) && !string.IsNullOrEmpty(temp.port)).ToList();
}
这里的 GetIpList 就是解析服务器返回
private IEnumerable<(string ip, string port)> GetIpList(string remoteIp)
{
var ipList = remoteIp.Split(';');
foreach (var ip in ipList)
{
yield return IpRegex.Parse(ip);
}
} class IpRegex
{
public static (string ip, string port) Parse(string str)
{
var regex = new Regex(@"(\d+\.\d+\.\d+\.\d+):(\d+)"); var match = regex.Match(str); if (match.Success)
{
var ip = match.Groups[1].Value;
var port = match.Groups[2].Value; return (ip, port);
} return default;
}
}
现在拿到了一些 IP 和端口,尝试访问这些客户端看能不能访问
2019-11-1-asp-dotnet-core-简单开发P2P中央服务器的更多相关文章
- asp dotnet core 通过图片统计 csdn 用户访问
在 csdn 的访问统计里面,只能用 csdn 提供的访问统计,因为在 csdn 中不支持在博客加上 js 代码,也就是无法使用友盟等工具统计. 通过在 asp dotnet core 创建一个图片链 ...
- win10 uwp 手把手教你使用 asp dotnet core 做 cs 程序
本文是一个非常简单的博客,让大家知道如何使用 asp dot net core 做后台,使用 UWP 或 WPF 等做前台. 本文因为没有什么业务,也不想做管理系统,所以看到起来是很简单. Visua ...
- 《ASP.NET Core应用开发入门教程》与《ASP.NET Core 应用开发项目实战》正式出版
“全书之写印,实系初稿.有时公私琐务猬集,每写一句,三搁其笔:有时兴会淋漓,走笔疾书,絮絮不休:有时意趣萧索,执笔木坐,草草而止.每写一段,自助覆阅,辄摇其首,觉有大不妥者,即贴补重书,故剪刀浆糊乃不 ...
- 《ASP.NET Core项目开发实战入门》带你走进ASP.NET Core开发
<ASP.NET Core项目开发实战入门>从基础到实际项目开发部署带你走进ASP.NET Core开发. ASP.NET Core项目开发实战入门是基于ASP.NET Core 3.1 ...
- ASP.NET Core Web开发学习笔记-1介绍篇
ASP.NET Core Web开发学习笔记-1介绍篇 给大家说声报歉,从2012年个人情感破裂的那一天,本人的51CTO,CnBlogs,Csdn,QQ,Weboo就再也没有更新过.踏实的生活(曾辞 ...
- ASP.NET Core 企业级开发架构简介及框架汇总
企业开发框架包括垂直方向架构和水平方向架构.垂直方向架构是指一个应用程序的由下到上叠加多层的架构,同时这样的程序又叫整体式程序.水平方向架构是指将大应用分成若干小的应用实现系统功能的架构,同时这样的系 ...
- 使用Asp.Net Core MVC 开发项目实践[第一篇:项目结构说明]
先从下图看整体项目结构: Mango.Manager: 为后台管理项目 Mango.Web: 为前台项目 Mango.Framework.Core: 为常用的基础操作类项目 Mango.Framewo ...
- ASP.NET Core 企业级开发架构简介及框架汇总 (转载)
ASP.NET Core 企业开发架构概述 企业开发框架包括垂直方向架构和水平方向架构.垂直方向架构是指一个应用程序的由下到上叠加多层的架构,同时这样的程序又叫整体式程序.水平方向架构是指将大应用分成 ...
- win10 uwp 使用 asp dotnet core 做图床服务器客户端
原文 win10 uwp 使用 asp dotnet core 做图床服务器客户端 本文告诉大家如何在 UWP 做客户端和 asp dotnet core 做服务器端来做一个图床工具 服务器端 从 ...
随机推荐
- 【python之路35】FTP文件断电续传作业
开发一个支持多用户在线FTP程序: 要求: 1.用户MD5加密认证 2.允许同时多用户登陆(socketserver) 3.执行命令 4.上传文件 文件传输过程中显示进度条 支持文件的断点续传
- NYoj536 矩阵链乘
经典问题没啥说的 #include<stdio.h> #include<string.h> #define max 100+1 #define min(a,b) (a<b ...
- 配置 CentOS 7 的网络,及重命名网卡名
Centos 安装时应配置网络,如果当时没配置好,则装完系统后, 也可通过修改配置文件并重启网络服务进行配置. 说明:CentOS 7.0默认安装好之后是没有自动开启网络连接的! cd /etc/sy ...
- webpack4进阶配置
移动端CSS px自动转换成rem 需要两步来实现: px2rem-loader 在构建阶段将px转换成rem lib-flexible 页面渲染时动态计算根元素的font-size值(手机淘宝开源库 ...
- spring boot 的 ApplicationContext 及 getbean
在spring中,我们通过如下代码取得一个spring托管类: ApplicationContext ac = new FileSystemXmlApplicationContext("ap ...
- Phpstrom 配置php版本语法支持
- HDU 2639 第K大背包问题
//状态方程和01背包类似,dp[j][k]表示背包容量为j的第k大背包的值.......... //应当注意的是此时dp[j][1.....k]应当是递减的.................... ...
- Python学习笔记(一)初识Python以及安装Python
一.Python简介 1.Python发展史 Python 是由 Guido van Rossum 在八十年代末和九十年代初,在荷兰国家数学和计算机科学研究所设计出来的. Python 本身也是由诸多 ...
- 生成mysql数据字典
data_dictionary.php <?php /** * 生成mysql数据字典 */ header("Content-type: text/html; charset=utf- ...
- 介绍Provide以及Inject
介绍 Vue 的 Provide 以及 Inject Provide 以及 Inject 是 Vue 中用于祖先元素向其所有后台元素注入依赖的接口. 具体用法 // Data.vue ... expo ...