ASP.NET Core 开源论坛项目 NETCoreBBS

    ASP.NET Core 开源论坛项目 NETCoreBBS

    • 商品编号:
      #12736002_127
      • 原价:
        免费
      • 会员价:
        免费
    • 版本:
      • V1.0.0
    • 数量:
      库存(不限)

    购物车中已存在此商品,请在购物车中操作单击跳转购物车

    • 大小:0
    • 编码格式:UTF-8
    • 开发环境:Visual Studio 2017
    • 源码架构:B/S
    • 开源协议:BSD协议
    • 开发者:LineZero's Blog
    • 版本工具管理:GitHub
    • 开发语言:C#
    • 数据库:MySql

    ASP.NET Core 轻量化开源论坛项目,ASP.NET Core Light forum NETCoreBBS

    采用 ASP.NET Core + EF Core Sqlite + Bootstrap 开发。

    开发

    1. 下载源码
    2. 使用 Visual Studio 2017 打开 NetCoreBBS.sln
    3. 点击 调试->开始调试 即可运行起来,或者直接点击工具栏上的NetCoreBBS即可。

    注意:默认为80端口,可能会和本地端口冲突,可以到Program.cs 中更改 .UseUrls("http://*:80"),然后更改启动URL既可。

    功能

    1. 节点功能
    2. 主题发布
    3. 主题回复
    4. 主题筛选
    5. 用户登录注册
    6. 主题置顶
    7. 后台管理
    8. 个人中心

    技术点大合集

    架构 Clean Architecture


    1. Areas

    重点代码:

    app.UseMvc(routes =>
    {
        routes.MapRoute(
            name: "areaRoute",
            template: "{area:exists}/{controller}/{action}",
            defaults: new { action = "Index" });
        routes.MapRoute(
            name: "default",
            template: "{controller=Home}/{action=Index}/{id?}");
    });

    增加一个 areaRoute ,然后添加对应的Areas 文件夹,然后Areas里的控制器里加上  [Area("Admin")] 。

    2. ViewComponents

    在项目里的ViewComponents 文件夹,注意对应视图在 Views\Shared\Components 文件夹里。

    3. Middleware

    RequestIPMiddleware 记录ip及相关信息的中间件

    public class RequestIPMiddleware
    {
        private readonly RequestDelegate _next;
        private readonly ILogger _logger;
    
        public RequestIPMiddleware(RequestDelegate next)
        {
            _next = next;
            _logger = LogManager.GetCurrentClassLogger();
        }
    
        public async Task Invoke(HttpContext httpContext)
        {
            var url = httpContext.Request.Path.ToString();
            if (!(url.Contains("/css") || url.Contains("/js") || url.Contains("/images") || url.Contains("/lib")))
            {
                _logger.Info($"Url:{url} IP:{httpContext.Connection.RemoteIpAddress.ToString()} 时间:{DateTime.Now}");
            }
            await _next(httpContext);
        }
    }
    
    public static class RequestIPMiddlewareExtensions
    {
        public static IApplicationBuilder UseRequestIPMiddleware(this IApplicationBuilder builder)
        {
            return builder.UseMiddleware<RequestIPMiddleware>();
        }
    }

    4. Identity

    集成Identity ,扩展User表,自定义用户表。

    权限策略

    services.AddAuthorization(options =>
    {
        options.AddPolicy(
            "Admin",
            authBuilder =>
            {
                authBuilder.RequireClaim("Admin", "Allowed");
            });
    });

    注册登录密码复杂度

    services.AddIdentity<User, IdentityRole>(options =>
    {
        options.Password = new PasswordOptions() {
            RequireNonAlphanumeric = false,
            RequireUppercase=false
        };
    }).AddEntityFrameworkStores<DataContext>().AddDefaultTokenProviders();

    5. EF Core

    EF Core 采用Sqlite 数据库。

    读取配置文件

    services.AddDbContext<DataContext>(options => options.UseSqlite(Configuration.GetConnectionString("DefaultConnection")));

    使用代码初始化数据库

    private void InitializeNetCoreBBSDatabase(IServiceProvider serviceProvider)
    {
        using (var serviceScope = serviceProvider.GetRequiredService<IServiceScopeFactory>().CreateScope())
        {
            var db = serviceScope.ServiceProvider.GetService<DataContext>();
            db.Database.Migrate();
            if (db.TopicNodes.Count() == 0)
            {
                db.TopicNodes.AddRange(GetTopicNodes());
                db.SaveChanges();
            }
        }
    }

    项目分层 DataContext 在 Infrastructure,使用dotnet  ef 命令注意事项 

    dotnet ef migrations add InitMigration --startup-project ../NetCoreBBS/NetCoreBBS.csproj

    更新指定字段,不用先查询实体。

    public IActionResult EditSave(Topic topic)
    {
        _context.Attach(topic);
        _context.Entry(topic).Property(r => r.Title).IsModified = true;
        _context.Entry(topic).Property(r => r.Content).IsModified = true;
        _context.SaveChanges();
        return RedirectToAction("Index");
    }

    6. Configuration

    读取链接字符串 Configuration.GetConnectionString("DefaultConnection")

    7. Partial Views

    _LoginPartial.cshtml 头部登录部分分布视图

    _PagerPartial.cshtml 分页分布视图

    @{
        var pageindex = Convert.ToInt32(ViewBag.PageIndex);
        var pagecount = Convert.ToInt32(ViewBag.PageCount);
        pagecount = pagecount == 0 ? 1 : pagecount;
        pageindex = pageindex > pagecount ? pagecount : pageindex;
        var path = Context.Request.Path.Value;
        var query = string.Empty;
        var querys = Context.Request.Query;
        foreach (var item in querys)
        {
            if (!item.Key.Equals("page"))
            {
                query += $"{item.Key}={item.Value}&";
            }
        }
        query = query == string.Empty ? "?" : "?" + query;
        path += query;
        var pagestart = pageindex - 2 > 0 ? pageindex - 2 : 1;
        var pageend = pagestart + 5 >= pagecount ? pagecount : pagestart + 5;
    }
    <ul class="pagination">
        <li class="prev previous_page @(pageindex == 1 ? "disabled" : "")">
            <a href="@(pageindex==1?"#":$"{path}page={pageindex - 1}")">&#8592; 上一页</a>
        </li>
        <li @(pageindex == 1 ? "class=active" : "")><a rel="start" href="@(path)page=1">1</a></li>
        @if (pagestart > 2)
        {
            <li class="disabled"><a href="#">&hellip;</a></li>
        }
        @for (int i = pagestart; i < pageend; i++)
        {
            if (i > 1)
            {
                <li @(pageindex == i ? "class=active" : "")><a rel="next" href="@(path)page=@i">@i</a></li>
            }
        }
        @if (pageend < pagecount)
        {
            <li class="disabled"><a href="#">&hellip;</a></li>
        }
        @if (pagecount > 1)
        {
            <li @(pageindex == pagecount ? "class=active" : "")><a rel="end" href="@(path)page=@pagecount">@pagecount</a></li>
        }
        <li class="next next_page @(pageindex==pagecount?"disabled":"")">
            <a rel="next" href="@(pageindex==pagecount?"#":$"{path}page={pageindex + 1}")">下一页 &#8594;</a>
        </li>
    </ul>

    写的不是很好,可以优化成TagHelper。

    8. Injecting Services Into Views

    @inject SignInManager SignInManager

    @inject 关键字

    9. Dependency Injection and Controllers

    public IActionResult Index([FromServices]IUserServices user)

    FromServices 在指定Action注入,也可以使用构造函数注入。

    private ITopicRepository _topic;
    private IRepository<TopicNode> _node;
    public UserManager<User> UserManager { get; }
    public HomeController(ITopicRepository topic, IRepository<TopicNode> node, UserManager<User> userManager)
    {
        _topic = topic;
        _node = node;
        UserManager = userManager;
    }

    10.发布

    之前写过对应的发布文章 ASP.NET Core 发布至Linux生产环境 Ubuntu 系统

    由于project.json 改成csproj,发布有所变动。

    默认发布还是相同 dotnet publish,自带运行时发布时更改csproj。

    编辑 NetCoreBBS.csproj

    <RuntimeIdentifiers>ubuntu.14.04-x64</RuntimeIdentifiers>

    后续同样是 dotnet publish -r ubuntu.14.04-x64

    注意这个节点,默认发布的,服务器也要安装相同版本的runtime。

    <RuntimeFrameworkVersion>1.0.0</RuntimeFrameworkVersion>

    代码里面还有一些大家可以自己去挖掘。

    NETCoreBBS 在RC2 的时候就已经开始了,有很多人应该已经看过这个项目,这篇文章是让大家更清楚的了解这个项目。

    ASP.NET Core跨平台开发从入门到实战》  京东  淘宝   亚马逊   当当


    权利声明:本站所有商品信息、客户评价等信息是初心商城重要的数据资源,未经许可,禁止非法转载使用。 注:本站商品信息均来自初心商城,其真实性、准确性和合法性由初心商城负责。

                    初心源说明:初心商城主要为程序员提供开发基础的代码源以及成熟项目,网站中所有的商品有提供收费版本的, 也有提供免费版本的,按照大家各自不同的需求进行购买。实实在在的让程序员只用专注于自己的业务实现你的小梦想, 如果您对我们的成果表示认同并且觉得对你有所帮助我们愿意接受来自各方面的支持^_^。

                    支持:用手机扫描二维码支付

                    支付宝支持我们 微信支持我们

                    您的支持将被用于:
                    1、持续深入的上传更多更好的源代码
                    2、建立更加完善的技术社区
                    3、完善现在系统出现各种问题
                    4、购买域名和租赁服务器

                    1、交易规则

                    2、发货方式

                    1、自动:在上方保障服务中标有自动发货的商品,拍下后,将会自动收到来自卖家的商品获取(下载)链接

                    2、手动:在上方保障服务中标有手动发货的商品,拍下后,卖家会收到邮件,也可通过QQ或订单中的电话联系对方。

                    3、退款说明

                    1、描述:源码描述(含标题)与实际源码不一致的(例:描述PHP实际为ASP、描述的功能实际缺少、版本不符等)

                    2、演示:有演示站时,与实际源码小于95%一致的(但描述中有"不保证完全一样、有变化的可能性"类似显著声明的除外)

                    3、发货:手动发货源码,在卖家未发货前,已申请退款的

                    4、服务:卖家不提供安装服务或需额外收费的(但描述中有显著声明的除外)

                    5、其它:如质量方面的硬性常规问题等

                    备注:经核实符合上述任一,均支持退款,但卖家予以积极解决问题则除外。交易中的商品,卖家无法对描述进行修改!

                    4、注意事项

                    1、客户买完之后未确认收货,将不会收到下载地址和下载码,确认收货之后才能收到下载地址和下载码。

                    2、在未拍下前,双方在QQ上所商定的内容,亦可成为纠纷评判依据(商定与描述冲突时,商定为准);

                    3、在商品同时有网站演示与图片演示,且站演与图演不一致时,默认按图演作为纠纷评判依据(特别声明或有商定除外);

                    4、在没有"无任何正当退款依据"的前提下,写有"一旦售出,概不支持退款"等类似的声明,视为无效声明;

                    5、虽然交易产生纠纷的几率很小,但请尽量保留如聊天记录这样的重要信息,以防产生纠纷时出现问题不明确的情况。

                    5、交易声明

                    1、本站作为直卖平台,依据交易合同(商品描述、交易前商定的内容)来保障交易的安全及买卖双方的权益;

                    2、非平台线上交易的商品,出现任何后果均与本站无关;无论卖家以何理由要求线下交易的,请联系管理举报。

                    初心Logo

                    初心商城| 初心系列| 初心博客| 版本历史| 通知公告| 系统反馈

                    © 2016-2019 上海吾永网络科技有限公司 保留所有权利 沪ICP备19028491号
                    违法和不良信息举报电话:186-2950-9347,本网站所列数据,除特殊说明,所有数据均出自我工作室
                    本网站兼容所有主流浏览器,不支持手机自适应

                    返回顶部小火箭