国产 无码 综合区,色欲AV无码国产永久播放,无码天堂亚洲国产AV,国产日韩欧美女同一区二区

基于ASP.NET ZERO,開發(fā)SaaS版供應(yīng)鏈管理系統(tǒng)

這篇具有很好參考價值的文章主要介紹了基于ASP.NET ZERO,開發(fā)SaaS版供應(yīng)鏈管理系統(tǒng)。希望對大家有所幫助。如果存在錯誤或未考慮完全的地方,請大家不吝賜教,您也可以點擊"舉報違法"按鈕提交疑問。

前言

2018年下半年,公司決定開發(fā)一款SaaS版行業(yè)供應(yīng)鏈管理系統(tǒng),經(jīng)過選型,確定采用ABP(ASP.NET Boilerplate)框架。為了加快開發(fā)效率,購買了商業(yè)版的 ASP.NET ZERO(以下簡稱ZERO),選擇ASP.NET Core + Angular的SPA框架進(jìn)行系統(tǒng)開發(fā)(ABP.IO屆時剛剛起步,還很不成熟,因此沒有選用)。

關(guān)于ABP與ZERO,園子里已經(jīng)有諸多介紹,因此不再贅述。本文側(cè)重介紹我們基于ZERO框架開發(fā)系統(tǒng)過程中進(jìn)行的一些優(yōu)化、調(diào)整、擴(kuò)展部分的內(nèi)容,方便有需要的園友們了解或者參考。

系統(tǒng)架構(gòu)

系統(tǒng)在2020年7月發(fā)布上線(部署在阿里云上),目前有超過500家企業(yè)/團(tuán)隊/個人注冊體驗,感興趣的園友可以在此系統(tǒng)的著陸網(wǎng)站 scm.plus 注冊一個免費賬號體驗一下,歡迎大家的批評與建議。

基于ASP.NET ZERO,開發(fā)SaaS版供應(yīng)鏈管理系統(tǒng)

ZERO框架總體上來說還是不錯的,可以快速的上手,集成的通用功能(版本、租戶、角色、用戶、設(shè)置等)初期都可以直接使用,但還達(dá)不到直接發(fā)布使用的水準(zhǔn),需要經(jīng)過諸多的優(yōu)化調(diào)整擴(kuò)展后才能發(fā)布上線。

A 后端(ASP.NET Core)部分

0、移除不需要的功能:Chat、SignalR、DynamicProperty、GraphQL、IdentityServer4。

基于系統(tǒng)功能定位,移除的這些不需要的功能,使系統(tǒng)盡可能的精簡。

1、Migrations內(nèi)移除Designer.cs。

在我們的開發(fā)環(huán)境內(nèi),經(jīng)過測試與驗證,使用mysql數(shù)據(jù)庫時候,可以安全移除add-migration時候生成的龐大的Designer.cs文件。移除Designer.cs文件時候,需要把該文件內(nèi)的DbContext與Migration聲明語句移到對應(yīng)的migration.cs文件內(nèi):文章來源地址http://www.zghlxwxcb.cn/news/detail-695050.html

[DbContext(typeof(SCMDbContext))]
[Migration("20230811015119_Upgraded_To_Abp_8_3")]
public partial class Upgraded_To_Abp_8_3 : Migration
{
   //...
}

2、替換必要的功能包,確保系統(tǒng)后端可以部署到linux環(huán)境:

  • 使用SkiaSharp替換System.Drawing.Common;
  • 使用EPPlus替換NPOI。

3、停用系統(tǒng)默認(rèn)的外部登錄( Facebook、Google、Microsoft、Twitter等),添加微信掃碼與小程序登錄。

4、停用系統(tǒng)默認(rèn)的支付選項( Paypal、Stripe等),添加支付寶(Alipay)支付。

5、Excel文件上傳,ZERO默認(rèn)沒有實現(xiàn),需要自行添加Excel文件的上傳與導(dǎo)入功能:

  • Excel文件上傳后先緩存該文件;
  • 創(chuàng)建一個后臺Job(HangFire)執(zhí)行Excel文件的讀取、處理等;
  • Job發(fā)送執(zhí)行后的結(jié)果(消息通知)。
[HttpPost]
[AbpMvcAuthorize(AppPermissions.Pages_Txxxs_Excel_Import)]
public async Task<JsonResult> ImportFromExcel()
{
    try
    {
        var jobArgs = await DoImportFromExcelJobArgs(AbpSession.ToUserIdentifier());

        var queueState = new EnqueuedState(GetJobQueueName());
        IBackgroundJobClient hangFireClient = new BackgroundJobClient();
        hangFireClient.Create<ImportTxxxsToExcelJob>(x => x.ExecuteAsync(jobArgs), queueState);

        return Json(new AjaxResponse(new { }));
    }
    catch (Exception ex)
    {
        return Json(new AjaxResponse(new ErrorInfo(ex.Message)));
    }
}

6、圖片與文件上傳存儲,ZERO的默認(rèn)實現(xiàn)是保存上傳的圖片文件到數(shù)據(jù)庫內(nèi),需要改造存儲到OSS中:

  • 使用MD5哈希前綴,生成OSS文件對象的名稱(含path),提高OSS并發(fā)性能:
private static string GetOssObjName(int? tenantId, Guid id, bool isThumbnail)
{
    string tid = (tenantId ?? 0).ToString();
    string ext = isThumbnail ? "thu" : "ori"; //thu - 縮略圖、ori - 原圖/原文件
    string hashStr = BitConverter.ToString(MD5.HashData(Encoding.UTF8.GetBytes(tid)), 0).Replace("-", string.Empty).ToLower();

    return $"{hashStr[..4]}/{tid}/{id}.{ext}";
}
  • 若OSS未啟用或者上傳失敗,則直接存儲到數(shù)據(jù)庫中:
public async Task SaveAsync(BinaryObject file)
{
    if (file?.Bytes == null) { return; }

    //1、OSS上傳,成功后直接返回
    if (OssPutObject(file.TenantId, file.Id, file.Bytes, isThumbnail: false)) { return; } 

    //2、若OSS未啟用或者上傳失敗,則直接上傳到數(shù)據(jù)庫中
    await _binaryObjectRepository.InsertAsync(file);
}
  • 獲取時候遵循一樣的邏輯:若OSS未啟用或者獲取不到,則直接自數(shù)據(jù)庫中獲取;自數(shù)據(jù)庫獲取成功后要同步數(shù)據(jù)庫中記錄到OSS中。

7、Webhook功能,需要改造支持推送數(shù)據(jù)到第三方接口,如:企業(yè)微信群、釘釘群、聚水潭API等:

  • 重寫WebhookManager的SignWebhookRequest方法;
  • 重寫DefaultWebhookSender的CreateWebhookRequestMessage、AddAdditionalHeaders、SendHttpRequest方法;
  • 緩存Webhook Subscription:
private SCMWebhookCacheItem SetAndGetCache(int? tenantId, string keyName = "SubscriptionCount")
{
   int tid = tenantId ?? 0; var cacheKey = $"{keyName}-{tid}";

   return _cacheManager.GetSCMWebhookCache().Get(cacheKey, () =>
   {
        int count = 0;
        var names = new Dictionary<string, List<WebhookSubscription>>();

        UnitOfWorkManager.WithUnitOfWork(() =>
        {
            using (UnitOfWorkManager.Current.SetTenantId(tenantId))
            {
                if (_featureChecker.IsEnabled(tid, "SCM.H"))            //Feature 核查
                {
                    var items = _webhookSubscriptionRepository.GetAllList(e => e.TenantId == tenantId && e.IsActive == true);
                    count = items.Count;

                    foreach (var item in items)
                    {
                        if (string.IsNullOrWhiteSpace(item.Webhooks)) { continue; }
                        var whNames = JsonHelper.DeserializeObject<string[]>(item.Webhooks); if (whNames == null) { continue; }
                        foreach (string whName in whNames)
                        {
                            if (names.ContainsKey(whName))
                            {
                                names[whName].Add(item.ToWebhookSubscription());
                            }
                            else
                            {
                                names.Add(whName, new List<WebhookSubscription> { item.ToWebhookSubscription() });
                            }
                        }
                    }
                }
            }
        });

        return new SCMWebhookCacheItem(count, names);
    });
}

8、在WebHostModule中設(shè)定只有一臺Server執(zhí)行后臺Work,避免多臺Server重復(fù)執(zhí)行:

public override void PostInitialize()
{
    //...

    string defaultEndsWith = _appConfiguration["Job:DefaultEndsWith"];
    if (string.IsNullOrWhiteSpace(defaultEndsWith)) { defaultEndsWith = "01"; }
    if (AppVersionHelper.MachineName.EndsWith(defaultEndsWith))
    {
        var workManager = IocManager.Resolve<IBackgroundWorkerManager>();

        workManager.Add(IocManager.Resolve<SubscriptionExpirationCheckWorker>());
        workManager.Add(IocManager.Resolve<SubscriptionExpireEmailNotifierWorker>());
        workManager.Add(IocManager.Resolve<SubscriptionPaymentsCheckWorker>());
        workManager.Add(IocManager.Resolve<ExpiredAuditLogDeleterWorker>());
        workManager.Add(IocManager.Resolve<PasswordExpirationBackgroundWorker>());
    }

    //...
}

9、限流功能,ZERO默認(rèn)沒有實現(xiàn),通過添加AspNetCoreRateLimit中間件集成限流功能:

  • 采用客戶端ID(ClientRateLimiting)進(jìn)行設(shè)置;
  • 重寫RateLimitConfigurationRegisterResolvers方法,添加定制化的ClientIpHeaderResolveContributor:存在客戶端ID則優(yōu)先獲取,反之獲取客戶端的IP:
    public class RateLimitConfigurationExtensions : RateLimitConfiguration  
    {
        //...
        public override void RegisterResolvers()
        {
            ClientResolvers.Add(new ClientIpHeaderResolveContributor(SCMConsts.TenantIdCookieName));
        }
    }

    public class ClientIpHeaderResolveContributor : IClientResolveContributor
    {
        private readonly string _headerName;

        public ClientIpHeaderResolveContributor(string headerName)
        {
            _headerName = headerName;     
        }

        public Task<string> ResolveClientAsync(HttpContext httpContext)
        {
            IPAddress clientIp = null;

            var headers = httpContext?.Request?.Headers;
            if (headers != null && headers.Count > 0)
            {
                if (headers.ContainsKey(_headerName))                               //0 scm_tid
                {
                    string clientId = headers[_headerName].ToString();
                    if (!string.IsNullOrWhiteSpace(clientId))
                    {
                        return Task.FromResult(clientId);
                    }
                }

                try
                {
                    if (headers.ContainsKey("X-Real-IP"))                           //1 X-Real-IP
                    {
                        clientIp = IpAddressUtil.ParseIp(headers["X-Real-IP"].ToString());
                    }
                    
                    if (clientIp == null && headers.ContainsKey("X-Forwarded-For")) //2 X-Forwarded-For
                    {
                        clientIp = IpAddressUtil.ParseIp(headers["X-Forwarded-For"].ToString());
                    }
                }
                catch {}

                clientIp ??= httpContext?.Connection?.RemoteIpAddress;             //3 RemoteIpAddress
            }

            return Task.FromResult(clientIp?.ToString());
        }
    }

B 前端(Angular)部分

0、類似后端,移除不需要的功能:Chat、SignalR、DynamicProperty等。

1、拆分精簡service-proxies.ts文件:

  • ZERO使用NSwag生成前端的TypeScript代碼文件service-proxies.ts,全部模塊的都生成到一個文件內(nèi),導(dǎo)致該文件非常龐大,最終編譯生成的main.js接近4MB;
  • 按系統(tǒng)執(zhí)行層次,拆分service-proxies.ts為多個文件,精簡其中的共用代碼,調(diào)整module的調(diào)用、拆分、懶加載等,最終大幅度減少了main.js的大?。壳笆?87KB)。

2、優(yōu)化表格組件primeng table,實現(xiàn)客戶端表格使用狀態(tài)的本地存儲:表格列寬、列順序、列顯示隱藏、列固定、分頁設(shè)定等。

3、實現(xiàn)客戶端的卡片視圖功能。

4、集成ng-lazyload-image,實現(xiàn)圖片展示的懶加載。

5、集成ngx-markdown,實現(xiàn)markdown格式的在線幫助。

6、業(yè)務(wù)組件設(shè)置為獨立組件,ChangeDetectionStrategy設(shè)置為OnPush:

@Component({
    changeDetection: ChangeDetectionStrategy.OnPush,
    templateUrl: './txxxs.component.html',
    standalone: true,
    imports: [...]
})
export class TxxxsComponent extends AppComponentBase {
    ...
    constructor(
        injector: Injector,
        changeDetector: ChangeDetectorRef,
    ) {
        super(injector);
        setInterval(() => { changeDetector.markForCheck(); }, AppConsts.ChangeDetectorMS);
    }
    ...
}

7、儀表盤升級為工作臺,除了可以添加圖表外,也可以添加業(yè)務(wù)組件(獨立組件)。

8、路由直接鏈接業(yè)務(wù)組件,實現(xiàn)懶加載:

import { Route } from '@angular/router';
export default [
    { path: 'p120303/t12030301s', loadComponent: () => import('./t12030301s.component').then(c => c.T12030301sComponent), ... },
    { path: 'p120405/t12040501s', loadComponent: () => import('./t12040501s.component').then(c => c.T12040501sComponent), ... },
    { path: 'p120405/t12040502s', loadComponent: () => import('./t12040502s.component').then(c => c.T12040502sComponent), ... },
] as Route[];

9、通過webpackInclude,減少打包后的文件數(shù)量;使用webpackChunkName設(shè)定打包后的文件名:

function registerLocales(
    resolve: (value?: boolean | Promise<boolean>) => void,
    reject: any,
    spinnerService: NgxSpinnerService
) {
    if (shouldLoadLocale()) {
        let angularLocale = convertAbpLocaleToAngularLocale(abp.localization.currentLanguage.name);
        import(
            /* webpackInclude: /(en|en-GB|zh|zh-Hans|zh-Hant)\.mjs$/ */
            /* webpackChunkName: "angular-common-locales" */
            `/node_modules/@angular/common/locales/${angularLocale}.mjs`).then((module) => {
                registerLocaleData(module.default);
                resolve(true);
                spinnerService.hide();
            }, reject);
    } else {
        resolve(true);
        spinnerService.hide();
    }
}

C 小程序(Vue3)部分

后端部分已經(jīng)實現(xiàn)小程序集成微信登錄,后端輸出的語言文本與API等小程序都可以直接調(diào)用,因此小程序的開發(fā)實現(xiàn)就相對比較容易,只需要實現(xiàn)必要的UI界面即可。

  • 小程序采用 uni-app(vue3) 框架進(jìn)行開發(fā),整體效率較高。
  • 有部分代碼可以基于前端 Angular 的代碼復(fù)制后稍加調(diào)整后即可使用。
  • 目前只輸出了微信小程序,方便同企業(yè)微信群內(nèi)的消息推送一體化集成。
  • 后端部分實現(xiàn)的Webhook功能,可以直接推送消息到企業(yè)微信群內(nèi),用戶可以單擊消息卡片,直接打開微信小程序內(nèi)對應(yīng)的頁面,查看數(shù)據(jù)或者進(jìn)行其他的維護(hù)操作。
  • 小程序中需要在onLaunch中進(jìn)行路由守衛(wèi)(登錄攔截),以處理通過分享單獨頁面或者企業(yè)微信群內(nèi)通過消息卡片直接打開小程序頁面的權(quán)限核查。

總結(jié)

若沒有優(yōu)秀的工具框架支持,開發(fā)SaaS化系統(tǒng)并不是一件容易的事。基于ABP框架,使用ZERO工具,極大的降低了開發(fā)SaaS化系統(tǒng)的門檻,也促成了這套系統(tǒng)的實踐與發(fā)布。

本文簡要介紹了我們實現(xiàn)這套系統(tǒng)中的一些要點,供有需要的人了解參考,就算是拋磚引玉吧!

到了這里,關(guān)于基于ASP.NET ZERO,開發(fā)SaaS版供應(yīng)鏈管理系統(tǒng)的文章就介紹完了。如果您還想了解更多內(nèi)容,請在右上角搜索TOY模板網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章,希望大家以后多多支持TOY模板網(wǎng)!

本文來自互聯(lián)網(wǎng)用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務(wù),不擁有所有權(quán),不承擔(dān)相關(guān)法律責(zé)任。如若轉(zhuǎn)載,請注明出處: 如若內(nèi)容造成侵權(quán)/違法違規(guī)/事實不符,請點擊違法舉報進(jìn)行投訴反饋,一經(jīng)查實,立即刪除!

領(lǐng)支付寶紅包贊助服務(wù)器費用

相關(guān)文章

  • SCM供應(yīng)鏈管理系統(tǒng)解決方案:實現(xiàn)供應(yīng)可視化、管理信息化,成本可控化

    SCM供應(yīng)鏈管理系統(tǒng)解決方案:實現(xiàn)供應(yīng)可視化、管理信息化,成本可控化

    SCM供應(yīng)鏈管理系統(tǒng) 是基于協(xié)同供應(yīng)鏈的管理思想,加上配合各實體行業(yè)的多樣化業(yè)務(wù)需求,讓企業(yè)管理過程、操作流程和信息系統(tǒng)緊密配合,輕松做到業(yè)務(wù)各環(huán)節(jié)無縫鏈接,形成集齊物流、信息流、單證流、商流和資金流五流合一的供應(yīng)鏈SCM管理系統(tǒng)。供應(yīng)鏈圍繞核心企業(yè)對

    2024年02月02日
    瀏覽(24)
  • 微信小程序的物料產(chǎn)品采購供應(yīng)鏈管理系統(tǒng)

    微信小程序的物料產(chǎn)品采購供應(yīng)鏈管理系統(tǒng)

    本論文主要論述了如何使用小程序與java語言開發(fā)一個微信小程序的供應(yīng)鏈管理系統(tǒng),本系統(tǒng)將嚴(yán)格按照軟件開發(fā)流程進(jìn)行各個階段的工作,后臺采用mysql數(shù)據(jù)庫,面向?qū)ο缶幊趟枷脒M(jìn)行產(chǎn)品開發(fā)。在引言中,系統(tǒng)將論述微信小程序的供應(yīng)鏈管理系統(tǒng)的當(dāng)前背景以及系統(tǒng)開發(fā)的

    2024年02月21日
    瀏覽(25)
  • 數(shù)商云:SCM系統(tǒng)助力醫(yī)療器械企業(yè)緊跟時代變局 搭建供應(yīng)鏈管理平臺

    數(shù)商云:SCM系統(tǒng)助力醫(yī)療器械企業(yè)緊跟時代變局 搭建供應(yīng)鏈管理平臺

    醫(yī)療器械產(chǎn)業(yè)是生物工程、電子信息和醫(yī)學(xué)影像等高新技術(shù)領(lǐng)域復(fù)合交叉的知識密集型、資金密集型產(chǎn)業(yè)。作為關(guān)系到人類生命健康的戰(zhàn)略性新興產(chǎn)業(yè),在龐大而穩(wěn)定的市場需求下,全球醫(yī)療器械產(chǎn)業(yè)長期以來一直保持著良好的增長勢頭,2020年全球醫(yī)療器械規(guī)模突破5000億美

    2024年02月06日
    瀏覽(27)
  • 基于區(qū)塊鏈技術(shù)改進(jìn)供應(yīng)鏈管理

    目錄 一、簡介 二、編寫智能合約 三、部署智能合約 四、開發(fā)后端服務(wù)器 五、創(chuàng)建前端界面 六、集成物聯(lián)網(wǎng)設(shè)備(可選) 在本文中,我們將探討如何利用區(qū)塊鏈技術(shù)改進(jìn)供應(yīng)鏈管理。我們將通過一個簡化的示例項目,展示如何構(gòu)建一個基于以太坊的供應(yīng)鏈管理系統(tǒng)。我們將

    2023年04月21日
    瀏覽(24)
  • 基于區(qū)塊鏈技術(shù)的供應(yīng)鏈金融平臺

    基于區(qū)塊鏈技術(shù)的供應(yīng)鏈金融平臺

    目錄 一、項目背景 1 1.1 供應(yīng)鏈+金融場景 1 1.2 功能需求分析 2 二、方案設(shè)計 2 2.1 存儲設(shè)計 2 我們使用一個列表來保存所有的債權(quán)憑證 3 2.2 基本功能 3 企業(yè)注冊(該方法僅限部署該合約的認(rèn)證機(jī)構(gòu)調(diào)用) 4 2.3 核心功能 9 2.4 前端實現(xiàn) 13 2.5后端實現(xiàn) 16 實現(xiàn)智能合約方法調(diào)用接口

    2023年04月10日
    瀏覽(29)
  • 基于區(qū)塊鏈技術(shù)的智慧供應(yīng)鏈創(chuàng)新應(yīng)用

    隨著區(qū)塊鏈技術(shù)的不斷成熟,智慧供應(yīng)鏈得到了長足的發(fā)展,通過基于華為云的幾個智慧供應(yīng)鏈和物流行業(yè)的應(yīng)用創(chuàng)新項目,詮釋在實際應(yīng)用場景中如何通過區(qū)塊鏈和物流面臨的調(diào)整和困難,達(dá)到優(yōu)化流程、提高效率、降低成本的效果。 隨著區(qū)塊鏈技術(shù)的不斷成熟,智慧供應(yīng)

    2024年02月12日
    瀏覽(26)
  • 基于計算機(jī)視覺的物流和供應(yīng)鏈管理

    作者:禪與計算機(jī)程序設(shè)計藝術(shù) 物流、供應(yīng)鏈?zhǔn)且粋€非常重要的現(xiàn)代經(jīng)濟(jì)活動,許多企業(yè)都面臨著如何提高效率,降低成本,改善供應(yīng)鏈服務(wù)質(zhì)量的問題。目前,人們已經(jīng)在探索如何通過人工智能、物聯(lián)網(wǎng)等新興技術(shù),實現(xiàn)自動化運(yùn)輸過程和管理?;谟嬎銠C(jī)視覺技術(shù)的物流

    2024年02月10日
    瀏覽(22)
  • 供應(yīng)鏈 | 大數(shù)據(jù)報童模型:基于機(jī)器學(xué)習(xí)的實踐見解

    供應(yīng)鏈 | 大數(shù)據(jù)報童模型:基于機(jī)器學(xué)習(xí)的實踐見解

    論文解讀:李欣 馬璽淵 作者:Gah-Yi Ban, Cynthia Rudin 引用:Ban, Gah-Yi and Cynthia Rudin. The big data newsvendor: Practical insights from machine learning. Operations Research 67.1 (2019): 90-108. 文章鏈接:https://doi.org/10.1287/opre.2018.1757 文章研究了大規(guī)模數(shù)據(jù)驅(qū)動的報童問題(包括 p p p 個關(guān)于需求的特征和

    2024年02月12日
    瀏覽(19)
  • 開發(fā)安全、軟件供應(yīng)鏈安全及開源軟件安全的概念差異

    開發(fā)安全、軟件供應(yīng)鏈安全和開源軟件安全是在軟件生命周期中不同階段涉及到的安全概念,它們有著一些共同點,同時也存在一些顯著的差異。以下是它們之間的比較總結(jié): 范圍 : 開發(fā)安全 :關(guān)注于整個軟件開發(fā)過程中的安全性,包括代碼編寫、測試、部署等環(huán)節(jié)。 軟

    2024年03月14日
    瀏覽(27)
  • 基于Java實現(xiàn)的區(qū)塊鏈供應(yīng)鏈金融系統(tǒng)平臺設(shè)計

    基于Java實現(xiàn)的區(qū)塊鏈供應(yīng)鏈金融系統(tǒng)平臺設(shè)計

    中小微企業(yè)融資難、融資貴是長久以來我國金融發(fā)展過程中需要解決的問題。世界銀行、中小企業(yè)金融論壇、國際金融公司聯(lián)合發(fā)布的《中小微企業(yè)融資缺口:對新興市場微型、小型和中型企業(yè)融資不足與機(jī)遇的評估》報告中表示,中國 40%的中小微企業(yè)存在信貸困難,或是完

    2024年01月17日
    瀏覽(25)

覺得文章有用就打賞一下文章作者

支付寶掃一掃打賞

博客贊助

微信掃一掃打賞

請作者喝杯咖啡吧~博客贊助

支付寶掃一掃領(lǐng)取紅包,優(yōu)惠每天領(lǐng)

二維碼1

領(lǐng)取紅包

二維碼2

領(lǐng)紅包