ASP.NET Core 開源論壇項目 NETCoreBBS

来源:http://www.cnblogs.com/linezero/archive/2017/07/18/NETCoreBBS.html
-Advertisement-
Play Games

ASP.NET Core 輕量化開源論壇項目,ASP.NET Core Light forum NETCoreBBS 採用 ASP.NET Core + EF Core Sqlite + Bootstrap 開發。 GitHub: https://github.com/linezero/NETCor ...


ASP.NET Core 輕量化開源論壇項目,ASP.NET Core Light forum NETCoreBBS

採用 ASP.NET Core + EF Core Sqlite + Bootstrap 開發。

GitHub: https://github.com/linezero/NETCoreBBS

開發

  1. git clone https://github.com/linezero/NETCoreBBS.git
  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>();
        }
    }
View Code

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<User> 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 的時候就已經開始了,有很多人應該已經看過這個項目,這篇文章是讓大家更清楚的瞭解這個項目。

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 來實驗室將近一個月,幾乎沒碰window,一直在ubuntu下開發,然後想總結一下最近常用的linux指令。 ...
  • 每個人電腦裡面都有些秘密,但是別人需要使用你的電腦時,有可能會看到,但是我們又不想讓別人發現時,我們可以將其隱藏,那麼別人就不會看到了。360文件保險柜、騰訊電腦管家等等。使用軟體繁瑣軟體過大還會拖慢系統,況且大多軟體還不能把整個盤符進行加密。下麵給大家分享一種隱藏整個盤符的方法 今天我給大家介紹的 ...
  • 介紹如何在紅帽6.5的系統中搭建Samba伺服器,windows訪問是需要進行用戶名和密碼的驗證。 ...
  • ls:列出目錄內容(預設的就是當前工作目錄了) 格式:命令 [選項] [參數] 選項: -a:列出所有文件、目錄(包含隱藏文件、目錄) -l:以長格式的方式列出 -d:僅列出目錄本身 -h:配合-l選項使用,以友好的方式,顯示文件的大小(若不加-h,預設文件大小是以byte來顯示); 實例: [ro ...
  • 1. 命令的概念 命令的執行過程 系統第一次執行外部命令時Hash緩存表為空,系統會先從PTAH路徑下尋找命令,找到後會將路徑加入到Hasa緩存中,當再次執行此命令時會直接從Hash的路徑下執行,如果存在直接執行,如果不存在將繼續從PATH下的路徑繼續查找,Hash表可以提高命令的調用速率。 命令的 ...
  • 從VSS上獲取以前的老項目,編譯時報System.Runtime.CompilerServices.ExtensionAttribute..ctor 網上寫的“刪除 Newtonsoft.Json.Net20.dll 後重新引用”,並沒有解決這個問題。 原文首發在我的主力博客 http://anfo ...
  • 一. Dapper 簡介 一個效率比較高的微型ORM。 二 . Dapper.Rainbow Dapper的擴展,在這個擴展裡面實現了 Dynamic 的 插入和更新,這個對於使用動態類型的對象很有用,可以少建很多貧血型的實體類。 三 . Dynamic Dynamic 這是一把雙刃劍,在提高效率的 ...
  • 作者在做短鏈接功能時,url參數裡帶了&字元,結果無法轉換。後來查了一下,發現可以用其它符號代替。下麵是對應表 + URL 中+號表示空格 %2B 空格 URL中的空格可以用+號或者編碼 %20 / 分隔目錄和子目錄 %2F ? 分隔實際的URL和參數 %3F % 指定特殊字元 %25 # 表示書簽 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...