Merge branch 'dev' into non-file-remote-stream

pull/9180/head
maliming 4 years ago
commit 0baccb330f

@ -0,0 +1,14 @@
{
"culture": "ro-RO",
"texts": {
"Account": "Cont ABP - Autentificare & Înregistrare | ABP.IO",
"Welcome": "Bun venit",
"UseOneOfTheFollowingLinksToContinue": "Folosiţi unul din următoarele linkuri pentru a continua",
"FrameworkHomePage": "Pagina principală a frameworkului",
"FrameworkDocumentation": "Documentaţia frameworkului",
"OfficialBlog": "Blogul oficial",
"CommercialHomePage": "Pagina principală comercială",
"CommercialSupportWebSite": "Site-ul web al suportului comercial",
"CommunityWebSite": "Site-ul web al comunităţii ABP"
}
}

@ -0,0 +1,308 @@
{
"culture": "ro-RO",
"texts": {
"Permission:Organizations": "Organizaţii",
"Permission:Manage": "Administrare Organizaţii",
"Permission:DiscountRequests": "Cereri de reduceri",
"Permission:DiscountManage": "Administrare cereri de reduceri",
"Permission:Disable": "Dezactivează",
"Permission:Enable": "Activează",
"Permission:EnableSendEmail": "Activează trimitere Email",
"Permission:SendEmail": "Trimite Email",
"Permission:NpmPackages": "Pachete NPM",
"Permission:NugetPackages": "Pachete Nuget",
"Permission:Maintenance": "Întreţinere",
"Permission:Maintain": "Întreţine",
"Permission:ClearCaches": "Goleşte cache-ul",
"Permission:Modules": "Module",
"Permission:Packages": "Pachete",
"Permission:Edit": "Modifică",
"Permission:Delete": "Şterge",
"Permission:Create": "Creează",
"Permission:Accounting": "Contabilitate",
"Permission:Accounting:Quotation": "Citat",
"Permission:Accounting:Invoice": "Factură",
"Menu:Organizations": "Organizaţii",
"Menu:Accounting": "Contabilitate",
"Menu:Packages": "Pachete",
"Menu:DiscountRequests": "Cereri de reduceri",
"NpmPackageDeletionWarningMessage": "Sunteţi sigur(ă) că doriţi să ştergeţi acest pachet de tip NPM?",
"NugetPackageDeletionWarningMessage": "Sunteţi sigur(ă) că doriţi să ştergeţi acest pachet de tip Nuget?",
"ModuleDeletionWarningMessage": "Sunteţi sigur(ă) că doriţi să ştergeţi acest modul?",
"Name": "Nume",
"DisplayName": "Nume de afişare",
"ShortDescription": "Descriere scurtă",
"NameFilter": "Nume",
"CreationTime": "Data şi ora creării",
"IsPro": "Este pro",
"IsFreeToActiveLicenseOwners": "Gratuit pentru deţinătorii de licenţă",
"ShowOnModuleList": "Afişează în lista modulelor",
"EfCoreConfigureMethodName": "Configurează numele metodei",
"IsProFilter": "Este pro",
"ApplicationType": "Tipul aplicaţiei",
"Target": "Ţinta",
"TargetFilter": "Ţinta",
"ModuleClass": "Clasa modulului",
"NugetPackageTarget.DomainShared": "Domain Shared",
"NugetPackageTarget.Domain": "Domain",
"NugetPackageTarget.Application": "Application",
"NugetPackageTarget.ApplicationContracts": "Application Contracts",
"NugetPackageTarget.HttpApi": "Http Api",
"NugetPackageTarget.HttpApiClient": "Http Api Client",
"NugetPackageTarget.Web": "Web",
"NugetPackageTarget.EntityFrameworkCore": "DeleteAllEntityFramework Core",
"NugetPackageTarget.MongoDB": "MongoDB",
"Edit": "Modifică",
"Delete": "Şterge",
"Refresh": "Reîncarcă",
"NpmPackages": "Pachete NPM",
"NugetPackages": "Pachete Nuget",
"NpmPackageCount": "Număr pachete NPM",
"NugetPackageCount": "Număr pachete Nuget",
"Module": "Module",
"ModuleInfo": "Info module",
"CreateANpmPackage": "Crează un pachet NPM",
"CreateAModule": "Crează un modul",
"CreateANugetPackage": "Crează un pachet Nuget",
"AddNew": "Adaugă o înregistrare nouă",
"PackageAlreadyExist{0}": "Pachetul \"{0}\" este deja adăugat.",
"ModuleAlreadyExist{0}": "Modulul \"{0}\" este deja adăugat.",
"ClearCache": "Goleşte cache",
"SuccessfullyCleared": "Golit cu succes",
"Menu:NpmPackages": "Pachete de tip NPM",
"Menu:Modules": "Module",
"Menu:Maintenance": "Mentenanţă",
"Menu:NugetPackages": "Pachete de tip Nuget",
"CreateAnOrganization": "Crează o organizaţie",
"Organizations": "Organizaţii",
"LongName": "Nume complet",
"LicenseType": "Tipul licenţei",
"MissingLicenseTypeField": "Câmpul de tip licenţă este obligatoriu!",
"LicenseStartTime": "Ora de început a licenţei",
"LicenseEndTime": "Ora de sfârşit a licenţei",
"AllowedDeveloperCount": "Număr permis de dezvoltatori",
"UserNameOrEmailAddress": "Numele de utilizator sau adresa de email",
"AddOwner": "Adaugă proprietar",
"UserName": "Nume de utilizator",
"Email": "Email",
"Developers": "Dezvoltatori",
"AddDeveloper": "Adaugă dezvoltator",
"Create": "Creează",
"UserNotFound": "Utilizatorul nu a fost găsit",
"{0}WillBeRemovedFromDevelopers": "{0} va fi eliminat dintre dezvoltatori, confirmaţi?",
"{0}WillBeRemovedFromOwners": "{0} va fi eliminat dintre proprietari, confirmaţi?",
"{0}WillBeRemovedFromMembers": "{0} va fi eliminat dintre membri, confirmaţi?",
"Computers": "Calculatoare",
"UniqueComputerId": "Identificator unic calculator",
"LastSeenDate": "Data ultimei accesări",
"{0}Computer{1}WillBeRemovedFromRecords": "Calculatorul {0} ({1}) va fi eliminat dintre înregistrări",
"OrganizationDeletionWarningMessage": "Organizaţia va fi ştearsă",
"DeletingLastOwnerWarningMessage": "Organizaţia trebuie să aibă cel puţin un proprietar! Prin urmare nu puteţi şterge acest proprietar.",
"This{0}AlreadyExistInThisOrganization": "Acest(a) {0} există deja în această organizaţie",
"AreYouSureYouWantToDeleteAllComputers": "Sunteţi sigur(ă) că doriţi să ştergeţi toate calculatoarele?",
"DeleteAll": "Şterge tot",
"DoYouWantToCreateNewUser": "Vreţi să creaţi un nou utilizator?",
"MasterModules": "Module master",
"OrganizationName": "Nume organizaţie",
"CreationDate": "Data creării",
"LicenseStartDate": "Data de început a licenţei",
"LicenseEndDate": "Data de sfărşit a licenţei",
"OrganizationNamePlaceholder": "Numele organizaţiei...",
"TotalQuestionCountPlaceholder": "Numărul total de înrebări...",
"RemainingQuestionCountPlaceholder": "Numărul de întrebări rămase...",
"LicenseTypePlaceholder": "Tipul licenţei...",
"CreationDatePlaceholder": "Data creării...",
"LicenseStartDatePlaceholder": "Data de început a licenţei...",
"LicenseEndDatePlaceholder": "Data de sfârşit a licenţei...",
"UsernameOrEmail": "Nume de utilizator sau email",
"UsernameOrEmailPlaceholder": "Nume de utilizator sau email...",
"Member": "Membru",
"PurchaseOrderNo": "Achiziţia numărul",
"QuotationDate": "Data citării",
"CompanyName": "Nume companie",
"CompanyAddress": "Adresă companie",
"Price": "Preţ",
"DiscountText": "Text discount",
"DiscountQuantity": "Cantitate discount",
"DiscountPrice": "Preţ discount",
"Quotation": "Citare",
"ExtraText": "Text extra",
"ExtraAmount": "Cantitate extra",
"DownloadQuotation": "Descarcă citarea",
"Invoice": "Factură",
"TaxNumber": "Număr taxă",
"InvoiceNumber": "Număr factură",
"InvoiceDate": "Dată factură",
"InvoiceNote": "Notă factură",
"Quantity": "Cantitate",
"AddProduct": "Adaugă produs",
"AddProductWarning": "Trebuie să adăugaţi un produs!",
"TotalPrice": "Preţ total",
"Generate": "Generează",
"MissingQuantityField": "Câmpul de cantitate este obligatoriu!",
"MissingPriceField": "Câmpul de preţ este obligatoriu!",
"CodeUsageStatus": "Stare",
"Country": "Ţara",
"DeveloperCount": "Număr dezvoltatori",
"RequestCode": "Cod solicitare",
"WebSite": "Web Site",
"GithubUsername": "Nume utilizator de Github",
"PhoneNumber": "Număr de telefon",
"ProjectDescription": "Descriere proiect",
"Referrer": "Referent",
"DiscountRequests": "Cerere de reducere",
"Copylink": "Copiază linkul",
"Disable": "Dezactivează",
"Enable": "Activează",
"EnableSendEmail": "Activează trimitere Email",
"SendEmail": "Trimite Email",
"SuccessfullyDisabled": "Dezactivat cu succes",
"SuccessfullyEnabled": "Activat cu succes",
"EmailSent": "Email trimis",
"SuccessfullySent": "Trimis cu succes",
"SuccessfullyDeleted": "Şters cu succes",
"DiscountRequestDeletionWarningMessage": "Cererea de reducere va fi ştearsă",
"BusinessType": "Tip afacere",
"TotalQuestionCount": "Număr total de întrebări",
"RemainingQuestionCount": "Număr de întrebări rămase",
"TotalQuestionMustBeGreaterWarningMessage": "Numărul total de întrebări trebuie să fie mai mare decât numărul de întrebări rămase!",
"QuestionCountsMustBeGreaterThanZero": "Numărul total de întrebări şi numărul de întrebări rămase trebuie să fie mai mai mare sau egal cu 0!",
"UnlimitedQuestionCount": "Număr nelimitat de întrebări",
"Notes": "Notiţe",
"Menu:Community": "Comunitate",
"Menu:Articles": "Articole",
"Wait": "Aşteaptă",
"Approve": "Aprobă",
"Reject": "Respinge",
"Details": "Detalii",
"Url": "Url",
"Title": "Titlu",
"ContentSource": "Sursa conţinutului",
"Status": "Status",
"ReadArticle": "Citeşte articolul",
"ArticleHasBeenWaiting": "Articolul a fost în aşteptare",
"ArticleHasBeenApproved": "Articolul a fost aprobat",
"ArticleHasBeenRejected": "Articolul a fost respins",
"Permission:Community": "Comunitate",
"Permission:CommunityArticle": "Articole",
"Link": "Link",
"Enum:ContentSource:0": "Github",
"Enum:ContentSource:1": "External",
"Enum:Status:0": "Aşteptare",
"Enum:Status:1": "Respins",
"Enum:Status:2": "Aprobat",
"Summary": "Sumar",
"AuthorName": "Nume autor",
"CoverImage": "Imagine de copertă",
"RemoveCacheConfirmationMessage": "Sunteţi sigur(ă) că doriţi să eliminaţi cache-ul pentru articolul \"{0}\"?",
"SuccessfullyRemoved": "Eliminat cu succes",
"RemoveCache": "Elimină cache",
"Language": "Limba",
"Optional": "Opţional",
"CreateArticleLanguageInfo": "Limba în care este scris articolul",
"Enum:ContentSource:2": "Post Video",
"VideoPreview": "Previzualizare Video",
"VideoPreviewErrorMessage": "Adresa URL a clipului video nu a putut fi recuperată de pe Youtube. Acest lucru poate fi cauzat de faptul că videoclipul este privat, sau adresa URL nu este disponibilă.",
"DeleteCoverImage": "Şterge imaginea de copertă",
"DeleteCoverImageConfirmationMessage": "Sunteţi sigur(ă) că doriţi să ştergeti imaginea de copertă pentru \"{0}\"?",
"DeleteCoverImageSuccessMessage": "Imaginea de copertă a fost ştearsă cu succes",
"PaymentsOf": "Plăţi de",
"ShowPaymentsOfOrganization": "Plăţi",
"Date": "Data",
"Products": "Produse",
"TotalAmount": "Cantitate totală",
"Currency": "Valută",
"Gateway": "Gateway",
"State": "Stare",
"FailReason": "Motivul eşuării",
"ReIndexAllPosts": "Reindexati toate postările",
"ReIndexAllPostsConfirmationMessage": "Sunteţi sigur(ă) că doriţi să reindexaţi toate postările?",
"SuccessfullyReIndexAllPosts": "Toate postările au fost reindexate.",
"Permission:FullSearch": "Căutare integrală a textului",
"Menu:CliAnalytics": "Analitice Cli",
"TemplateName": "Nume şablon",
"TemplateVersion": "Versiune şablon",
"DatabaseProvider": "Furnizor bază de date",
"IsTiered": "Este nivelat",
"ProjectName": "Nume proiect",
"Username": "Nume utilizator",
"Tool": "Unealtă",
"Command": "Comandă",
"UiFramework": "Ui framework",
"Options": "Opţiuni",
"CliAnalytics": "Analitice Cli",
"Permission:CliAnalyticses": "Analitice Cli",
"Permission:CliAnalytics": "Analitice Cli",
"Search": "Caută",
"ClearFilter": "Şterge filtru",
"LicensePrivateKey": "Cheie privată licenţă",
"LicensePublicKey": "Cheie publică licenţă",
"ApiKey": "Cheie API NuGet",
"ShowInvoiceRequestsOfOrganization": "Solicitări de factură",
"ShowQuestionsOfOrganization": "Întrebări",
"Question": "Întrebare",
"Open": "Deschide",
"Questions": "Întrebări",
"InvoiceRequests": "Solicitări de factură",
"Address": "Adresa",
"TaxNo": "Taxa numărul",
"Permission:InvoiceRequest": "Solicitare de factură",
"Permission:Question": "Întrebare",
"AddNoteSuccessMessage": "Notă adăugată cu succes",
"NameSurname": "Nume Prenume",
"Note": "Notează",
"Add": "Adaugă",
"ProjectDownloads": "Descărcările proiectului",
"ShowProjectDownloadsOfOrganization": "Descărcările proiectului",
"ShowAuditLogsOfOrganization": "Jurnale de audit",
"Enum:EntityChangeType:0": "Creat",
"Enum:EntityChangeType:1": "Actualizat",
"Enum:EntityChangeType:2": "Şters",
"TenantId": "ID Tenant",
"ChangeTime": "Schimbă ora",
"EntityTypeFullName": "Numele complet al tipului entităţii",
"AuditLogsFor{0}Organization": "Jurnale de audit pentru organizaţia \"{0}\"",
"Permission:EntityChange": "Schimbă entitatea",
"Permission:ProjectDownload": "Descarcă proiectul",
"Permission:PaymentRequest": "Solicitări de plată",
"CreatorEmailAddress": "Adresa de email a creatorului",
"EmailSendDate": "Data trimiterii emailului",
"PaymentRequestsFor{0}Organization": "Solicitări de plată pentru organizaţia \"{0}\"",
"PaymentDetails": "Detalii plată",
"PaymentProduct": "Produs plată",
"ProductName": "Nume produs",
"Code": "Cod",
"GenerateInvoice": "Generează factură",
"ExportOrganizationsToExcel": "Exportă în Excel",
"ThisExtensionIsNotAllowed": "Această extensie nu este permisă.",
"TheFileIsTooLarge": "Dimensiunea fişierului este prea mare!",
"ArticleDeletionConfirmationMessage": "Sunteţi sigur(ă) că doriţi şă ştergeţi definitiv acest articol?",
"ChooseCoverImage": "Alege o imagine de copertă...",
"Menu:Quotation": "Citare",
"Menu:Invoice": "Factură",
"Menu:PaymentRequests": "Solicitări plată",
"Permission:PaymentRequests": "Solicitări plată",
"PaymentRequests": "Solicitări plată",
"Creator": "Creator",
"ExtraProperties": "Proprietăţi extra",
"Organization": "Organizaţia",
"Waiting": "În aşteptare",
"Completed": "Finalizat",
"Failed": "Eşuat",
"PaymentRequestDeletionWarningMessage": "Sunteţi sigur(ă) că doriţi să ştergeţi această plată?",
"Payment": "Plata",
"AddPayment": "Adaugă plată",
"Enum:PurchaseType:1": "Licenţă nouă",
"Enum:PurchaseType:2": "Extindere licenţă",
"Enum:PurchaseType:3": "Îmbunătăţire licenţă",
"Enum:PurchaseType:4": "Dezvoltator suplimentar",
"LicenceType": "Tipul licenţei",
"PurchaseType": "Tipul achiziţiei",
"ReceiptNo": "Chitanţa numărul",
"PaymentTime": "Data plăţii",
"ProductPrice": "Preţul produsului",
"AdditionalDeveloper": "Dezvoltator suplimentar",
"ThisPaymentHasBeenAlreadyUsed": "Această plată a fost deja folosită",
"PaymentTimeCannotBeFutureTime": "Data plăţii nu poate fi în viitor"
}
}

@ -0,0 +1,40 @@
{
"culture": "ro-RO",
"texts": {
"Volo.AbpIo.Domain:010004": "Numărul maxim de membri a fost atins!",
"Volo.AbpIo.Domain:010005": "Numărul maxim de proprietari a fost atins!",
"Volo.AbpIo.Domain:010006": "Există deja un proprietar în această organizaţie!",
"Volo.AbpIo.Domain:010007": "Acest utilizator este deja dezvoltator în această organizaţie!",
"Volo.AbpIo.Domain:010008": "Numărul permis de dezvoltatori nu poate fi mai mic decât numărul curent de dezvoltatori!",
"Volo.AbpIo.Domain:010009": "Numărul permis de dezvoltatori nu poate fi mai mic decât 0!",
"Volo.AbpIo.Domain:010010": "Numărul maxim de adrese de tip mac este depăşit!",
"Volo.AbpIo.Domain:010011": "Licenţa personală nu poate avea mai mult de un dezvoltator!",
"Volo.AbpIo.Domain:010012": "Licenţa nu poate fi prelungită la o lună după ce aceasta a expirat!",
"Volo.AbpIo.Domain:020001": "Nu s-a putut şterge acest pachet NPM deoarece \"{NugetPackages}\" există pachete Nuget dependente de acest pachet.",
"Volo.AbpIo.Domain:020002": "Nu s-a putut şterge acest pachet NPM deoarece \"{Modules}\" modulele folosesc acest pachet.",
"Volo.AbpIo.Domain:020003": "Nu s-a putut şterge acest pachet NPM deoarece \"{Modules}\" modulele folosesc acest pachet \"{NugetPackages}\" şi există pachete Nuget dependente de acest pachet.",
"Volo.AbpIo.Domain:020004": "Nu s-a putut şterge acest pachet Nuget deoarece \"{Modules}\" modulele folosesc acest pachet.",
"WantToLearn?": "Vrei să înveţi?",
"ReadyToGetStarted?": "Eşti gata pentru a începe?",
"JoinOurCommunity": "Alăturaţi-vă comunităţii noastre",
"GetStartedUpper": "ÎNCEPE",
"ForkMeOnGitHub": "Fork me on GitHub",
"Features": "Caracteristici",
"GetStarted": "Începe",
"Documents": "Documente",
"Community": "Comunitate",
"ContributionGuide": "Ghid de contribuţie",
"Blog": "Blog",
"Commercial": "Comercial",
"MyAccount": "Contul meu",
"Permission:License": "Licenţă",
"Permission:UserInfo": "Info utilizator",
"SeeDocuments": "Vezi documente",
"Samples": "Mostre",
"Framework": "Framework",
"Support": "Suport",
"FreeDDDBook": "DDD E-book gratis",
"New": "Nou",
"Volo.AbpIo.Domain:020005": "Anul prelungirii licenţei nu poate fi mai mic decât {MinExtendLicenseYear} şi mai mare decât {MaxExtendLicenseYear}"
}
}

@ -0,0 +1,363 @@
{
"culture": "ro-RO",
"texts": {
"OrganizationManagement": "Administrarea organizaţiilor",
"OrganizationList": "Lista organizaţiilor",
"Volo.AbpIo.Commercial:010003": "Nu sunteţi proprietarul acestei organizaţii!",
"OrganizationNotFoundMessage": "Nicio organizaţie găsită!",
"DeveloperCount": "Alocaţi / total dezvoltatori",
"QuestionCount": "Rămase / total întrebări",
"Unlimited": "Nelimitat",
"Owners": "Proprietari",
"AddMember": "Adaugă membru",
"AddOwner": "Adaugă proprietar",
"AddDeveloper": "Adaugă dezvoltator",
"UserName": "Nume utilizator",
"Name": "Nume",
"EmailAddress": "Adresă de email",
"Developers": "Dezvoltatori",
"LicenseType": "Tip licenţă",
"Manage": "Administrează",
"StartDate": "Data de început",
"EndDate": "Data de sfârşit",
"Modules": "Module",
"LicenseExtendMessage": "Data expirării licenţei dumneavoastră a fost prelungită până în {0}",
"LicenseUpgradeMessage": "Licenţa dumneavoastră a fost actualizată la {0}",
"LicenseAddDeveloperMessage": "{0} dezvoltatori au fost ataşaţi licenţei dumneavoastră",
"Volo.AbpIo.Commercial:010004": "Utilizatorul nu a putut fi găsit! Acesta trebuie să se fi înregistrat deja.",
"MyOrganizations": "Organizaţiile mele",
"ApiKey": "Cheie API",
"UserNameNotFound": "Nu există niciun utilizator cu numele {0}",
"SuccessfullyAddedToNewsletter": "Vă mulţumim pentru că v-aţi abonat la newsletter-ul nostru!",
"MyProfile": "Profilul meu",
"EmailNotValid": "Vă rugăm să introduceţi o adresa de email validă.",
"JoinOurMarketingNewsletter": "Alăturaţi-va newsletter-ului nostru de marketing",
"WouldLikeToReceiveMarketingMaterials": "Aş dori să primesc materiale de marketing precum oferte şi oferte speciale.",
"StartUsingYourLicenseNow": "Începe folosirea licenţei tale acum!",
"WelcomePage": "Pagina de bun venit",
"UnsubscriptionExpireEmail": "Dezabonează-te de la emailuri informative despre data expirării licenţei.",
"UnsubscribeLicenseExpireEmailReminderMessage": "Această abonament prin email conţine numai amintirea datei de expirare a licenţei.",
"UnsubscribeFromLicenseExpireEmails": "Dacă nu mai doriţi să primiţi emailuri informative despre data de expirare a licenţei dumneavoastră, puteţi să vă dezabonaţi oricând.",
"Unsubscribe": "Dezabonează-te",
"NotOrganizationMember": "Nu sunteţi membru al niciunei organizaţii.",
"UnsubscribeLicenseExpirationEmailSuccessTitle": "V-aţi dezabonat cu succes.",
"UnsubscribeLicenseExpirationEmailSuccessMessage": "Nu veţi mai primi emailuri informative despre data de expirare a licenţei dumneavoastră.",
"IndexPageHeroSection": "<span class=\"first-line\">A complete</span><span class=\"second-line\"> web development platform</span><span class=\"third-line\">built-on <img src=\"{0}\" width=\"110\" class=\"ml-1\" /> framework</span>",
"AbpCommercialShortDescription": "ABP Commercial oferă module de aplicaţii pre-construite, instrumente de dezvoltare rapidă a aplicaţiilor, teme UI profesionale, asistenţă premium şi multe altele.",
"LiveDemo": "Live Demo",
"GetLicence" :"Obţine o licenţă",
"Application": "Application",
"StartupTemplates": "Şabloane de lansare",
"Startup": "Lansare",
"Templates": "Şabloane",
"Developer": "Dezvoltator",
"Tools": "Unelte",
"Premium": "Premium",
"PremiumSupport": "Asistenţă Premium",
"PremiumForumSupport": "Forum asistenţă premium",
"UI": "UI",
"Themes": "Teme",
"JoinOurNewsletter": "Abonaţi-va newsletter-ului nostru",
"Send": "Trimite",
"Learn": "Învaţă",
"AdditionalServices": "Servicii suplimentare",
"WhatIsABPFramework": "CE ESTE FRAMEWORK-UL ABP?",
"OpenSourceBaseFramework": "Open Source Base Framework",
"ABPFrameworkExplanation": "<p class=\"lead\">ABP Commercial se bazează pe framework-ul ABP, un framework open source şi bazat pe comunitate pentru ASP.NET Core.</p><p>ABP Framework oferă o infrastructură excelentă pentru a scrie cod care poate fi întreţinut, extins şi testat cu cele mai bune practici.</p><p>Construit şi integrat în instrumentele populare pe care le cunoaşteţi deja. Curbă de învăţare redusă, adaptare uşoară, dezvoltare confortabilă.</p>",
"Modular": "Modular",
"MicroserviceCompatible": "Compatibil cu microservicii",
"DomainDrivenDesignInfrastructure": "Domain Driven Design Infrastructure",
"MultiTenancy": "Multi-Tenancy",
"DistributedMessaging": "Mesaje distribuite",
"DynamicProxying": "Proxy dinamic",
"BackgroundJobs": "Background Jobs",
"AuditLogging": "Înregistrarea auditului",
"BLOBStoring": "BLOB Storing",
"BundlingMinification": "Bundling & Minification",
"AdvancedLocalization": "Localizare avansată",
"ManyMore": "Multe altele",
"ExploreTheABPFramework": "Exploraţi framework-ul ABP",
"WhyUseTheABPCommercial": "De ce să folosiţi ABP Commercial?",
"WhyUseTheABPCommercialExplanation": "<p class=\"lead mt- 5\">Construirea de aplicaţii web de nivel enterprise poate fi complexă şi poate consuma mult timp.</p><p>ABP Commercial oferă infrastructura de bază pentru toate soluţiile moderne bazate pe ASP.NET Core de nivel enterprise. De la proiectare până la implementare, întregul ciclu de dezvoltare este împuternicit de funcţiile şi modulele integrate ale ABP.</p>",
"StartupTemplatesShortDescription": "Şabloanele de pornire vă fac să începeţi proiectul în câteva secunde.",
"UIFrameworksOptions": "UI frameworks options;",
"DatabaseProviderOptions": "Opţiuni furnizor de bază de date;",
"PreBuiltApplicationModules": "Module de aplicaţii pre-construite",
"PreBuiltApplicationModulesShortDescription": "Cele mai comune cerinţe de aplicaţie sunt deja dezvoltate pentru dumneavoastră sub formă de module reutilizabile.",
"Account": "Cont",
"Blogging": "Blogging",
"Identity": "Identitate",
"IdentityServer": "Identity Server",
"Saas": "Saas",
"LanguageManagement": "Administrarea limbilor",
"TextTemplateManagement": "Text Template Management",
"SeeAllModules": "Vezi toate modulele",
"ABPSuite": "ABP Suite",
"AbpSuiteShortDescription": "ABP Suite este un instrument complementar pentru ABP Commercial.",
"AbpSuiteExplanation": "Vă permite să creaţi pagini web în câteva minute. Este un instrument .NET Core global, care poate fi instalat din linia de comandă. Poate crea o nouă soluţie ABP, poate genera pagini CRUD, începând de la baza de date până la front-end.",
"Details": "Detalii",
"LeptonTheme": "Tema Lepton",
"ProfessionalModernUIThemes": "Teme UI profesionale, moderne",
"LeptonThemeExplanation": "Lepton oferă o gamă de teme de administrare Bootstrap, care servesc ca o bază solidă pentru orice proiect care necesită tabloul de bord de administrare.",
"DefaultTheme": "Tema Default",
"MaterialTheme": "Tema Material",
"Default2Theme": "Tema Default 2",
"DarkTheme": "Tema Dark",
"DarkBlueTheme": "Tema Dark Blue",
"LightTheme": "Tema Light",
"ProudToWorkWith": "Mândru să lucrez cu",
"OurConsumers": "Sute de întreprinderi şi dezvoltatori din peste 50 de ţări din jurul lumii se bazează pe ABP Commercial.",
"JoinOurConsumers": "Alăturaţi-va lor şi construiţi produse uimitoare rapid.",
"AdditionalServicesExplanation": "Aveţi nevoie de servicii suplimentare sau personalizate? <strong>Noi şi partenerii noştri vi le putem oferi;</strong>",
"CustomProjectDevelopment": "Custom Project Development",
"CustomProjectDevelopmentExplanation": "Dezvoltatori dedicaţi pentru proiectele dumneavoastră personalizate.",
"PortingExistingProjects": "Portarea proiectelor existente",
"PortingExistingProjectsExplanation": "Migrarea proiectelor dumneavoastră vechi pe platforma ABP.",
"LiveSupport": "Asistenţă live",
"LiveSupportExplanation": "Opţiune de asistenţă live la distanţă oricând aveţi nevoie.",
"Training": "Instruire",
"TrainingExplanation": "Instruire dedicată pentru dezvoltatorii dumneavoastră.",
"OnBoarding": "Integrare",
"OnBoardingExplanation": "Ajutor pentru configurarea dezvoltării, mediilor CI & CD.",
"PrioritizedTechnicalSupport": "Asistenţă tehnică prioritară",
"PremiumSupportExplanation": "Pe lângă excelentul suport comunitar al framework-ului ABP, echipa noastră de asistenţă răspunde la întrebările tehnice şi la problemele utilizatorilor comerciali cu prioritate crescută.",
"SeeTheSupportOptions": "Vedeţi opţiunile de asistenţă",
"Contact": "Contact",
"TellUsWhatYouNeed": "Spuneţi-ne de ce aveţi nevoie.",
"YourMessage": "Mesajul dumneavoastră",
"YourFullName": "Numele dumneavoastră complet",
"EmailField": "Adresa de email",
"YourEmailAddress": "Adresa dumneavoastră de email",
"HowMayWeHelpYou": "Cum vă putem ajuta?",
"SendMessage": "Trimite mesaj",
"Success": "Succes",
"WeWillReplyYou": "Am primit mesajul dumneavoastră şi vă vom contacta cât de repede posibil.",
"GoHome": "Navigaţi acasă",
"CreateLiveDemo": "Crează Demo Live",
"RegisterToTheNewsletter": "Înregistraţi-va la newsletter pentru a primi informaţii despre ABP.IO, inclusiv lansări noi etc.",
"EnterYourEmailOrLogin": "Introduceţi-vă adresa de email pentru a vă crea demo-ul sau <a href=\"{0}\">Autentificaţi-vă</a> folosind contul existent.",
"ApplicationTemplate": "Application Template",
"ApplicationTemplateExplanation": "Şablonul de pornire a aplicaţiei este utilizat pentru a crea o nouă aplicaţie web.",
"EfCoreProvider": "Entity Framework (susţine SQL Server, MySQL, PostgreSQL, Oracle <a href=\"https://docs.microsoft.com/en-us/ef/core/providers/\">şi altele</a>)",
"AlreadyIncludedInTemplateModules": "Următoarele module sunt deja incluse şi configurate în acest şablon:",
"ApplicationTemplateArchitecture": "Acest şablon de aplicaţie susţine arhitecturi structurate pe nivele, în care nivelul UI, nivelul API şi serviciul de autentificare sunt separate fizic.",
"SeeTheGuideOrGoToTheLiveDemo": "Consultaţi ghidul dezvoltatorului pentru informaţii tehnice despre acest şablon sau mergeţi la demo-ul live.",
"DeveloperGuide": "Ghidul dezvoltatorului",
"ModuleTemplate": "Module Template",
"ModuleTemplateExplanation1": "Vreţi să creaţi un modul pe care sa-l reutilizaţi în cadrul diferitelor aplicaţii? Acest şablon de pornire pregăteşte totul pentru a putea începe să creaţi un <strong>modul</strong> sau un <strong>microserviciu</strong> reutilizabil.",
"ModuleTemplateExplanation2": "<p>Puteţi susţine cadre UI simple sau multiple, furnizori unici sau multipli pentru baza de date pentru un singur modul. Şablonul de pornire este configurat pentru a rula <strong>şi testa modulul dumneavoastră</strong> într-o aplicaţie minimă, în plus faţă de infrastructura de testare şi integrare.</p> <p>Consultaţi ghidul dezvoltatorului pentru informaţii tehnice despre acest şablon.</p>",
"WithAllStyleOptions": "cu toate opţiunile de stil",
"Demo": "Demo",
"SeeAllModules": "Vezi toate modulele",
"ABPCLIExplanation": "ABP CLI (Command Line Interface) este un instrument de linii de comandă pentru executarea unor operaţii comune pentru soluţiile ABP.",
"ABPSuiteEasilyCURD": "Suita ABP este un instrument care vă permite crearea cu uşurinţă a paginilor CRUD",
"WeAreHereToHelp": "Suntem aici să <span class=\"zero-text\">Ajutăm</span>",
"BrowseOrAskQuestion": "Puteţi răsfoi subiectele noastre de ajutor sau puteţi căuta în cadrul secţiunii întrebărilor frecvent adresate, sau ne puteţi adresa o întrebare folosind <a href=\"{0}\" class=\"text-success\">formularul de contact</a>.",
"SearchQuestionPlaceholder": "Caută în întrebările frecvent adresate",
"WhatIsTheABPCommercial": "Ce este ABP Commercial?",
"WhatAreDifferencesThanAbpFramework": "Care sunt diferenţele dintre ABP Framework şi ABP Comercial?",
"ABPCommercialExplanation": "ABP Commercial este un set de module premium, instrumente, teme şi servicii construite pe baza <a target=\"_blank\" href=\"{0}\">ABP framework</a>. ABP Commercial este dezvoltat şi întreţinut de către aceeaşi echipă din spatele ABP framework.",
"WhatAreDifferencesThanABPFrameworkExplanation": "<p> <a target=\"_blank\" href=\"{0}\">ABP framework</a> este un framework de dezvoltare de aplicaţii modulare, compatibil cu micro-servicii dedicate ASP.NET Core. Oferă o arhitectură completă şi o infrastructură puternică pentru a vă face să vă concentraţi asupra propriului cod, mai degrabă decât să vă repetaţi pentru fiecare proiect nou. Se bazează pe cele mai bune practici de dezvoltare software, şi pe instrumentele populare pe care le cunoaşteţi deja. </p><p>ABP Framework este complet gratuit, open-source şi bazat pe comunitate. De asemenea, oferă şi o temă gratuită şi câteva module pre-construite (de exemplu, gestionarea identităţii şi a chiriaşilor).</p>",
"VisitTheFrameworkVSCommercialDocument": "Accesaţi următorul link, pentru mai multe informaţii <a href=\"{0}\" target=\"_blank\"> {1} </a>",
"ABPCommercialFollowingBenefits": "ABP Commercial adaugă următoarele beneficii peste framework-ul ABP;",
"Professional": "Profesional",
"UIThemes": "Teme UI",
"EnterpriseModules": "<a href=\"{0}\">Module de aplicaţii</a> pregătite pentru întreprinderi, bogate în funcţionalităţi, pre-construite (e.g. Identity Server management, SaaS management, language management)",
"ToolingToSupport": "Instrumente pentru a vă susţine productivitatea de dezvoltare (e.g. <a href=\"{0}\">Suita ABP</a>)",
"PremiumSupportLink": "<a href=\"{0}\" target=\"_blank\">Asistenţă</a> premium",
"WhatDoIDownloadABPCommercial": "Ce descarc când cumpăr ABP Commercial?",
"CreateUnlimitedSolutions": "Odată ce aţi cumpărat o licenţă ABP Commercial, veţi putea să creaţi soluţii nelimitate, descrise în documentul <a href=\"{0}\">Noţiuni de bază</a>.",
"ABPCommercialSolutionExplanation": "Când creaţi o aplicaţie nouă, veţi primi o soluţie de Visual Studio (un şablon de pornire) bazat pe preferinţele dumneavoastră. Soluţia descărcată conţine module comerciale şi teme pre-instalate şi configurate pentru dumneavoastră. Puteţi şterge un modul pre-instalat sau să adăugaţi altul dacă doriţi. Toate modulele şi temele folosesc un pachet NuGet/NPM în mod implicit.",
"StartDevelopWithTutorials": "Soluţia descărcată este bine structurată şi documentată. Puteţi începe să vă dezvoltaţi codul dumneavoastră pe baza unuia dintre <a href=\"{0}\">tutoriale</a>",
"TryTheCommercialDemo": "Puteţi încerca versiunea <a href=\"{0}\">demo</a> pentru a vedea un exemplu de aplicaţie creată folosind şablonul de pornire ABP Commercial.",
"HowManyProducts": "Câte produse/soluţii diferite pot construi folosind ABP Commercial?",
"HowManyProductsExplanation": "Nu există nicio limită pentru crearea unui proiect ABP. Puteţi crea cât de multe proiecte doriţi, pe care să le dezvoltaţi sau încărcaţi pe diferite servere.",
"HowManyDevelopers": "Câţi dezvoltatori pot lucra pe ABP Commercial?",
"HowManyDevelopersExplanation": "Licenţele de ABP Commercial sunt per dezvoltator. Tipuri diferite de licenţă au limite diferite de dezvoltatori. Totuşi, puteţi ataşa mai mulţi dezvoltatori unei licenţe oricând doriţi. Consultaţi <a href=\"{0}\">preţurile</a> pentru tipurile de licenţă, limitele de dezvoltatori şi costurile adiţionale pentru dezvoltatori.",
"ChangingLicenseType": "Pe viitor pot să schimb tipul licenţei?",
"ChangingLicenseTypeExplanation": "Puteţi adăuga oricând noi dezvoltatori în licenţa dumneavoastră de acelaşi tip. Consultaţi \"Câţi dezvoltatori pot lucra pe ABP Commercial?\". De asemenea, puteţi opta pentru o licenţă superioară prin plătirea achitarea de preţ. Atunci când optaţi pentru o licenţă superioară, veţi avea parte de beneficiile noii licenţe, însă acest lucru nu va schimba data expirării licenţei.",
"LicenseExtendUpgradeDiff": "Care este diferenţa dintre prelungirea licenţei şi îmbunătăţirea acesteia?",
"LicenseExtendUpgradeDiffExplanation": "<strong>Prelungirea:</strong> Prin prelungirea/reînnoirea licenţei dumneavoastră, veţi continua să aveţi acces la asistenţă premium, şi veţi avea parte de actualizările majore ale modulelor şi temelor. Pe lângă asta, veţi putea în continuare să creaţi proiecte noi. Veţi putea folosi în continuare suita ABP, ceea ce va accelera dezvoltarea.<hr/><strong>Îmbunătăţirea:</strong> Prin îmbunătăţirea licenţei dumneavoastră, veţi promova la un plan de licenţă superior, care vă va permite să beneficiaţi de anumite avantaje. Consultaţi <a href=\"/pricing\">tabelul de comparare a licenţelor</a> pentru a verifica diferenţele dintre planurile de licenţă.<strong>Pe de altă parte, când îmbunătăţiţi licenţa, data de expirare a acesteia nu se va schimba!</strong>Pentru a vă extinde data de încheiere a licenţei, va trebui să vă extindeţi licenţa.",
"LicenseRenewalCost": "Care este costul de reînnoire a licenţei dupa 1 an?",
"LicenseRenewalCostExplanation": "Rata de reînnoire (extindere) a tuturor licenţelor ABP Comercial reprezintă {0} din preţul listei licenţelor. Preţul de reînnoire a licenţei standard de echipă este de ${1}, cel pentru licenţa standard de afaceri este de ${2} iar cel pentru licenţa standard enterprise este de ${3}. Dacă deja sunteţi clientul nostru, <a href='{4}' target='_blank'>autentificaţi-vă</a> pentru a examina preţurile disponibile pentru reînnoire.",
"HowDoIRenewMyLicense": "Cum îmi reînnoiesc licenţa?",
"HowDoIRenewMyLicenseExplanation": "Puteţi să vă reînnoiţi licenţa prin accesarea paginii <a href='{0}' target='_blank'>pagina de gestionare a organizaţiei</a>. Pentru a profita de tarifele noastre reduse de reînnoire anticipată, asiguraţi-vă că reînnoiţi înainte de expirarea licenţei. Totuşi, nu vă faceţi griji dacă nu ştiţi când expiră oportunitatea dumneavoastră de reînnoire anticipată. Veţi primi 2 e-mailuri de reamintire înainte de expirarea abonamentului. Vi le vom trimite cu 30, respectiv 7 zile înainte de expirare.",
"IsSourceCodeIncluded": "Licenţa mea include şi codul sursă al modulelor comerciale şi a temelor?",
"IsSourceCodeIncludedExplanation1": "Depinde de tipul de licenţă pe care aţi achiziţionat-o:",
"IsSourceCodeIncludedExplanation2": "<strong>Team</strong>: Soluţia dumneavoastră foloseste modulele şi temele sub formă de pachete NuGet şi NPM. Nu include şi codul lor sursă. În acest mod, puteţi cu uşurinţă să îmbunătăţiţi aceste module şi teme oricând este o nouă versiune disponibilă. Totuşi, nu veţi primi codul sursă al modulelor şi temelor.",
"IsSourceCodeIncludedExplanation3": "<strong>Business/Enterprise</strong>: Pe lângă licenţa de tip Team, veţi putea să descărcaţi codul sursă al oricărui modul sau al oricărei teme doriţi. Puteţi chiar să eliminaţi referinţele pachetelor NuGet/NPM pentru un anume modul, şi să adăugaţi codul sursă al acestuia direct în soluţia dumneavoastră pentru a-l putea modifica pe deplin.",
"IsSourceCodeIncludedExplanation4": "<p>Includerea codului sursă al unui modul în soluţia dumneavoastră vă oferă libertate maximă de a personaliza modulul respectiv. Totuşi, atunci nu va fi posibilă actualizarea automată a modulului atunci când o nouă versiune este publicată.</p><p>Niciuna dintre licenţe nu include codul sursă al suitei ABP, care este un instrument extern care generează cod pentru dumneavoastră şi contribuie la dezvoltare.</p><p>Consultaţi <a href=\"{0}\">preţurile</a> pentru alte diferenţe între tipurile de licenţe.</p>",
"ChangingDevelopers": "Pe viitor, pot schimba dezvoltatorii ataşaţi organizaţiei mele?",
"ChangingDevelopersExplanation": "În plus faţă de adăugarea de noi dezvoltatori la licenţa dumneavoastră, puteţi schimba şi dezvoltatorii existenţi(puteţi elimina un dezvoltator şi adăuga unul nou în locul lui) fără costuri suplimentare.",
"WhatHappensWhenLicenseEnds": "Ce se întâmplă când se încheie perioada licenţei mele?",
"WhatHappensWhenLicenseEndsExplanation1": "Tipul de licenţă ABP Comercial este o <a href=\"{0}\" target=\"_blank\">licenţă perpetuă</a>. După ce vă expiră licenţa, puteţi continua dezvoltarea proiectului dumneavoastră. Şi nu sunteţi obligat să vă reînnoiţi licenţa. Licenţa dumneavoastră vine la pachet cu actualizări şi plan de asistenţă valabile timp de un an. Pentru a continua să beneficiaţi de noi caracteristici, îmbunătăţiri de performanţe, corectarea erorilor, asistenţă şi să continuaţi să utilizaţi suita ABP, asiguraţi-vă că vă reînnoiţi licenţa anual. Când vă expiră licenţa, nu veţi mai beneficia de niciunul din beneficiile următoare;",
"WhatHappensWhenLicenseEndsExplanation2": "Nu puteţi crea noi soluţii folosind ABP Commercial, dar puteţi continua dezvoltarea aplicaţiilor existente oricând.",
"WhatHappensWhenLicenseEndsExplanation3": "You will be able to get updates for the modules and themes within your MAJOR version. For example; if you are using v3.2.0 of a module, you can still get updates for v3.x.x (v3.3.0, v3.5.2... etc.) of that module. But you cannot get updates for the next major version (like v4.x, v5.x)",
"WhatHappensWhenLicenseEndsExplanation4": "Nu puteţi instala noi module şi teme adăugate pe platforma ABP Comercial după ce se încheie licenţa dumneavoastră.",
"WhatHappensWhenLicenseEndsExplanation5": "Nu puteţi folosi suita ABP.",
"WhatHappensWhenLicenseEndsExplanation6": "Nu veţi mai avea parte de <a href=\"{0}\">asistenţă premium</a>.",
"WhatHappensWhenLicenseEndsExplanation7": "Puteţi să vă reînnoiţi abonamentul dacă vreţi să continuaţi să aveţi parte de aceste beneficii. Dacă vă reînnoiţi abonamentul în termen de o lună după ce vă expiră licenţa, vi se aplică un discount de 20%.",
"WhenShouldIRenewMyLicense": "Când ar trebui să-mi reînnoiesc licenţa?",
"WhenShouldIRenewMyLicenseExplanation1": "Dacă vă reînnoiţi licenţa în termen de <strong>1 lună</strong> după ce vă expiră licenţa, se va aplica un discount de 20%.",
"WhenShouldIRenewMyLicenseExplanation2": "Dacă vă reînnoiţi licenţa la o lună după ce aceasta a expirat, preţul de reînnoire va fi acelaşi cu cel iniţial, şi nu va fi aplicat niciun discount pentru reînnoirea dumneavoastră.",
"TrialPlan": "Aveţi un plan de încercare?",
"TrialPlanExplanation": "Pentru moment, ABP Comercial nu include un plan de încercare. Pentru licenţa de Team oferim înapoirea banilor în termen de 30 de zile. Puteţi să solicitaţi o restituire în primele 30 de zile. Pentru licenţele de tip Business şi Enterprise, noi asigurăm restituirea a 60% din sumă în 30 de zile. Acest lucru se datorează faptului că licenţele de tip Business şi Enterprise includ codul sursă complet al tuturor modulelor şi temelor.",
"DoYouAcceptBankWireTransfer": "Acceptaţi transfer bancar?",
"DoYouAcceptBankWireTransferExplanation": "Da, acceptăm transfer bancar.<br />După trimiterea taxei de licenţă prin transfer bancar, trimiteţi-ne pe e-mail la info@abp.io chitanţa dumneavoastră şi tipul de licenţă solicitat. Informaţiile despre contul bancar internaţional al nostru:",
"HowToUpgrade": "Cum se actualizează aplicaţiile existente atunci când este disponibilă o nouă versiune?",
"HowToUpgradeExplanation1": "Cand creaţi o nouă aplicaţie folosind ABP Commercial, toate modulele şi tema sunt folosite sub formă de pachete NuGet şi NPM. Astfel, puteţi cu uşurinţă să actualizaţi pachetele atunci când o versiune nouă este disponibilă.",
"HowToUpgradeExplanation2": "Pe lângă actualizările standard NuGet-NPM, <a href=\"{0}\">ABP CLI</a> furnizează o comandă de actualizare automată care identifică şi actualizează toate pachetele legate de ABP din soluţia dumneavoastră.",
"DatabaseSupport": "Ce sisteme de baze de date sunt suportate?",
"DatabaseSupportExplanation": "ABP Framework în sine este agnostic din punct de vedere al bazei de date, şi poate funcţiona cu orice furnizor de baze de date prin natura sa. Consultaţi <a href=\"{0}\" target=\"_blank\">documentul de acces la date</a> pentru o listă a furnizorilor implementaţi până în clipa de faţă.",
"UISupport": "Ce UI frameworks sunt suportate?",
"Supported": "Suportat",
"UISupportExplanation": "ABP Framework în sine este agnostic din punct de vedere al UI framework şi poate funcţiona cu orice UI framework. Cu toate acestea, şabloanele de pornire, UI-urile modulelor şi temele nu au fost implementate pentru toate module UIs. Consultaţi <a href=\"{0}\">documentul de început</a> pentru lista actualizată a opţiunilor UI.",
"MicroserviceSupport": "Suportă arhitectura de micro-servicii?",
"MicroserviceSupportExplanation1": "Unul din scopurile principale ale ABP framework este să furnizeze o structură convenabilă pentru crearea soluţiilor micro-service. Consultaţi documentul <a href=\"{0}\">arhitectura micro-service</a> pentru a înţelege cum ajută la crearea sistemelor de micro-servicii.",
"MicroserviceSupportExplanation2": "Toate modulele ABP Commercial sunt dezvoltate pentru a suporta scenarii de implementare a micro-serviciilor (cu propriul API şi bază de date) prin respectarea <a href=\"{0}\">celor mai bune practici de dezvoltare a modulelor</a>.",
"MicroserviceSupportExplanation3": "Vă oferim <a href=\"{0}\">soluţie demo de micro-servicii</a> care demonstrează o implementare a arhitecturii de micro-servicii pentru a vă ajuta să vă creaţi propria soluţie.",
"MicroserviceSupportExplanation4": "Deci, răspunsul pe scurt este \"<strong>da, suportă o arhitectură de micro-service</strong>\".",
"MicroserviceSupportExplanation5": "Totuşi, un sistem de tip micro-service reprezintă o soluţie, şi fiecare soluţie va avea cerinţe diferite, topologie de reţea, scenarii de comunicare, posibilităţi de autentificare, separarea bazei de date/împărţirea deciziilor, configurări în timp real, integrări de sisteme terţe parte şi multe altele.",
"MicroserviceSupportExplanation6": "ABP Framework şi ABP Commercial furnizează infrastructura pentru scenarii de micro-servicii, module de micro-servicii compatibile, exemple şi documentaţie pentru a vă ajuta să vă construiţi propria soluţie. Dar nu vă aşteptaţi să descărcaţi direct soluţia visurilor dumneavoastră. Veţi fi nevoit(ă) să o întelegeţi şi să puneţi câteva piese cap la cap pe baza cerinţelor dumneavoastră.",
"WhereCanIDownloadSourceCode": "De unde pot descărca codul sursă?",
"WhereCanIDownloadSourceCodeExplanation": "Puteţi descărca codul sursă al tuturor modulelor ABP, pachetelor Angular şi al temelor prin suita ABP sau ABP CLI. Consultaţi <a href=\"{0}\">Cum descarc codul sursă?</a>",
"ComputerLimitation": "Pe câte calculatoare se poate autentifica un dezvoltator atunci când foloseşte ABP ABP?",
"ComputerLimitationExplanation": "Permitem în mod specific <strong>{0} staţii</strong> per dezvoltator individual/autorizat. Ori de câte ori este nevoie ca un dezvoltator să dezvolte produse ABP Commercial pe o altă staţie, un e-mail ar trebui să fie trimis la license@abp.io pentru a explica situaţia, şi atunci vom face alocarea corespunzătoare în sistemul nostru.",
"RefundPolicy": "Aveţi o politică de rambursare?",
"RefundPolicyExplanation": "Puteţi solicita un ramburs în decurs de <strong>30 de zile</strong> de la cumpărarea licenţei. Licenţele de tip Business şi Enterprise au opţiune de descărcare a codului sursă, prin urmare rambursul nu este posibil pentru licenţele Business and Enterprise (şi pentru oricare altă licenţă care beneficiază de dreptul de a primi codul sursă). În plus, nu se fac rambursări pentru reînnoiri şi achiziţii de licenţe suplimentare.",
"HowCanIRefundVat": "Cum pot deduce TVA-ul?",
"HowCanIRefundVatExplanation1": "Dacă aţi făcut plata folosind 2Checkout, puteţi deduce TVA-ul în contul dumneavoastră 2Checkout:",
"HowCanIRefundVatExplanation2": "Autentificaţi-vă în contul dumneavoastră <a href=\"https://secure.2checkout.com/cpanel/login.php\" target=\"_blank\">2Checkout</a>",
"HowCanIRefundVatExplanation3": "Găsiţi comanda respectivă şi apăsaţi pe \"Deducerea tardivă a TVA-ului \" (introduceţi CUI-ul dumneavoastră)",
"HowCanIGetMyInvoice": "Cum îmi pot obţine factura?",
"HowCanIGetMyInvoiceExplanation": "Există două procesatori de plăţi pentru cumpărarea unei licenţă: PayU şi 2Checkout. Dacă veţi cumpăra o licenţă prin procesatorul 2Checkout, vă va trimite chitanţa PDF pe adresa dumneavoastră de email, consultaţi <a href=\"https://knowledgecenter.2checkout.com/Documentation/03Billing-and-payments/Payment-operations/How-do-invoices-work\">facturarea 2Checkout.</a> Dacă veţi cumpăra o licenţă prin procesatorul PayU gateway sau prin transfer bancar, vă vom pregăti chitanţa şi vă vom trimite. Puteţi solicita chitanţa dumneavoastră din <a href=\"{0}\">pagina de gestionare a organizaţiei.</a>",
"Forum": "Forum",
"SupportExplanation": "Licenţele ABP Commercial furnizează asistenţă premium pe forum din partea unei echipe alcătuite din experţi ai ABP.",
"PrivateTicket": "Tichet privat",
"PrivateTicketExplanation": "Licenţa de tip Enterprise include de asemenea şi asistenţă privată cu sisteme de e-mail şi tichete.",
"AbpSuiteExplanation1": "Suita ABP vă permite să construiţi pagini web în decurs de câteva minute. Este un instrument global .NET Core care poate fi instalat din linia de comandă.",
"AbpSuiteExplanation2": "Poate crea o nouă soluţie ABP, genera paginii CRUD începând de la baza de date şi până la front-end. Pentru o prezentare tehnică consultaţi <a href=\"{0}\">documentul</a>",
"FastEasy": "Rapid şi usor",
"AbpSuiteExplanation3": "Suita ABP vă permite să creaţi pagini CRUD cu uşurinţă. Trebuie doar să vă definiţi entitatea şi proprietăţile sale, şi să lăsaţi restul în seama suitei ABP! Suita ABP generează tot codul necesar pentru pagina dumneavoastră CRUD în câteva secunde. Suportă interfeţe de utilizator Angular, MVC şi Blazor.",
"RichOptions": "Opţiuni ample",
"AbpSuiteExplanation4": "Suita ABP suportă multiple opţiuni UI precum <a href=\"https://docs.microsoft.com/en-us/aspnet/core/razor-pages\">Pagini Razor</a> şi <a href=\"https://angular.io\">Angular</a>. Suportă de asemenea şi baze de date multiple, precum <a href=\"https://www.mongodb.com\">MongoDB</a> şi toate bazele de date suportate de <strong>EntityFramework Core</strong> (MS SQL Server, Oracle, MySql, PostgreSQL şi <a href=\"https://docs.microsoft.com/en-us/ef/core/providers/?tabs=dotnet-core-cli\">altele</a>).",
"AbpSuiteExplanation5": "Partea bună este că dumneavoastră nu trebuie să vă faceţi griji despre aceste opţiuni. Suita ABP înţelege tipul proiectului dumneavoastră şi generează cod pentru proiectul dumneavoastră şi îl plasează în locul potrivit din proiect.",
"SourceCode": "Cod sursă",
"AbpSuiteExplanation6": "Suita ABP generează codul sursă pentru dumneavoastră! Nu generează fişiere magice pentru a genera pagina web. Suita ABP generează codul sursă petnru <strong>Entity, Repository, Application Service, Code First Migration, JavaScript/TypeScript şi CSHTML/HTML</strong> şi interfeţele necesare de asemenea. Suita ABP generează de asemenea codul respectând <strong>Cele mai bune practici</strong> ale dezvoltării software, deci nu trebuie să vă faceţi griji în legătură cu calitatea codului generat.",
"AbpSuiteExplanation7": "Deoarece aveţi codul sursă al blocurilor de construcţie ale paginii CRUD generate în straturile de aplicaţie corecte, puteţi modifica cu uşurinţă codul sursă şi puteţi injecta logica personalizată/business a codului generat.",
"CrossPlatform": "Platformă transversală",
"AbpSuiteExplanation8": "Suita ABP este construită cu .NET Core şi este o platformă transversală. Se execută ca o aplicaţie web pe calculatorul dumneavoastră local. O puteţi rula pe <strong>Windows</strong>, <strong>Mac</strong> şi <strong>Linux</strong>",
"OtherFeatures": "Alte funcţionalităţi",
"OtherFeatures1": "Actualizeată pachetele <strong>NuGet</strong> şi <strong>NPM</strong> din soluţia dumneavoastră cu uşurinţă.",
"OtherFeatures2": "Regenerează paginile deja generate de la zero.",
"OtherFeatures3": "Crează noi soluţii",
"ThanksForCreatingProject": "Mulţumim pentru că v-aţi creat proiectul!",
"HotToRunSolution": "Cum să rulaţi soluţia dumneavoastră?",
"HotToRunSolutionExplanation": "Consultaţi documentul de noţiuni de bază pentru a învăţa cum să vă configuraţi şi rulaţi aplicaţia.",
"GettingStarted": "Noţiuni de bază",
"WebAppDevTutorial": "Tutorial Web App Dev",
"WebAppDevTutorialExplanation": "Consultaţi tutorialul de dezvoltare a aplicaţiilor web pentru un exemplu de dezvoltare pas cu pas.",
"Document": "Document",
"UsingABPSuiteToCURD": "Folosind suita ABp pentru generarea şi utilizarea paginilor CRUD",
"SeeABPSuiteDocument": "Consultaţi documentaţia suitei ABP pentru a învăţa utilizarea suitei ABP.",
"AskQuestionsOnSupport": "Puteţi adresa întrebări pe ABP Commercial Support.",
"Documentation": "Documentaţie",
"SeeModulesDocument": "Consultaţi documentul modulelor pentru lista tuturor modulelor comerciale (pro) şi documentele acestora.",
"Pricing": "Preţuri",
"PricingExplanation": "Alegeţi caracteristicile şi funcţionalitatea nevoilor afacerii dumneavoastră acum. Actualizaţi-le cu uşurinţă pe măsură ce afacerea dumneavoastră se dezvoltă.",
"Team": "Team",
"Business": "Business",
"Enterprise": "Enterprise",
"Custom": "Personalizat",
"IncludedDeveloperLicenses": "Licenţe de dezvoltator incluse",
"CustomLicenceOrAdditionalServices": "Aveţi nevoie de o licenţă personalizată sau de servicii suplimentare?",
"CustomOrVolumeLicense": "Licenţă personalizată sau de volum",
"LiveTrainingSupport": "Instruire şi asistenţă în timp real",
"AndMore": "şi multe altele",
"AdditionalDeveloperLicense": "Licenţă suplimentară de dezvoltator",
"ProjectCount": "Număr de proiecte",
"AllProModules": "Toate modulele pro",
"AllProThemes": "Toate temele pro",
"AllProStartupTemplates": "Toate şabloanele de pornire pro",
"SourceCodeOfAllModules": "Codul sursă al tuturor modulelor",
"SourceCodeOfAllThemes": "Codul sursă al tuturor temelor",
"PerpetualLicense": "Licenţă perpetuă",
"UnlimitedServerDeployment": "Implementări nelimitate a serverului",
"YearUpgrade": "actualizare de 1 an",
"YearPremiumForumSupport": "Asistenţă premium pe forum de 1 an",
"ForumSupportIncidentCountYear": "Asistenţă pe forum număr de incidente/an",
"PrivateTicketEmailSupport": "Tichet privat şi asistenţă pe email",
"BuyNow": "Cumpără acum",
"PayViaAmexCard": "Cum pot plăti folosind cardul meu AMEX?",
"PayViaAmexCardDescription": "Procesatorul implicit 'Iyzico' s-ar putea să refuze anumite carduri AMEX ca măsură de securitate. În acest caz, puteţi plăti prin procesatorul alternativ '2Checkout'.",
"ThankYou": "Vă mulţumim",
"InvalidReCaptchaErrorMessage": "A intervenit o eroare în verificarea reCAPTCHA. Vă rugăm să încercaţi din nou.",
"CompanyName": "Nume companie",
"YourCompanyName": "Numele companiei dumneavoastră",
"FirstName": "Nume",
"LastName": "Prenume",
"Optional": "Opţional",
"YourFirstName": "Numele dumneavoastră",
"YourLastName": "Prenumele dumneavoastră",
"SpecialOffer": "Ofertă specială",
"SpecialOfferMessage": "Grăbiţi-vă! Preţurile sunt disponibile pentru o perioadă limitată.",
"DiscountRequest": "Cerere de discount",
"DiscountRequestDescribeCustomerQuestion": "Care din următoarele vă descrie?",
"DiscountRequestStudentEmailMessage": "Adresa de email trebuie să conţină 'edu'.",
"DiscountRequestDeveloperCount": "Câţi dezvoltatori aveţi?",
"DiscountRequestDeveloperCountExceedMessage": "Nu furnizăm licenţe reduse pentru companii care au peste {0} dezvoltatori.",
"DiscountRequestOrganizationName": "Numele companiei/organizaţiei/şcolii",
"Website": "Site web",
"GithubUsername": "Nume de utilizator GitHub",
"PhoneNumber": "Număr de telefon",
"Country": "Ţara",
"DescribeABPCommercialUsage": "Descrieţi proiectul pe care folosiţi să-l dezvoltaţi pe baza APB Commercial",
"DiscountRequestCertifyInformationMessage": "Certific că toate informaţiile sunt adevărate şi corecte.",
"DiscountRequestReceived": "Am primit solicitarea dumneavoastră de reducere.",
"DiscountRequestStatusMessage": "Vă vom răspunde după verificarea informaţiilor pe care le-aţi furnizat.",
"MVCOrRazorPages": "MVC (Razor Pages)",
"Angular": "Angular",
"Blazor": "Blazor",
"New": "New",
"MongoDB": "MongoDB",
"EBookDDD": "E-Book Domain Driven Design",
"PracticalGuideForImplementingDDD": "Această carte reprezintă un ghid practic pentru implementarea proiectării bazate pe domeniu în ABP Framework.",
"IntroducingDDD": "Introducing Domain Driven Design",
"DDDLayersAndCleanArchitecture": "DDD Layers & Clean Architecture",
"LayeringOfADotnetSolution": "Layering of a .NET Solution",
"ImplementingDDDBuildingBlocks": "Implementing DDD Building Blocks",
"DomainVsApplicationLogic": "Domain Logic vs Application Logic",
"SamplesAndDiscussions": "Exemple & Discuţii",
"Free": "Gratuit",
"Download": "Descarcă",
"DDDEBook": "DDD E-book",
"ImplementingDDD": "Implementing Domain Driven Design",
"DDDBookExplanation": "Ghid practic pentru implementarea proiectării bazate pe domeniu în ABP Framework.",
"Overview": "Prezentare generală",
"DDDBookPracticalGuide": "Acesta este un ghid practic pentru implementarea proiectării bazate pe domeniu (PBD). În timp ce detaliile de implementare se bazeată pe infrastructura ABP, conceptele de bază, principiile şi modelele sunt aplicabile în orice fel de soluţie, chiar dacă nu este o soluţie .NET.",
"TableOfContents": "Cuprins",
"IntroductionToImplementingDDD": "Introducere în implementarea proiectării bazate pe domeniu",
"WhatIsDDD": "Ce este proiectarea bazată pe domeniu?",
"Implementation": "Implementare",
"TheBigPicture": "Imaginea de ansamblu",
"TheBuildingBlock": "The Building Block",
"ExampleUseCase": "Exemplu de utilizare",
"DomainAndApplicationLogic": "Domain Logic & Application Logic",
"Author": "Autor",
"PublishedOn": "Publicat în",
"Page": "Pagina",
"FreeEBook": "E-Book gratuit",
"EBookSignInForDownload": "Pentru a descărca e-book-ul trebuie să vă autentificaţi",
"SignIn": "Autentificare",
"Or": "Sau",
"TellUsAboutYourself": "Spuneţi-ne despre dumneavoastră",
"Surname": "Nume",
"DoYouAgreePrivacyPolicy": "Sunt de acord cu <a href=\"/TermsConditions\">Termenii şi condiţiile<a/> şi <a href=\"/Privacy\">Politica de confidenţialitate</a>.",
"VolosoftMarketingInformationMessage": "Sunt de acord să primesc informaţii, sfaturi şi oferte despre soluţii pentru afaceri şi organizaţii şi alte produse şi servicii Volosoft.",
"VolosoftSharingInformationMessage": "Sunt de acord ca Volosoft să partajeze informaţiile mele cu partenerii selectaţi astfel încât să primesc informaţii relevante despre produsele şi serviciile lor.",
"WeWillSendYouADownloadLink": "Un link pentru descărcarea e-book-ului a fost trimis către {0}.<br/> Verificaţi-vă căsuţele de inbox / junk / spam!",
"InvalidFormInputs": "Vă rugăm să introduceţi informaţia validă specificată în formular.",
"DDDBookEmailBody": "Vă mulţumim. <br /> Pentru a descărca e-book-ul, <a href=\"{0}\">click aici</a>.",
"FreeDDDEBook": "DDD E-Book gratuit"
}
}

@ -0,0 +1,147 @@
{
"culture": "ro-RO",
"texts": {
"Permission:CommunityArticle": "Articol comunitar",
"Permission:Edit": "Modifică",
"Waiting": "În aşteptare",
"Approved": "Aprobat",
"Rejected": "Respins",
"Wait": "Pune în aşteptare",
"Approve": "Aprobă",
"Reject": "Respinge",
"ReadArticle": "Citeşte articolul",
"Status": "Starea",
"ContentSource": "Sursa conţinutului",
"Details": "Detalii",
"Url": "Url",
"Title": "Titlu",
"CreationTime": "Data creării",
"Save": "Salvează",
"SameUrlAlreadyExist": "Dacă doriţi să adăugaţi acest articol, ar trebui să schimbaţi adresa Url, întrucât aceeaşi adresa url există deja!",
"UrlIsNotValid": "Url-ul nu este valid.",
"UrlNotFound" : "Url nu a fost găsit.",
"UrlContentNotFound": "Conţinutul url-ului nu a fost găsit.",
"Summary": "Rezumat",
"MostRead": "Cele mai citite",
"Latest": "Recente",
"ContributeAbpCommunity": "Contribuiţi la comunitatea ABP",
"SubmitYourArticle": "Trimiteţi-vă postul",
"ContributionGuide": "Ghid de contribuţie",
"BugReport": "Raportează eroare",
"SeeAllArticles": "Vezi toate postările",
"WelcomeToABPCommunity!": "Bun venit în comunitatea ABP!",
"MyProfile": "Profilul meu",
"MyOrganizations": "Organizaţia mea",
"EmailNotValid": "Vă rugăm să introduceţi o adresa de email validă.",
"FeatureRequest": "Cerere de funcţionalitate",
"CreateArticleTitleInfo": "Titlul postării care va apărea în lista postărilor.",
"CreateArticleSummaryInfo": "Un scurt rezumat al postării care va apărea în lista postărilor.",
"CreateArticleCoverInfo": "Pentru crearea unei postări eficace, adăugaţi o poză de copertă. Încărcaţi imagini cu raport de aspect 16:9 pentru cea mai bună vizualizare. Dimensiunea maximă a fişierului: 1MB.",
"ThisExtensionIsNotAllowed": "Această extensie nu este permisă.",
"TheFileIsTooLarge": "Dimensiunea fişierului este prea mare.",
"GoToTheArticle": "Vezi articolul",
"Contribute": "Contribuie",
"OverallProgress": "Progres total",
"Done": "Gata",
"Open": "Deschide",
"Closed": "Închis",
"LatestQuestionOnThe": "Ultima întrebare de pe",
"Stackoverflow": "Stackoverflow",
"Votes": "voturi",
"Answer": "Răspunde",
"Views": "vizualizări",
"Answered": "Răspunse",
"WaitingForYourAnswer": "În aşteptarea răspunsului dumneavoastră",
"Asked": "întrebate",
"AllQuestions": "Toate întrebările",
"NextVersion": "Următoarea versiune",
"MilestoneErrorMessage": "Nu s-au putut obţine detaliile de referinţă actuale de pe Github.",
"QuestionItemErrorMessage": "Nu s-au putut obţine detaliile de referinţă actuale de pe Stackoverflow.",
"Oops": "Oops!",
"CreateArticleSuccessMessage": "Articolul a fost trimis cu succes. Va fi publicat după ce va fi revizuit de un administrator al site-ului.",
"ChooseCoverImage": "Alegeţi o poză de copertă...",
"CoverImage": "Poză de copertă",
"ShareYourExperiencesWithTheABPFramework": "Împărtăşiţi-vă experienţele cu ABP Framework!",
"Optional": "Opţional",
"UpdateUserWebSiteInfo": "Exemplu: https://johndoe.com",
"UpdateUserTwitterInfo": "Exemplu: johndoe",
"UpdateUserGithubInfo": "Exemplu: johndoe",
"UpdateUserLinkedinInfo": "Exemplu: https://www.linkedin.com/...",
"UpdateUserCompanyInfo": "Exemplu: Volosoft",
"UpdateUserJobTitleInfo": "Exemplu: Software Developer",
"UserName": "Nume de utilizator",
"Company": "Companie",
"PersonalWebsite": "Site web personal",
"RegistrationDate": "Data înregistrării",
"Social": "Social",
"Biography": "Biografie",
"HasNoPublishedArticlesYet": "nu a publicat niciun articol încă",
"Author": "Autor",
"LatestGithubAnnouncements": "Ultimele anunţuri de pe Github",
"SeeAllAnnouncements": "Vezi toate anunţurile",
"LatestBlogPost": "Ultima postare de pe blog",
"Edit": "Modifică",
"ProfileImageChange": "Schimbă poza de profil",
"BlogItemErrorMessage": "Nu s-au putut obţine detaliile ultimei postări de pe from ABP.",
"PlannedReleaseDate": "Data planificată de lansare",
"CommunityArticleRequestErrorMessage": "Nu s-a putut obţine ultima cerere de articol de pe Github.",
"ArticleRequestFromGithubIssue": "Momentan nu există cereri de articole.",
"LatestArticles": "Ultimele postări",
"ArticleRequests": "Cereri de articole",
"AllArticleRequests": "Vedeţi toate cererile de articole",
"SubscribeToTheNewsletter": "Abonaţi-vă la newsletter-ul nostru",
"NewsletterEmailDefinition": "Obţineţi informaţii despre evenimentele din ABP, cum ar fi lansări noi, surse gratuite, articole şi multe altele.",
"NoThanks": "Nu, mulţumesc",
"MaybeLater": "Poate mai târziu",
"JoinOurArticleNewsletter": "Abonaţi-vă la newsletter-ul nostru de articole",
"Community": "Comunitate",
"Marketing": "Marketing",
"CommunityPrivacyPolicyConfirmation": "Sunt de acord cu termenii şi condiţiile şi <a href=\"https://commercial.abp.io/Privacy\">politica de confidenţialitate</a>.",
"ArticleRequestMessageTitle": "<a href=\"https://github.com/abpframework/abp/issues/new\">Deschideţi o problemă pe</a> GitHub pentru a solicita un articol/tutorial pe care doriţi să îl vedeţi pe acest site web.",
"ArticleRequestMessageBody": "Aici, lista articolelor solicitate de către comunitate. Doriţi să scrieţi un articol solicitat? Vă rugăm să daţi click pe cerere şi să vă alăturaţi discuţiei.",
"Language": "Limba",
"CreateArticleLanguageInfo": "Limba pentru conţinutul postării.",
"VideoPost": "Post video",
"Article": "Articol",
"Read": "Citeşte",
"CreateGithubArticleUrlInfo": "Adresa URL originală GitHub a articolului.",
"CreateVideoContentUrlInfo": "Adresa URL originală YouTube a articolului",
"CreateExternalArticleUrlInfo": "Adresa URL originală externă a articolului.",
"VideoContentForm": "Trimite videoclip pe YouTube",
"GithubPostForm": "Trimite articol pe GitHub",
"ExternalPostForm": "Trimite un conţinut extern",
"HowToPost": "Cum postez?",
"Posts": "Postări",
"VideoUrl": "Url Video",
"GithubArticleUrl": "Adresa Url articol Github",
"ExternalArticleUrl": "Adresa Url externă Article",
"CreatePostCoverInfo": "Pentru crearea unei postări eficace, adăugaţi o poză de copertă. Încărcaţi imagini cu raport de aspect 16:9 pentru cea mai bună vizualizare. Dimensiunea maximă a fişierului: 1MB.",
"ThankYouForContribution": "Vă mulţumim pentru că aţi contribuit la comunitatea ABP.",
"GithubArticle": "Articol GitHub",
"GithubArticleSubmitStepOne": "<span class=\"font-weight-bold\">1.</span> Scrieţi un articol pe orice repository GitHub folosind formatul Markup. <a target=\"_blank\" href=\"https://github.com/abpframework/abp/blob/dev/docs/en/Community-Articles/2020-12-04-Event-Organizer/Post.md\">exemplu</a>",
"GithubArticleSubmitStepTwo": "<span class=\"font-weight-bold\">2.</span> Trimiteţi Url-ul articolului folosind formularul.",
"GithubArticleSubmitStepThree": "<span class=\"font-weight-bold\">3.</span> Articolul dumneavoastră va fi redat în acest site web.",
"YoutubeVideo": "Videoclip YouTube",
"YoutubeVideoSubmitStepOne": "<span class=\"font-weight-bold\">1.</span> Publicaţi-vă videoclipul pe YouTube.",
"YoutubeVideoSubmitStepTwo": "<span class=\"font-weight-bold\">2.</span> Trimieţi Url-ul videoclipului folosind formularul.",
"YoutubeVideoSubmitStepThree": "<span class=\"font-weight-bold\">3.</span> Vizitatorii vor putea să vă vizioneze conţinutul video direct de pe acest site web.",
"ExternalContent": "Conţinut extern",
"ExternalContentSubmitStepOne": "<span class=\"font-weight-bold\">1.</span> Creaţi conţinut pe orice platformă publică (mediu, blogul dumneavoastră sau oriunde doriţi).",
"ExternalContentSubmitStepTwo": "<span class=\"font-weight-bold\">2.</span> Trimiteţi-vă conţinutul URL folosind formularul.",
"ExternalContentSubmitStepThree": "<span class=\"font-weight-bold\">3.</span> Vizitatorii sunt redirecţionaţi către conţinutul de pe site-ul web original.",
"ChooseYourContentType": "Vă rugăm să alegeţi modalitatea pri ncare vreţi să vă adăugaţi conţinutul.",
"PostContentViaGithub": "Vreau să adaug articolul meu de pe <span class=\"font-weight-bold\"><i class=\"fa fa-github\"></i> GitHub</span> conform cu regulile de reducere.",
"PostContentViaYoutube": "Vreau să distribui videoclipurile mele disponibile de pe <span class=\"font-weight-bold\"><i class=\"fa fa-youtube\"></i> YouTube</span> aici.",
"PostContentViaExternalSource": "Vreau să adaug conţinutul pe care l-am publicat pe <span class=\"font-weight-bold\">altă platformă</span> aici.",
"GitHubUserNameValidationMessage": "Numele de utilizator al contului dumneavoastră de GitHub nu poate conţine spaţii goale, vă rugăm să verificaţi că numele de utilizator al contului dumneavoastră de GitHub este corect.",
"PersonalSiteUrlValidationMessage": "Url-ul site-ului dumneavoastră personal nu poate conţine spaţii goale, vă rugăm să verificaţi ca Url-ul site-ului dumneavoastră personal este corect.",
"TwitterUserNameValidationMessage": "Url-ul dumneavoastră de Twitter nu poate conţine spaţii goale, vă rugăm să verificaţi ca Url-ul de Twitter este corect.",
"LinkedinUrlValidationMessage": "Url-ul dumneavoastră de LinkedIn nu poate conţine spaţii goale, vă rugăm să verificaţi ca Url-ul de LinkedIn este corect.",
"NoPostsFound": "Nu s-au găsit postări!",
"SearchInPosts": "Caută în postări...",
"MinimumSearchContent": "Trebuie să introduceţi cel putin 3 caractere!",
"Volo.AbpIo.Domain:060001": "Sursa URL(\"{ArticleUrl}\") nu este URL GitHub",
"Volo.AbpIo.Domain:060002": "Conţinutul articolului nu este disponibil din resursa de pe GitHub(\"{ArticleUrl}\").",
"Volo.AbpIo.Domain:060003": "Nu a fost găsit conţinutul articolului!"
}
}

@ -0,0 +1,278 @@
{
"culture": "ro-RO",
"texts": {
"GetStarted": "Începeţi - Şabloane de pornire",
"Create": "Creează",
"NewProject": "Proiect nou",
"DirectDownload": "Descărcare directă",
"ProjectName": "Nume proiect",
"ProjectType": "Tip proiect",
"DatabaseProvider": "Provider bază de date",
"DatabaseManagementSystem": "Sistem de administrare bază de date",
"NTier": "N-Tier",
"IncludeUserInterface": "Include interfaţa userului",
"CreateNow": "Creează acum",
"TheStartupProject": "Proiectul de pornire",
"Tutorial": "Tutorial",
"UsingCLI": "Folosind CLI",
"SeeDetails": "Vedeţi detalii",
"AbpShortDescription": "ABP Framework este o infrastructură completă pentru crearea aplicaţiilor moderne web prin folosirea celor mai bune practici şi convenţii de dezvoltare software.",
"SourceCodeUpper": "COD SURSĂ",
"LatestReleaseLogs": "Cele mai recente jurnale de lansare",
"Infrastructure": "Infrastructura",
"Architecture": "Arhitectura",
"Modular": "Modular",
"DontRepeatYourself": "Nu vă repetaţi",
"DeveloperFocused": "Axat pe dezvoltator",
"FullStackApplicationInfrastructure": "Infrastructură de aplicaţii full stack.",
"DomainDrivenDesign": "Domain Driven Design",
"DomainDrivenDesignExplanation": "Proiectat şi dezvoltat pe baza modelelor şi principiilor DDD. Oferă un model stratificat pentru aplicaţia dumneavoastră.",
"Authorization": "Autorizare",
"AuthorizationExplanation": "Autorizare avansată pe bază de utilizator, rol şi sistem de permisiuni. Construit pe biblioteca Microsoft Identity.",
"MultiTenancy": "Multi-Tenancz",
"MultiTenancyExplanationShort": "Aplicaţii SaaS simplificate! Multi-Tenancy integrată de la baza de date până la UI.",
"CrossCuttingConcerns": "Cross Cutting Concerns",
"CrossCuttingConcernsExplanationShort": "Infrastructură completă pentru autorizare, validare, gestionarea excepţiilor, stocarea în cache, înregistrarea auditului, gestionarea tranzacţiilor şi multe altele.",
"BuiltInBundlingMinification": "Grupare şi minificare incorporate",
"BuiltInBundlingMinificationExplanation": "Nu este nevoie să utilizaţi instrumente externe pentru grupare şi minificare. ABP oferă un mod mai simplu, dinamic, puternic, modular şi incorporat!",
"VirtualFileSystem": "Sistem de fişiere virtuale",
"VirtualFileSystemExplanation": "Încorporaţi view-uri, scripturi, stiluri, imagini... in pachete/librării şi refolosiţi-le în diferite aplicaţii.",
"Theming": "Tematică",
"ThemingExplanationShort": "Utilizaţi şi personalizaţi tema UI standard, bazată pe UI, sau creaţi-o pe a dumneavoastră.",
"BootstrapTagHelpersDynamicForms": "Bootstrap Tag Helpers & Dynamic Forms",
"BootstrapTagHelpersDynamicFormsExplanation": "În loc să scrieţi manual detaliile repetate ale componentelor bootstrap, utilizaţi tag helperele ABP pentru a simplifica şi profita de inteligenţă. Construiţi rapid formulare de interfaţă bazate pe un model C# utilizând ajutorul pentru formulare dinamice.",
"HTTPAPIsDynamicProxies": "HTTP APIs & Dynamic Proxies",
"HTTPAPIsDynamicProxiesExplanation": "Expuneţi automat serviciile aplicaţiilor sub formă de API-uri HTTP şi folosiţi-le cu JavaScript dinamic si proxy C#.",
"CompleteArchitectureInfo": "Arhitectură modernă pentru a crea soluţii software care pot fi întreţinute.",
"DomainDrivenDesignBasedLayeringModelExplanation": "Vă ajută să implementaţi o arhitectură stratificată DDD şi să construiţi o bază de cod uşor de întreţinut.",
"DomainDrivenDesignBasedLayeringModelExplanationCont": "Oferă şabloane de pornire, abstractizări, clase de bază, servicii, documentaţie şi ghiduri pentru a vă ajuta să vă dezvoltaţi aplicaţia bazându-vă pe principiile şi tiparele DDD.",
"MicroserviceCompatibleModelExplanation": "Framework-ul de bază şi modulele pre-construite sunt proiectate având în vedere arhitectura microservice.",
"MicroserviceCompatibleModelExplanationCont": "Oferă infrastructură, integrări, exemple şi documentaţie pentru a implementa mai uşor soluţiile de microservicii, în timp ce nu aduce complexitate suplimentară dacă doriţi o aplicaţie monolitică.",
"ModularInfo": "ABP oferă un sistem de module care vă permite să dezvoltaţi module de aplicaţie reutilizabile, să vă conectaţi la evenimentele din ciclul de viaţă ale aplicaţiei şi să exprimaţi dependenţe între părţile centrale ale sistemului dumneavoastră.",
"PreBuiltModulesThemes": "Module şi teme pro-construite.",
"PreBuiltModulesThemesExplanation": "Modulele şi temele open source sunt gata de utilizare în aplicaţia dumneavoastră.",
"NuGetNPMPackages": "Pachete NuGet & NPM",
"NuGetNPMPackagesExplanation": "Distribuite sub formă de pachete NuGet & NPM. Uşor de instalat şi actualizat.",
"ExtensibleReplaceable": "Extensibil/Înlocuibil",
"ExtensibleReplaceableExplanation": "Toate serviciile şi modulele sunt dezvoltate pentru extensibilitate. Puteţi înlocui servicii, pagini, stiluri şi componente.",
"CrossCuttingConcernsExplanation2": "Menţineţi baza codului redusă, astfel încât să vă puteţi concentra asupra codului care e specific afacerii dumneavoastră.",
"CrossCuttingConcernsExplanation3": "Nu pierdeţi timp implementând cerinţe comune de aplicaţie pentru mai multe proiecte.",
"AuthenticationAuthorization": "Autentificare & Autorizare",
"ExceptionHandling": "Tratarea excepţiilor",
"Validation": "Validare",
"DatabaseConnection": "Conectare la baza de date",
"TransactionManagement": "Administrarea tranzacţiilor",
"AuditLogging": "Audit Logging",
"Caching": "Caching",
"Multitenancy": "Multitenancy",
"DataFiltering": "Filtrarea datelor",
"ConventionOverConfiguration": "Convention Over Configuration",
"ConventionOverConfigurationExplanation": "ABP implementează convenţii de aplicaţii comune în mod implicit, cu o configuraţie minimă sau zero.",
"ConventionOverConfigurationExplanationList1": "Auto înregistrează serviciile cunoscute pentru injectarea dependinţei..",
"ConventionOverConfigurationExplanationList2": "Expune serviciile aplicaţiei ca API-uri HTTP prin convenţii de denumire.",
"ConventionOverConfigurationExplanationList3": "Creează proxy-uri dinamice HTTP client pentru C# şi JavaScript.",
"ConventionOverConfigurationExplanationList4": "Oferă repository-uri implicite pentru entităţile dumneavoastră.",
"ConventionOverConfigurationExplanationList5": "Gestionează unitatea de lucru pe request web sau metodă de serviciu a aplicaţiei.",
"ConventionOverConfigurationExplanationList6": "Publică crearea, actualizarea şi ştergerea evenimentelor pentru entităţile dumneavoastră.",
"BaseClasses": "Clase de bază",
"BaseClassesExplanation": "Clase de bază pre-construite pentru tipare comune de aplicaţie.",
"DeveloperFocusedExplanation": "ABP este pentru dezvoltatori.",
"DeveloperFocusedExplanationCont": "Aceasta îşi propune să vă simplifice dezvoltarea zilnică de software, fără a vă restricţiona să scrieţi cod de nivel scăzut.",
"SeeAllFeatures": "Vedeţi toate funcţionalităţile",
"CLI_CommandLineInterface": "CLI (Command Line Interface)",
"CLI_CommandLineInterfaceExplanation": "Include un CLI pentru a vă ajuta să automatizaţi crearea de noi proiecte şi adăugarea de noi module.",
"StartupTemplates": "Proiecte de lansare",
"StartupTemplatesExplanation": "Diverse proiecte de lansare furnizează o soluţie complet configurată pentru a vă începe dezvoltarea.",
"BasedOnFamiliarTools": "Bazat pe instrumente familiare",
"BasedOnFamiliarToolsExplanation": "Construit şi integrat cu instrumente populare pe care le ştiţi deja. Curbă de învăţare joasă, adaptare uşoară, dezvoltare confortabilă.",
"ORMIndependent": "ORM Independent",
"ORMIndependentExplanation": "Framework-ul de bază este independent de ORM/bază de date şi poate funcţiona cu orice sursă de date. Furnizorii Entity Framework Core şi MongoDB sunt deja disponibili.",
"Features": "Exploraţi caracteristicile framework-ului ABP Framework",
"ABPCLI": "ABP CLI",
"Modularity": "Modularitate",
"BootstrapTagHelpers": "Bootstrap Tag Helpers",
"DynamicForms": "Formulare dinamice",
"BundlingMinification": "Grupare & Minificare",
"BackgroundJobs": "Background Jobs",
"BackgroundJobsExplanation": "Definiţi clase simple pentru a executa task-uri în fundal, aşezate în coadă. Folosiţi managerul de taskuri încorporat sau integraţi-l pe al dumneavoastră. Integrările <a href=\"{0}\">Hangfire</a> & <a href=\"{1}\">RabbitMQ</a> sunt deja disponibile.",
"DDDInfrastructure": "Infrastructură DDD",
"DomainDrivenDesignInfrastructure": "Infrastructură Domain Driven Design",
"AutoRESTAPIs": "Auto REST APIs",
"DynamicClientProxies": "Dynamic Client Proxies",
"DistributedEventBus": "Distributed Event Bus",
"DistributedEventBusWithRabbitMQIntegration": "Distributed Event Bus with RabbitMQ Integration",
"TestInfrastructure": "Infrastructură de test",
"AuditLoggingEntityHistories": "Jurnalul de audit şi istoricul entităţilor",
"ObjectToObjectMapping": "Mapare obiect la obiect",
"ObjectToObjectMappingExplanation": "Abstractizarea <a href=\"{0}\">mapării obiect la obiect</a> cu integrare AutoMapper.",
"EmailSMSAbstractions": "Abstractizări Email & SMS",
"EmailSMSAbstractionsWithTemplatingSupport": "Abstractizări de Email & SMS cu asistenţă pentru modelare",
"Localization": "Localizare",
"SettingManagement": "Administrarea setărilor",
"ExtensionMethods": "Metode de extindere",
"ExtensionMethodsHelpers": "Metode de extindere & Ajutoare",
"AspectOrientedProgramming": "Programare orientată spre aspect",
"DependencyInjection": "Injectarea de dependinţe",
"DependencyInjectionByConventions": "Injectarea dependinţei prin convenţii",
"ABPCLIExplanation": "ABP CLI (Command Line Interface) este un instrument de linie de comandă pentru efectuarea câtorva operaţii comune pentru soluţiile bazate pe ABP.",
"ModularityExplanation": "ABP asigură o infrastructură completă pentru construirea modulelor aplicaţiei dumneavoastră, care pot avea entităţi, servicii, integrări cu baze de date, API-uri, componente UI şi aşa mai departe..",
"MultiTenancyExplanation": "ABP nu numai că acceptă dezvoltarea de aplicaţii multi-tenant, dar, de asemenea, face ca şi codul dumneavoastră să nu fie în mare parte conştient de multi-tenancy.",
"MultiTenancyExplanation2": "Poate detecta automat tenantul curent, izola datele unor tenanţi diferiţi unul de celălalt.",
"MultiTenancyExplanation3": "Acceptă o singură bază de date, bază de date per tenant şi abordări hibride.",
"MultiTenancyExplanation4": "Vă focusaţi pe codul dumneavoastră şi lăsaţi framework-ul să se ocupe de multi-tenancy pentru dumneavoastră.",
"BootstrapTagHelpersExplanation": "În loc să scrieţi manual detaliile repetate ale componentelor bootstrap, folosiţi tag helperele ABP pentru a simplifica şi profita de inteligenţă. Puteţi folosi cu siguranţă Bootstrap ori de câte ori aveţi nevoie de el.",
"DynamicFormsExplanation": "Formulare dinamice & tag helpere de intrare pot crea formularul complet dintr-o clasă C# ca model.",
"AuthenticationAuthorizationExplanation": "Opţiuni variate de autentificare şi autorizare integrate cu ASP.NET Core Identity & IdentityServer4. Furnizează un sistem de permisiuni extensibil şi detaliat.",
"CrossCuttingConcernsExplanation": "Nu vă repetaţi pentru a implementa din nou şi din nou toate aceste lucruri obişniuite. Concentraţi-vă pe codul dumneavoastră de business şi lăsaţi ABP să le automatizeze prin convenţii.",
"DatabaseConnectionTransactionManagement": "Conectare la baza de date & Gestionarea tranzacţiilor",
"CorrelationIdTracking": "Correlation-Id Tracking",
"BundlingMinificationExplanation": "ABP oferă un sistem simplu, dinamic, puternic, modular şi încorporat de grupare şi minimizare.",
"VirtualFileSystemnExplanation": "Sistemul fişierelor virtuale face posibilă administrare fişierelor care nu există fizic pe sistemul de fişiere (disk). Este folosit în principal pentru a încorpora fişiere (js, css, images, cshtml...) în ansambluri şi să le folosească sub formă de fişiere fizice în timpul rulării.",
"ThemingExplanation": "Sistemul de teme permite dezvoltarea aplicaţiei & modulelor independente de tema prin definirea unui set comun de librării şi aspecte de bază, bazate pe ultimul framework Bootstrap.",
"DomainDrivenDesignInfrastructureExplanation": "O infrastructură completă pentru construirea aplicaţiilor stratificate, bazată pe tipare & principii Domain Driven Design;",
"Specification": "Specificaţie",
"Repository": "Repository",
"DomainService": "Domain Service",
"ValueObject": "Value Object",
"ApplicationService": "Application Service",
"DataTransferObject": "Obiect de transfer de date",
"AggregateRootEntity": "Aggregate Root, Entity",
"AutoRESTAPIsExplanation": "ABP poate configura automat serviciile aplicaţiei sub formă de controllere API prin convenţie.",
"DynamicClientProxiesExplanation": "Folosiţi-vă cu uşurinţă API-urile din JavaScript şi clienţi C#.",
"DistributedEventBusWithRabbitMQIntegrationExplanation": "Publicaţi şi utilizaţi cu uşurinţă evenimentele distribuite utilizând Distributed Event Bus încorporat cu integrarea RabbitMQ disponibilă.",
"TestInfrastructureExplanation": "Framework-ul a fost dezvoltat cu testarea unităţii şi integrării în minte. Vă oferă clase de bază pentru a o uşura. Şabloanele de pornire sunt pre-configurate pentru testare.",
"AuditLoggingEntityHistoriesExplanation": "Jurnal de audit încorporat pentru aplicaţii critice pentru afaceri. Solicitare, serviciu, jurnal de audit la nivel de metodă şi istoric de entităţi cu detalii la nivel de proprietate.",
"EmailSMSAbstractionsWithTemplatingSupportExplanation": "Abstractizările IEmailSender şi ISmsSender decuplează logica aplicaţiei de infrastructură. Sistemul avansat de şabloane de email permite crearea şi localizarea şabloanelor de email şi utilizarea lor cu uşurinţă ori de câte ori este necesar.",
"LocalizationExplanation": "Sistemul de localizare vă permite să creaţi resurse în JSON simplu şi să le folosiţi pentru a vă localiza UI-ul dumneavoastră. Suportă sisteme avansate precum moştenirea, extinderea, şi integrarea cu JavaScript în timp ce e pe deplin compatibilă cu sistemul de localizare al AspNet Core.",
"SettingManagementExplanation": "Definiţi setări pentru aplicaţia dumneavoastră şi Define settings for your application and get values on runtime based on the current configuration, tenant and user.",
"ExtensionMethodsHelpersExplanation": "Don't repeat yourself even for trivial code parts. Extensions & helpers for standard types makes your code much cleaner and easy to write.",
"AspectOrientedProgrammingExplanation": "Furnizează o infrastructură confortabilă pentru crearea proxy-urilor dinamice şi implementarea programării orientată aspect. Interceptează orice clasă şi execută codul dumneavoastră înainte & după orice executare de metodă.",
"DependencyInjectionByConventionsExplanation": "Nu este nevoie să vă înregistraţi clasele pentru injectarea manuală a dependinţelor. Înregistrează în mod automat tipurile de servicii comune prin convenţie. Pentru alte tipuri de servicii, puteţi folosi interfeţe şi atribute pentru a o face mai uşor si pe loc.",
"DataFilteringExplanation": "Definiţi şi utilizaţi filtre de date care sunt aplicate automat atunci când interogaţi entităţi din baza de date. Filtre Soft Delete & MultiTenant sunt prevăzute în mod implicit atunci când implementaţi interfeţe simple.",
"PublishEvents": "Publicaţi evenimente",
"HandleEvents": "Gestionaţi evenimente",
"AndMore": "şi multe altele...",
"Code": "Cod",
"Result": "Rezultat",
"SeeTheDocumentForMoreInformation": "Consultaţi <a href=\"{1}\">{0} documentul</a> pentru mai multe informaţii",
"IndexPageHeroSection": "<span class=\"first-line shine\"><strong>open source</strong></span><span class=\"second-line text-uppercase\">Web Application<br />Framework </span><span class=\"third-line shine2\"><strong>pentru asp.net core</strong></span>",
"UiFramework": "Framework UI",
"EmailAddress": "Adresă de email",
"Mobile": "Mobil",
"ReactNative": "React nativ",
"Strong": "Puternic",
"Complete": "Complet",
"BasedLayeringModel": "Based Layering Model",
"Microservice": "Microserviciu",
"Compatible": "Compatibil",
"MeeTTheABPCommunityInfo": "Misiunea noastră este de a crea un mediu în care dezvoltatorii să se ajute reciproc cu articole, tutoriale, studii de caz etc. şi să cunoască oameni cu aceleaşi idei.",
"JoinTheABPCommunityInfo": "Implicaţi-vă într-o comunitate vibrantă şi deveniţi un contribuitor la ABP Framework!",
"AllArticles": "Toate articolele",
"SubmitYourArticle": "Trimiteţi-vă articolul",
"DynamicClientProxyDocument": "Consultaţi documentaţia pentru client proxy dinamic pentru <a href=\"{0}\">JavaScript</a> & <a href=\"{1}\">C#</a>.",
"EmailSMSAbstractionsDocument": "Consultaţi documentele de <a href=\"{0}\">emailing</a> and <a href=\"{1}\">trimiterea de SMS</a> pentru mai multe informaţii.",
"CreateProjectWizard": "Acest formular crează un proiect nou începând cu şablonul de pornire, care este configurat corespunzător pentru a începe direct cu proiectul.",
"TieredOption": "Crează o soluţie pe mai multe niveluri în care straturile Web API şi HTTP sunt separate fizic. Dacă nu este bifată, creează o soluţie stratificată mai puţin complexă şi potrivită pentru majoritatea scenariilor.",
"SeparateIdentityServerOption": "Separă partea serverului în două aplicaţii: prima este pentru serverul de identitate, iar a doua pentru serverul dumneavoastră de HTTP API.",
"UseslatestPreVersion": "Foloseşte ultima versiune pre-lansare",
"ReadTheDocumentation": "<span class=\"text-primary\">Citiţi</span><span class=\"text-success\"> Documentaţia</span>",
"Documentation": "Documentaţie",
"GettingStartedTutorial": "Tutorial de pornire",
"ApplicationDevelopmentTutorial": "Tutorial de dezvoltare a aplicaţiei",
"TheStartupTemplate": "Şablonul de pornire",
"InstallABPCLIInfo": "ABP CLI este cea mai rapidă metodă de a porni o soluţie nouă cu framework-ul ABP. Instalaţi ABP CLI folosind o fereastră de linii de comandă:",
"DifferentLevelOfNamespaces": "Puteţi folosi nivele diferite de spaţii de nume; e.g. BookStore, Acme.BookStore or Acme.Retail.BookStore.",
"ABPCLIExamplesInfo": "comanda <strong>new</strong> creează o <strong>aplicaţie stratificată MVC</strong> cu <strong>Entity Framework Core</strong> ca şi furnizor de baze de date. Totuşi, are şi alte opţiuni. Exemple:",
"SeeCliDocumentForMoreInformation": "Consultaţi <a href=\"{0}\">documentul ABP CLI</a> pentru mai multe opţiuni sau selectaţi tab-ul de \"Descărcare directă\" de deasupra.",
"Optional": "Opţional",
"LocalFrameworkRef": "Pastraţi referinţa proiectului local pentru pachetele framework-ului.",
"BlobStoring": "Depozitare BLOB",
"BlobStoringExplanation": "Sistemul de depozitare BLOB oferă o abstractizare pentru lucrul cu BLOB-uri. ABP oferă câteva integrări prestabilite pentru furnizorii de stocare (Azure, AWS, File System, Database, etc.) pe care le puteţi folosi cu uşurinţă în aplicaţiile dumneavoastră.",
"TextTemplating": "Modelare text",
"TextTemplatingExplanation": "Modelarea textului este folosită pentru a reda dinamic conţinuturile bazate pe un şablon şi un model (un obiect de date). De exemplu, o puteţi folosi pentru a crea conţinuturi dinamice de email cu un şablon predefinit.",
"MultipleUIOptions": "Opţiuni UI multiple",
"MultipleDBOptions": "Furnizori de baze de date multipli",
"MultipleUIOptionsExplanation": "Framework-ul de bază este conceput ca independent de UI şi poate funcţiona cu orice tip de sistem UI, în timp ce mai multe opţiuni pre-construite şi integrate sunt oferite din start.",
"MultipleDBOptionsExplanation": "Framework-ul poate lucra cu orice sursă de date, în timp ce furnizorii următori sunt dezvoltaţi şi acceptaţi oficial;",
"SelectLanguage": "Selectaţi limba",
"LatestArticleOnCommunity": "Ultimul articol de pe comunitatea ABP",
"Register": "Întregistrare",
"IsDownloadable": "Poate fi descărcat",
"DatabaseOptions": "Opţiunile bazei de date",
"BackToPackagesPage": "Înapoi la pagina pachetelor",
"HowToInstall": "Cum instalez",
"SeeOnNpm": "Vedeţi pe NPM",
"SeeOnNuget": "Vedeţi pe Nuget",
"MVCGulpCommandExplanation": "Dacă folosiţi MVC (Razor Pages) UI, atunci rulaţi comanda the \"gulp\" după instalarea pachetelor.",
"UsingABPCLI": "Folosind <a href=\"https://docs.abp.io/en/abp/latest/CLI\">Abp CLI</a>",
"WithoutABPCLI": "Fară ABP CLI",
"ABPCLIModuleDependency": "Abp Cli adaugă automat dependenţa modulelor.",
"AddModuleDependency": "Apoi adăugaţi dependenţa modulului",
"Packages": "Pachete",
"NugetPackages": "Pachete Nuget",
"NPMPackages": "Pachete NPM",
"SeeDocs": "Consultaţi documentele",
"None": "Niciuna",
"Application": "Aplicaţie",
"Module": "Module",
"PackageName": "Nume pachet",
"LicenseURL": "URL licenţă",
"License": "Licenţă",
"ProjectCreationSuccessMessage": "Proiectul dumneavoastră a fost creat cu succes",
"HowToRunSolution": "Cum vă rulaţi soluţia?",
"GettingStartedMessage": "Consultaţi documentul de pornire pentru a învăţa cum să vă configuraţi şi să rulaţi soluţia dumneavoastră.",
"WebAppDevTutorial": "Tutorial Web App Dev",
"WebAppDevTutorialMessage": "Consultaţi documentul de dezvoltare a aplicaţiilor web pentru un exemplu de dezvoltare pas cu pas.",
"CommunityArticles": "Articolele comunităţii",
"CommunityArticleMessage": "Verificaţi platforma comunităţii ABP pentru a citi articole utile pentru framework-ul ABP.",
"InvestigateSolutionDetails": "Investigaţi detaliile soluţiei",
"StartupTemplateDocumentationMessage": "Consultaţi documentul şablonului de pornire a aplicaţiei pentru a învăţa arhitectura şi strucutra soluţiei dumneavoastră.",
"ClientSideDevelopment": "Dezvoltarea clientului",
"ClientSideDevelopmentDocumentationMessage": "Consultaţi documentul {0} pentru a învăţa punctele cheie pentru dezvoltarea interfeţei de utilizator.",
"DatabaseProviderDocumentationMessage": "Consultaţi documentul {0} pentru a învăţa punctele cheie pentru dezvoltarea straturilor bazei de date.",
"ABPCommercialExplanationMessage": "ABP Commercial oferă module premium, teme, isntrumente şi asistenţă pentru framework-ul ABP.",
"ImplementingDDD": "Implementarea Domain Driven Design",
"DDDBookExplanation": "Un ghid practic pentru implementarea Domain Driven Design în framework-ul ABP.",
"Overview": "Prezentare generală",
"DDDBookPracticalGuide": "Acesta este un ghid practic pentru implementarea Domain Driven Design (DDD). În timp ce detaliile implementării se bazează pe infrastructura framework-ului ABP, conceptele de bază, principiile şi modelele pot fi aplicate oricărei soluţii, chiar dacă nu este una .NET.",
"TableOfContents": "Cuprins",
"IntroductionToImplementingDDD": "Introducere în Implementarea Domain Driven Design",
"WhatIsDDD": "Ce este Domain Driven Design?",
"Implementation": "Implementare",
"TheBigPicture": "Privire de ansamblu",
"TheBuildingBlock": "The Building Block",
"ExampleUseCase": "Exemplu de Use Case",
"DomainAndApplicationLogic": "Domain Logic & Application Logic",
"Author": "Autor",
"Pages": "Pagini",
"PublishedOn": "Publicat pe",
"FreeEBook": "E-Book gratuit",
"Download": "Descărcaţi",
"EBookSignInForDownload": "Pentru a descărca e-book-ul trebuie să vă autentificaţi",
"SignIn": "Autentificaţi-vă",
"Or": "Sau",
"TellUsAboutYourself": "Spuneţi-ne un pic despre dumneavoastră",
"Name": "Nume",
"Surname": "Nume",
"CompanyName": "Nume companie",
"DoYouAgreePrivacyPolicy": "Sunt de acord cu <a href=\"https://account.abp.io/Account/TermsConditions\">Termenii & condiţiile</a> şi <a href=\"https://account.abp.io/Account/Privacy\">Politica de confidenţialitate</a>.",
"Free": "Gratuit",
"DDDEBook": "E-book DDD",
"PracticalGuideForImplementingDDD": "Această carte este un ghid practic pentru implementarea Domain Driven Design în framework-ul ABP.",
"IntroducingDDD": "Introducere în Domain Driven Design",
"DDDLayersAndCleanArchitecture": "Straturile DDD & Arhitectură curată",
"LayeringOfADotnetSolution": "Stratificarea unei soluţii .NET",
"ImplementingDDDBuildingBlocks": "Implementând DDD Building Blocks",
"DomainVsApplicationLogic": "Domain Logic vs Application Logic",
"SamplesAndDiscussions": "Exemple & Discuţii",
"EmailNotValid": "Vă rugăm să introduceţi o adresa de email validă.",
"WeWillSendYouADownloadLink": "Un link care conţine e-book-ul a fost trimis către {0}. Verificaţi-vă folderele de inbox, junk sau spam!",
"GoHome": "Pagina principală",
"InvalidFormInputs": "Vă rugăm, introduceţi informaţia validă specificată în formular.",
"DDDBookEmailBody": "Vă mulţumim. <br /> Pentru a descărca cartea, <a href=\"{0}\">click aici</a>.",
"SubscribeToNewsletter": "Abonaţi-vă la newsletter pentru a primi informaţii despre evnimentele din platforma ABP.IO, precum noi lansări, articole, oferte şi multe altele.",
"FirstEdition": "Prima ediţie",
"ThankYou": "Vă mulţumim!",
"CheckboxMandatory": "Trebuie să bifaţi asta pentru a continua!"
}
}

@ -19,4 +19,13 @@
<IncludeAssets>runtime; build; native; contentfiles; analyzers</IncludeAssets>
</PackageReference>
</ItemGroup>
<ItemGroup>
<Content Remove="*.abppkg.json"/>
<None Include="*.abppkg.json">
<ExcludeFromSingleFile>true</ExcludeFromSingleFile>
<CopyToPublishDirectory>Never</CopyToPublishDirectory>
</None>
</ItemGroup>
</Project>

@ -0,0 +1,140 @@
# Swagger Integration
[Swagger (OpenAPI)](https://swagger.io/) is a language-agnostic specification for describing REST APIs. It allows both computers and humans to understand the capabilities of a REST API without direct access to the source code. Its main goals are to:
- Minimize the amount of work needed to connect decoupled services.
- Reduce the amount of time needed to accurately document a service.
ABP Framework offers a prebuilt module for full Swagger integration with small configurations.
## Installation
> This package is already installed by default with the startup template. So, most of the time, you don't need to install it manually.
If installation is needed, it is suggested to use the [ABP CLI](CLI.md) to install this package.
### Using the ABP CLI
Open a command line window in the folder of the `Web` or `HttpApi.Host` project (.csproj file) and type the following command:
```bash
abp add-package Volo.Abp.Swashbuckle
```
### Manual Installation
If you want to manually install;
1. Add the [Volo.Abp.Swashbuckle](https://www.nuget.org/packages/Volo.Abp.Swashbuckle) NuGet package to your `Web` or `HttpApi.Host` project:
`Install-Package Volo.Abp.Swashbuckle`
2. Add the `AbpSwashbuckleModule` to the dependency list of your module:
```csharp
[DependsOn(
//...other dependencies
typeof(AbpSwashbuckleModule) // <-- Add module dependency like that
)]
public class YourModule : AbpModule
{
}
```
## Configuration
First, we need to use `AddAbpSwaggerGen` extension to configure Swagger in `ConfigureServices` method of our module.
```csharp
public override void ConfigureServices(ServiceConfigurationContext context)
{
var services = contex.Services;
//... other configarations.
services.AddAbpSwaggerGen(
options =>
{
options.SwaggerDoc("v1", new OpenApiInfo { Title = "Test API", Version = "v1" });
options.DocInclusionPredicate((docName, description) => true);
options.CustomSchemaIds(type => type.FullName);
}
);
}
```
Then we can use Swagger UI by calling `UseAbpSwaggerUI` method in the `OnApplicationInitialization` method of our module.
```csharp
public override void OnApplicationInitialization(ApplicationInitializationContext context)
{
var app = context.GetApplicationBuilder();
//... other configarations.
app.UseAbpSwaggerUI(options =>
{
options.SwaggerEndpoint("/swagger/v1/swagger.json", "Test API");
});
//... other configarations.
}
```
## Using Swagger with OAUTH
For non MVC/Tiered applications, we need to configure Swagger with OAUTH to handle authorization.
> ABP Framework uses IdentityServer by default. To get more information about IDS, check this [documentation](Modules/IdentityServer.md).
To do that, we need to use `AddAbpSwaggerGenWithOAuth` extension to configure Swagger with OAuth issuer and scopes in `ConfigureServices` method of our module.
```csharp
public override void ConfigureServices(ServiceConfigurationContext context)
{
var services = contex.Services;
//... other configarations.
services.AddAbpSwaggerGenWithOAuth(
"https://localhost:44341", // authority issuer
new Dictionary<string, string> //
{ // scopes
{"Test", "Test API"} //
}, //
options =>
{
options.SwaggerDoc("v1", new OpenApiInfo { Title = "Test API", Version = "v1" });
options.DocInclusionPredicate((docName, description) => true);
options.CustomSchemaIds(type => type.FullName);
}
);
}
```
Then we can use Swagger UI by calling `UseAbpSwaggerUI` method in the `OnApplicationInitialization` method of our module.
> Do not forget to set `OAuthClientId` and `OAuthClientSecret`.
```csharp
public override void OnApplicationInitialization(ApplicationInitializationContext context)
{
var app = context.GetApplicationBuilder();
//... other configarations.
app.UseAbpSwaggerUI(options =>
{
options.SwaggerEndpoint("/swagger/v1/swagger.json", "Test API");
var configuration = context.ServiceProvider.GetRequiredService<IConfiguration>();
options.OAuthClientId("Test_Swagger"); // clientId
options.OAuthClientSecret("1q2w3e*"); // clientSecret
});
//... other configarations.
}
```

@ -79,3 +79,26 @@ After you have installed these NuGet packages, you need to configure your projec
}
````
### Dashboard Authorization
Hangfire Dashboard provides information about your background jobs, including method names and serialized arguments as well as gives you an opportunity to manage them by performing different actions retry, delete, trigger, etc. So it is important to restrict access to the Dashboard.
To make it secure by default, only local requests are allowed, however you can change this by following the [official documentation](http://docs.hangfire.io/en/latest/configuration/using-dashboard.html) of Hangfire.
You can integrate the Hangfire dashboard to [ABP authorization system](Authorization.md) using the **AbpHangfireAuthorizationFilter**
class. This class is defined in the `Volo.Abp.Hangfire` package. The following example, checks if the current user is logged in to the application:
app.UseHangfireDashboard("/hangfire", new DashboardOptions
{
AsyncAuthorization = new[] { new AbpHangfireAuthorizationFilter() }
});
If you want to require an additional permission, you can pass it into the constructor as below:
app.UseHangfireDashboard("/hangfire", new DashboardOptions
{
AsyncAuthorization = new[] { new AbpHangfireAuthorizationFilter("MyHangFireDashboardPermissionName") }
});
**Important**: `UseHangfireDashboard` should be called after the authentication middleware in your `Startup` class (probably at the last line). Otherwise,
authorization will always fail!

@ -213,7 +213,7 @@ There are some breaking changes with the Blazor UI. If you've built an applicati
When you create a new project, profile management doesn't work, you get an exception because it can't find the `/libs/cropperjs/css/cropper.min.css` file. To fix the issue;
* Add `"@volo/account": "^3.3.0-rc.1"` to the `package.json` in the `.Host` project.
* Run `yarn` (or `npm install`), then `gulp` on a command line terminal in the root folder of the `.Host` project.
* Run `yarn` (or `npm install`), then `abp install-libs` on a command line terminal in the root folder of the `.Host` project.
### Multi-Tenant Social Logins

@ -0,0 +1,42 @@
# ABP.IO Platform 4.4 Final Has Been Released!
[ABP Framework](https://abp.io/) and [ABP Commercial](https://commercial.abp.io/) 4.4 versions have been released today.
## What's New With 4.4?
Since all the new features are already explained in details with the 4.4 RC announcement posts, I will not repeat all the details again. See [the related blog post](https://blog.abp.io/abp/ABP-Platform-4-4-RC-Has-Been-Released) for all the features and enhancements.
## How to Upgrade an Existing Solution
### Install/Update the ABP CLI
First of all, install the ABP CLI or upgrade to the latest version.
If you haven't installed yet:
```bash
dotnet tool install -g Volo.Abp.Cli
```
To update an existing installation:
```bash
dotnet tool update -g Volo.Abp.Cli
```
### ABP UPDATE Command
[ABP CLI](https://docs.abp.io/en/abp/latest/CLI) provides a handy command to update all the ABP related NuGet and NPM packages in your solution with a single command:
```bash
abp update
```
Run this command in the root folder of your solution.
## The Road Map
The next feature version will be 5.0. It is planned to release the 5.0 RC (Release Candidate) in November 2021. See the updated road maps;
* [ABP Framework Road Map](https://docs.abp.io/en/abp/latest/Road-Map)
* [ABP Commercial Road Map](https://docs.abp.io/en/commercial/latest/road-map)

@ -12,6 +12,7 @@ The following features are currently available:
* Provides a [**comment**](Comments.md) system to add comments feature to any kind of resource, like blog post or a product review page.
* Provides a [**reaction**](Reactions.md) system to add reactions (smileys) feature to any kind of resource, like a blog post or a comment.
* Provides a [**rating**](Ratings.md) system to add rating feature to any kind of resource.
* Provides a [**menu**](Menus.md) system to manage public menus dynamically.
Click to a feature to understand and learn how to use it.

@ -0,0 +1,88 @@
# CMS Kit: Pages
CMS Kit Menu system allows to manage public menus dynamically.
## The User Interface
### Menu items
CMS Kit module admin side adds the following items to the main menu, under the *CMS* menu item:
* **Menus**: Menu management page.
`CmsKitAdminMenus` class has the constants for the menu item names.
### Menus
#### Menu Management
Menus page is used to manage dynamic public menus in the system.
![cms-kit-menus-page](../../images/cmskit-module-menus-page.png)
Created menus will be visible on public site.
![cms-kit-public-menus](../../images//cmskit-module-menus-public.png)
# Internals
## Domain Layer
#### Aggregates
This module follows the [Entity Best Practices & Conventions](https://docs.abp.io/en/abp/latest/Best-Practices/Entities) guide.
##### Menus
- `MenuItem` (aggregate root): A Menu Item presents a single node at menu tree.
#### Repositories
This module follows the [Repository Best Practices & Conventions](https://docs.abp.io/en/abp/latest/Best-Practices/Repositories) guide.
Following custom repositories are defined for this feature:
- `IMenuItemRepository`
#### Domain services
This module follows the [Domain Services Best Practices & Conventions](https://docs.abp.io/en/abp/latest/Best-Practices/Domain-Services) guide.
##### Menu Item Manager
`MenuItemManager` is used to perform some operations for the `MenuItemManager` aggregate root.
### Application layer
#### Application services
- `MenuItemAdminAppService` (implements `IMenuItemAdminAppService`): Implements the management operations of menu system.
- `MenuItemPublicAppService` (implements `IMenuItemPublicAppService`): Implements the public use cases of menu system.
### Database providers
#### Common
##### Table / collection prefix & schema
All tables/collections use the `Cms` prefix by default. Set static properties on the `CmsKitDbProperties` class if you need to change the table prefix or set a schema name (if supported by your database provider).
##### Connection string
This module uses `CmsKit` for the connection string name. If you don't define a connection string with this name, it fallbacks to the `Default` connection string.
See the [connection strings](https://docs.abp.io/en/abp/latest/Connection-Strings) documentation for details.
#### Entity Framework Core
##### Tables
- CmsMenuItems
#### MongoDB
##### Collections
- CmsMenuItems

@ -188,7 +188,7 @@ Open `package.json` and add `@abp/docs": "^2.9.0` as shown below:
Then open the command line terminal in the `Acme.MyProject.Web` project folder and run the following command:
1. `yarn`
2. `gulp`
2. `abp install-libs`
### 4- Database Integration

@ -61,7 +61,7 @@ Or you can also manually install nuget package to `Acme.MyProject.Web` project:
Then open the command line terminal in the `Acme.MyProject.Web` project folder and run the following command:
1. `yarn`
2. `gulp`
2. `abp install-libs`
That's all,Now run the application and Navigate to `/VirtualFileExplorer`. You will see virtual file explorer page:

@ -70,10 +70,10 @@ This will add the `@abp/signalr` to the dependencies in the `package.json` of yo
}
````
Run the `gulp` in the root folder of your web project:
Run `abp install-libs` command in the root folder of your web project:
````bash
gulp
abp install-libs
````
This will copy the SignalR JavaScript files into your project:

@ -1,3 +0,0 @@
# Swagger UI Integration
TODO

@ -0,0 +1,107 @@
# Angular UI: Basic Theme
The Basic Theme is a theme implementation for the Angular UI. It is a minimalist theme that doesn't add any styling on top of the plain [Bootstrap](https://getbootstrap.com/). You can take the Basic Theme as the **base theme** and build your own theme or styling on top of it. See the *Customization* section.
> If you are looking for a professional, enterprise ready theme, you can check the [Lepton Theme](https://commercial.abp.io/themes), which is a part of the [ABP Commercial](https://commercial.abp.io/).
> See the [Theming document](Theming.md) to learn about themes.
## Installation
**This theme is already installed** when you create a new solution using the [startup templates](../../Startup-Templates/Index.md). If you need to manually install it, follow the steps below:
* Install the [@abp/ng.theme.basic](https://www.npmjs.com/package/@abp/ng.theme.basic) NPM package to your Angular project.
* Open the `src/app/app.module.ts` file, import `ThemeBasicModule` (it can be imported from `@abp/ng.theme.basic` package), and add `ThemeBasicModule.forRoot()` to the `imports` array.
* Open the `src/app/shared/shared.module` file, import `ThemeBasicModule` (it can be imported from `@abp/ng.theme.basic` package), and add `ThemeBasicModule` to the `imports` and `exports` array.
The `ThemeBasicModule` is registered own layouts (`ApplicationLayoutComponent`, `AccountLayoutComponent`, `EmptyLayoutComponent`) to a service which is exposed by `@abp/ng.core` package on application initialization.
## Application Layout
![basic-theme-application-layout](../../images/basic-theme-application-layout.png)
Application Layout implements the following parts, in addition to the common parts mentioned above;
* Logo area
* Routes area
* Language selection & user menu
* [Page Alerts](Page-Alerts.md)
See Application Layout components:
![application layout components](./images/layout-components.png)
### How to Use a Layout
Routes should be added to the menu by calling `add` method `RoutesService`. A layout can be set in the object of your route. See the [modifying the menu](Modifying-the-Menu#how-to-add-a-navigation-element) for more information.
## Customization
You have two options two customize this theme:
### Overriding Styles / Components
In this approach, you continue to use the theme as an NPM package and customize the parts you need to. There are several ways to customize it;
#### Override the Styles
You can simply override the styles in the global styles (`src/styles.scss`) file of your application.
#### Override the Components
See the [Component Replacement](Component-Replacement.md) to learn how you can replace components, customize and extend the user interface.
### Copy & Customize
You can run the following [ABP CLI](../../CLI.md) command in **Angular** project directory to copy the source code to your solution:
`abp add-package @abp/ng.theme.basic --with-source-code`
----
Or, you can download the [source code](https://github.com/abpframework/abp/blob/dev/npm/ng-packs/packages/theme-basic) of the Basic Theme, manually copy the project content into your project (`projects/theme-basic` folder), open `angular.json` file and add configuration below to the `projects` object:
```json
{
"projects": {
...
"theme-basic": {
"projectType": "library",
"root": "projects/theme-basic",
"sourceRoot": "projects/theme-basic/src",
"prefix": "abp",
"architect": {
"build": {
"builder": "@angular-devkit/build-ng-packagr:build",
"options": {
"tsConfig": "projects/theme-basic/tsconfig.lib.json",
"project": "projects/theme-basic/ng-package.json"
},
"configurations": {
"production": {
"tsConfig": "projects/theme-basic/tsconfig.lib.prod.json"
}
}
}
}
}
}
}
```
Then, open the `tsconfig.json` file and add new paths as follows:
```json
"paths": {
...
"@abp/ng.theme.basic": ["projects/theme-basic/src/public-api.ts"],
"@abp/ng.theme.basic/testing": ["projects/theme-basic/testing/src/public-api.ts"]
}
```
You can now freely customize the theme based on your application requirements.
## See Also
* [Theming](Theming.md)

@ -0,0 +1,228 @@
# Angular UI: Theming
## Introduction
ABP Framework provides a complete **UI Theming** system with the following goals:
* Reusable [application modules](../../Modules/Index.md) are developed **theme-independent**, so they can work with any UI theme.
* UI theme is **decided by the final application**.
* The theme is distributed via an NPM package, so it is **easily upgradable**.
* The final application can **customize** the selected theme.
In order to accomplish these goals, ABP Framework;
* Determines a set of **base libraries** used and adapted by all the themes. So, module and application developers can depend on and use these libraries without depending on a particular theme.
* Provides a system that consists of layout parts (like navigation menus and toolbars) that is implemented by all the themes. So, the modules and the application contribute to the layout to compose a consistent application UI.
### Current Themes
Currently, two themes are **officially provided**:
* The [Basic Theme](Basic-Theme.md) is the minimalist theme with the plain Bootstrap style. It is **open source and free**.
* The [Lepton Theme](https://commercial.abp.io/themes) is a **commercial** theme developed by the core ABP team and is a part of the [ABP Commercial](https://commercial.abp.io/) license.
## Overall
### The Base Libraries
All the themes must depend on the [@abp/ng.theme.shared](https://www.npmjs.com/package/@abp/ng.theme.shared) NuGet package, so they are indirectly depending on the following libraries:
* [Twitter Bootstrap](https://getbootstrap.com/) as the fundamental HTML/CSS framework.
* [FontAwesome](https://fontawesome.com/) as the fundamental CSS font library.
* [NG Bootstrap](https://ng-bootstrap.github.io/#/home) as a component library that supports the Bootstrap and adds extra components like modal and datepicker.
* [Ngx-Datatable](https://swimlane.gitbook.io/ngx-datatable/) as a datatable library.
* [ngx-validate](https://github.com/ng-turkey/ngx-validate) a dynamic validation of reactive forms library.
* [Chart.js](https://www.chartjs.org/) as a widget library.
These libraries are selected as the base libraries and available to the applications and modules.
> Bootstrap's JavaScript part is not used since the NG Bootstrap library already provides the necessary functionalities to the Bootstrap components in a native way.
### The Layout
All themes must define a layout for the application. The following image shows the user management page in the [Basic Theme](Basic-Theme.md) application layout:
![basic-theme-application-layout](../../images/basic-theme-application-layout.png)
And the same page is shown below with the [Lepton Theme](https://commercial.abp.io/themes) application layout:
![lepton-theme-application-layout](../../images/lepton-theme-application-layout.png)
As you can see, the page is the same, but the look is completely different in the themes above.
The application layout typically includes the following parts;
* Main menu
* Nav items area with the following components;
* User menu
* Language switch dropdown
* [Page alerts](Page-Alerts.md)
* The page content (aka `<router-outlet>`)
## Implementing a Theme
A theme is simply an NPM package and comes with startup templates.
### The Easy Way
The easiest way to create a new theme is to add Basic Theme Source Code to your project via [ABP CLI](../../CLI.md) command and customize it.
You can run the following command in **Angular** project directory to copy the source code to your solution:
`abp add-package @abp/ng.theme.basic --with-source-code`
### Global/Component Styles
Angular can bundle global style files and component styles with components.
See the [component styles](https://angular.io/guide/component-styles) guide on Angular documentation for more information.
### Layout Parts
A typical layout consists of several parts. The theme should include the necessary parts in each layout.
**Example: The Basic Theme has the following parts for the Application Layout**
![basic-theme-application-layout-parts](images/basic-theme-application-layout-parts.png)
The application code and the modules can only show contents in the Page Content part. If they need to change the other parts (to add a menu item, to add a nav item, to change the application name in the logo area...) they should use the ABP Framework APIs.
The following sections explain the fundamental parts pre-defined by the ABP Framework and can be implemented by the themes.
> It is a good practice to split the layout into components/partials, so the final application can override them partially for customization purpose.
#### Logo
The `application` object of an environment file should be configured to get the name and the logo URL of the application to render in the logo part. Additionally, `LogoComponent` can be replaced. See [Component Replacement](Component-Replacement.md) document for more.
The [Application Startup Template](../../Startup-Templates/Application.md) has an implementation of this interface to set the values by the application developer.
#### Main Menu / Routes
`RoutesService` service is used to manage the main menu items and render them on the layout.
**Example: Adding a route to the main menu**
```ts
import { RoutesService, eLayoutType } from '@abp/ng.core';
import { Component } from '@angular/core';
@Component(/* component metadata */)
export class AppComponent {
constructor(routes: RoutesService) {
routes.add([
{
path: '/your-path',
name: 'Your navigation',
order: 101,
iconClass: 'fas fa-question-circle',
requiredPolicy: 'permission key here',
layout: eLayoutType.application,
},
{
path: '/your-path/child',
name: 'Your child navigation',
parentName: 'Your navigation',
order: 1,
requiredPolicy: 'permission key here',
},
]);
}
}
```
See the [Modifying the Menu](Modifying-the-Menu.md) document to learn more about the navigation system.
#### Toolbar / Nav Items
`NavItemsService` service is used to get the menu's right part items and render on the layout. You can add an HTML content or Angular component as an element to render.
**Example: Adding an element to right part of the menu**
````ts
import { NavItemsService } from '@abp/ng.theme.shared';
import { Component } from '@angular/core';
@Component({
template: `
<input type="search" placeholder="Search" class="bg-transparent border-0 color-white" />
`,
})
export class MySearchInputComponent {}
@Component(/* component metadata */)
export class AppComponent {
constructor(private navItems: NavItemsService) {
navItems.addItems([
{
id: 'MySearchInput',
order: 1,
component: MySearchInputComponent,
},
{
id: 'SignOutIcon',
html: '<i class="fas fa-sign-out-alt fa-lg text-white m-2"><i>',
action: () => console.log('Clicked the sign out icon'),
order: 101, // puts as last element
},
]);
}
}
````
> See the [How to Add an Element to Right Part of the Menu](Modifying-the-Menu#how-to-add-an-element-to-right-part-of-the-menu) document to learn more on the nav items system.
The theme has a responsibility to add two pre-defined items to the toolbar: Language Selection and User Menu.
##### Language Selection
Language Selection toolbar item is generally a dropdown that is used to switch between languages. `ConfigStateService` is used to get the list of available languages and `SessionStateService` is used to learn the current language.
`SessionStateService` is used to get and set the current language.
**Example: Get the currently selected language**
````ts
import {SessionStateService} from '@abp/ng.core';
//...
constructor(private sessionState: SessionStateService) {
const lang = this.sessionState.getLanguage()
}
````
**Example: Set the selected language**
````ts
import {SessionStateService} from '@abp/ng.core';
//...
constructor(private sessionState: SessionStateService) {
const lang = this.sessionState.setLanguage('en')
}
````
##### User Menu
User menu is a component that can be replaceable. See an example to learn how can you replace it:
````ts
import { eThemeBasicComponents } from '@abp/ng.theme.basic';
import { NavItemsService } from '@abp/ng.theme.shared';
import { Component } from '@angular/core';
@Component({/* component metadata */})
export class AppComponent {
constructor(private navItems: NavItemsService) {
this.navItems.patchItem(eThemeBasicComponents.CurrentUser, { component: MyUserMenuComponent });
}
}
````
[`ConfigStateService`](Config-State-Service.md) service can be used to obtain the `application-configuration` API response (e.g. getting current user or tenant).
#### Page Alerts
`PageAlertService` service is used to get the current page alerts to render on the layout. See the [Page Alerts](Page-Alerts.md) document to learn more.

Binary file not shown.

After

Width:  |  Height:  |  Size: 45 KiB

@ -15,7 +15,7 @@ The Basic Theme has RTL (Right-to-Left language) support.
* Install the [Volo.Abp.AspNetCore.Mvc.UI.Theme.Basic](https://www.nuget.org/packages/Volo.Abp.AspNetCore.Mvc.UI.Theme.Basic) NuGet package to your web project.
* Add `AbpAspNetCoreMvcUiBasicThemeModule` into the `[DependsOn(...)]` attribute for your [module class](../../Module-Development-Basics.md) in the web project.
* Install the [@abp/aspnetcore.mvc.ui.theme.basic](https://www.npmjs.com/package/@abp/aspnetcore.mvc.ui.theme.basic) NPM package to your web project (e.g. `npm install @abp/aspnetcore.mvc.ui.theme.basic` or `yarn add @abp/aspnetcore.mvc.ui.theme.basic`).
* Run `gulp` command in a command line terminal in the web project's folder.
* Run `abp install-libs` command in a command line terminal in the web project's folder.
## Layouts
@ -85,7 +85,7 @@ See the [User Interface Customization Guide](Customization-User-Interface.md) to
You can run the following [ABP CLI](../../CLI.md) command in **Web** project directory to copy the source code to your solution:
`abp add-package Volo.Abp.AspNetCore.Components.Server.BasicTheme --with-source-code --add-to-solution-file`
`abp add-module Volo.BasicTheme --with-source-code --add-to-solution-file`
----

@ -18,7 +18,7 @@ To solve the versioning problem, we created a **standard set of packages** those
The benefit of a **standard package** is:
* It depends on a **standard version** of a package. Depending on this package is **safe** because all modules depend on the same version.
* It contains the gulp task to copy library resources (js, css, img... files) from the **node_modules** folder to **wwwroot/libs** folder. See the *Mapping The Library Resources* section for more.
* It contains the necessary mapping configuration to copy library resources (js, css, img... files) from the **node_modules** folder to **wwwroot/libs** folder. See the *Mapping The Library Resources* section for more.
Depending on a standard package is easy. Just add it to your **package.json** file like you normally do. Example:
@ -61,7 +61,13 @@ Using NPM packages and NPM/Yarn tool is the de facto standard for client side li
Next challenge is copying needed resources (js, css, img... files) from the `node_modules` into a folder inside the **wwwroot** folder to make it accessible to the clients/browsers.
ABP defines a [Gulp](https://gulpjs.com/) based task to **copy resources** from **node_modules** to **wwwroot/libs** folder. Each **standard package** (see the *@ABP NPM Packages* section) defines the mapping for its own files. So, most of the time, you only configure dependencies.
[ABP CLI](../../CLI.md) offers the command below to **copy resources** from **node_modules** to **wwwroot/libs** folder. You have to run it in the root folder of your web project:
````bash
abp install-libs
````
Each **standard package** (see the *@ABP NPM Packages* section) defines the mapping for its own files. So, most of the time, you only configure dependencies.
The **startup templates** are already configured to work all these out of the box. This section will explain the configuration options.
@ -100,17 +106,17 @@ mappings: {
}
````
#### Using The Gulp
#### Using ABP CLI To Copy Resources
Once you properly configure the `abp.resourcemapping.js` file, you can run the gulp command from the command line:
Once you properly configure the `abp.resourcemapping.js` file, you can run the [ABP CLI](../../CLI.md) command from the command line:
````
gulp
````bash
abp install-libs
````
When you run the `gulp`, all packages will copy their own resources into the **wwwroot/libs** folder. Running `yarn & gulp` is only necessary if you make a change in your dependencies in the **package.json** file.
When you run this command, all packages will copy their own resources into the **wwwroot/libs** folder. Running `yarn` & `abp install-libs` are only necessary if you make a change in your dependencies in the **package.json** file.
> When you run the Gulp command, dependencies of the application are resolved using the package.json file. The Gulp task automatically discovers and maps all resources from all dependencies (recursively).
> When you run the `abp install-libs` command, dependencies of the application are resolved using the package.json file. [ABP CLI](../../CLI.md) automatically discovers and maps all resources from all dependencies (recursively).
#### See Also

@ -38,7 +38,7 @@ You have two options two customize this theme:
### Overriding Styles / Components
In this approach, you continue to use the the theme as NuGet and NPM packages and customize the parts you need to. There are several ways to customize it;
In this approach, you continue to use the theme as NuGet and NPM packages and customize the parts you need to. There are several ways to customize it;
#### Override the Styles

@ -149,8 +149,8 @@ Once ABP determines a validation error, it throws an exception of type `AbpValid
In addition to the automatic validation, you may want to manually validate an object. In this case, [inject](Dependency-Injection.md) and use the `IObjectValidator` service:
* `Validate` method validates the given object based on the validation rules and throws an `AbpValidationException` if it is not in a valid state.
* `GetErrors` doesn't throw an exception, but only returns the validation errors.
* `ValidateAsync` method validates the given object based on the validation rules and throws an `AbpValidationException` if it is not in a valid state.
* `GetErrorsAsync` doesn't throw an exception, but only returns the validation errors.
`IObjectValidator` is implemented by the `ObjectValidator` by default. `ObjectValidator` is extensible; you can implement `IObjectValidationContributor` interface to contribute a custom logic. Example:
@ -158,13 +158,14 @@ In addition to the automatic validation, you may want to manually validate an ob
public class MyObjectValidationContributor
: IObjectValidationContributor, ITransientDependency
{
public void AddErrors(ObjectValidationContext context)
public Task AddErrorsAsync(ObjectValidationContext context)
{
//Get the validating object
var obj = context.ValidatingObject;
//Add the validation errors if available
context.Errors.Add(...);
return Task.CompletedTask;
}
}
````

@ -38,7 +38,7 @@ This configuration recursively adds all files under the **MyResources** folder o
Embedding a file in the project/assembly may cause problems if a file name contains some special chars. To overcome this limitation;
1. Add [Microsoft.Extensions.FileProviders.Embedded](https://www.nuget.org/packages/Microsoft.Extensions.FileProviders.Embedded) NuGet package to the project that contains the embedded resource(s).
2. Add `<GenerateEmbeddedFilesManifest>true</GenerateEmbeddedFilesManifest>` into the `<PropertyConfig>...</PropertyConfig>` section of your `.csproj` file.
2. Add `<GenerateEmbeddedFilesManifest>true</GenerateEmbeddedFilesManifest>` into the `<PropertyGroup>...</PropertyGroup>` section of your `.csproj` file.
> While these two steps are optional and ABP can work without these configuration, it is strongly suggested to make it.
@ -193,4 +193,4 @@ So, if you need to replace a file of a module, just create the file in the exact
### Physical Files
Physical files always override the virtual files. That means if you put a file under the `/wwwroot/my-folder/my-file.css`, it will override the file in the same location of the virtual file system. So, you need to know the file paths defined in the modules to override them.
Physical files always override the virtual files. That means if you put a file under the `/wwwroot/my-folder/my-file.css`, it will override the file in the same location of the virtual file system. So, you need to know the file paths defined in the modules to override them.

@ -365,8 +365,8 @@
"path": "Object-To-Object-Mapping.md"
},
{
"text":"String Encryption",
"path":"String-Encryption.md"
"text": "String Encryption",
"path": "String-Encryption.md"
},
{
"text": "Text Templating",
@ -578,6 +578,10 @@
"path": "API/Application-Configuration.md"
}
]
},
{
"text": "Swagger Integration",
"path": "API/Swagger-Integration.md"
}
]
},
@ -1025,6 +1029,19 @@
"text": "Customization Guide",
"path": "UI/Angular/Customization-User-Interface.md"
},
{
"text": "Theming",
"items": [
{
"text": "Overall",
"path": "UI/Angular/Theming.md"
},
{
"text": "The Basic Theme",
"path": "UI/Angular/Basic-Theme.md"
}
]
},
{
"text": "Modifying the Menu",
"path": "UI/Angular/Modifying-the-Menu.md"

Binary file not shown.

After

Width:  |  Height:  |  Size: 18 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 12 KiB

@ -211,7 +211,7 @@ public class OrderLine : Entity
> 如果你不想你的聚合根继承`AggregateRoot<TKey>`类,你可以直接实现`IAggregateRoot<TKey>`接口
`Order`是一个具有`Guid`类型`Id`属性的 **聚合根**.它有一个`OrderLine`实体集合.`OrderLine`是一个具有组合键(`OrderLine`和 ` ProductId`)的实体.
`Order`是一个具有`Guid`类型`Id`属性的 **聚合根**.它有一个`OrderLine`实体集合.`OrderLine`是一个具有组合键(`OrderId`和 ` ProductId`)的实体.
虽然这个示例可能无法实现聚合根的所有最佳实践,但它仍然遵循良好的实践:

@ -130,8 +130,8 @@ namespace Acme.BookStore
除了自动验证你可能需要手动验证对象,这种情况下[注入](Dependency-Injection.md)并使用 `IObjectValidator` 服务:
* `Validate` 方法根据验证​​规则验证给定对象,如果对象没有被验证通过会抛出 `AbpValidationException` 异常.
* `GetErrors` 不会抛出异常,只返回验证错误.
* `ValidateAsync` 方法根据验证​​规则验证给定对象,如果对象没有被验证通过会抛出 `AbpValidationException` 异常.
* `GetErrorsAsync` 不会抛出异常,只返回验证错误.
`IObjectValidator` 默认由 `ObjectValidator` 实现. `ObjectValidator`是可扩展的; 可以实现`IObjectValidationContributor`接口提供自定义逻辑.
示例 :
@ -140,13 +140,14 @@ namespace Acme.BookStore
public class MyObjectValidationContributor
: IObjectValidationContributor, ITransientDependency
{
public void AddErrors(ObjectValidationContext context)
public Task AddErrorsAsync(ObjectValidationContext context)
{
//Get the validating object
var obj = context.ValidatingObject;
//Add the validation errors if available
context.Errors.Add(...);
return Task.CompletedTask;
}
}
````

@ -383,6 +383,8 @@ Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Volo.Abp.TextTemplating.Scr
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Volo.Abp.MongoDB.Tests.SecondContext", "test\Volo.Abp.MongoDB.Tests.SecondContext\Volo.Abp.MongoDB.Tests.SecondContext.csproj", "{90B1866A-EF99-40B9-970E-B898E5AA523F}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Volo.Abp.Threading.Tests", "test\Volo.Abp.Threading.Tests\Volo.Abp.Threading.Tests.csproj", "{7B2FCAD6-86E6-49C8-ADBE-A61B4F4B101B}"
EndProject
Global
GlobalSection(SolutionConfigurationPlatforms) = preSolution
Debug|Any CPU = Debug|Any CPU
@ -1141,6 +1143,10 @@ Global
{90B1866A-EF99-40B9-970E-B898E5AA523F}.Debug|Any CPU.Build.0 = Debug|Any CPU
{90B1866A-EF99-40B9-970E-B898E5AA523F}.Release|Any CPU.ActiveCfg = Release|Any CPU
{90B1866A-EF99-40B9-970E-B898E5AA523F}.Release|Any CPU.Build.0 = Release|Any CPU
{7B2FCAD6-86E6-49C8-ADBE-A61B4F4B101B}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{7B2FCAD6-86E6-49C8-ADBE-A61B4F4B101B}.Debug|Any CPU.Build.0 = Debug|Any CPU
{7B2FCAD6-86E6-49C8-ADBE-A61B4F4B101B}.Release|Any CPU.ActiveCfg = Release|Any CPU
{7B2FCAD6-86E6-49C8-ADBE-A61B4F4B101B}.Release|Any CPU.Build.0 = Release|Any CPU
EndGlobalSection
GlobalSection(SolutionProperties) = preSolution
HideSolutionNode = FALSE
@ -1334,6 +1340,7 @@ Global
{C996F458-98FB-483D-9306-4701290E2FC1} = {447C8A77-E5F0-4538-8687-7383196D04EA}
{75D8DADB-3FA9-4C1D-B23A-DBFD08133B7C} = {447C8A77-E5F0-4538-8687-7383196D04EA}
{90B1866A-EF99-40B9-970E-B898E5AA523F} = {447C8A77-E5F0-4538-8687-7383196D04EA}
{7B2FCAD6-86E6-49C8-ADBE-A61B4F4B101B} = {447C8A77-E5F0-4538-8687-7383196D04EA}
EndGlobalSection
GlobalSection(ExtensibilityGlobals) = postSolution
SolutionGuid = {BB97ECF4-9A84-433F-A80B-2A3285BDD1D5}

@ -25,18 +25,19 @@ namespace Volo.Abp.AspNetCore.Components.Server
{
public override void ConfigureServices(ServiceConfigurationContext context)
{
context.Services.AddServerSideBlazor();
var serverSideBlazorBuilder = context.Services.AddServerSideBlazor();
context.Services.ExecutePreConfiguredActions(serverSideBlazorBuilder);
Configure<AbpAspNetCoreUnitOfWorkOptions>(options =>
{
options.IgnoredUrls.AddIfNotContains("/_blazor");
});
Configure<AbpAspNetCoreAuditingOptions>(options =>
{
options.IgnoredUrls.AddIfNotContains("/_blazor");
});
Configure<AbpEndpointRouterOptions>(options =>
{
options.EndpointConfigureActions.Add(endpointContext =>
@ -56,4 +57,4 @@ namespace Volo.Abp.AspNetCore.Components.Server
);
}
}
}
}

@ -64,6 +64,7 @@ namespace Volo.Abp.AspNetCore.MultiTenancy
context,
requestCulture
);
context.Items[AbpRequestLocalizationMiddleware.HttpContextItemName] = true;
}
await next(context);

@ -0,0 +1,13 @@
{
"culture": "ro-RO",
"texts": {
"GivenTenantIsNotExist": "Tenantul {0} nu există",
"GivenTenantIsNotAvailable": "Tenantul {0} nu este disponibil",
"Tenant": "Tenant",
"Switch": "schimbă",
"Name": "Nume",
"SwitchTenantHint": "Lăsaţi câmpul de nume gol pentru a schimba pe host.",
"SwitchTenant": "Schimbă tenant",
"NotSelected": "Neselectat"
}
}

@ -33,6 +33,8 @@ namespace Volo.Abp.AspNetCore.Mvc.Localization
new RequestCulture(culture, uiCulture)
);
HttpContext.Items[AbpRequestLocalizationMiddleware.HttpContextItemName] = true;
var context = new QueryStringCultureReplacementContext(HttpContext, new RequestCulture(culture, uiCulture), returnUrl);
await QueryStringCultureReplacement.ReplaceAsync(context);

@ -10,6 +10,8 @@ namespace Microsoft.AspNetCore.RequestLocalization
{
public class AbpRequestLocalizationMiddleware : IMiddleware, ITransientDependency
{
public const string HttpContextItemName = "__AbpSetCultureCookie";
private readonly IAbpRequestLocalizationOptionsProvider _requestLocalizationOptionsProvider;
private readonly ILoggerFactory _loggerFactory;
@ -29,6 +31,23 @@ namespace Microsoft.AspNetCore.RequestLocalization
_loggerFactory
);
context.Response.OnStarting(() =>
{
if (context.Items[HttpContextItemName] == null)
{
var requestCultureFeature = context.Features.Get<IRequestCultureFeature>();
if (requestCultureFeature?.Provider is QueryStringRequestCultureProvider)
{
AbpRequestCultureCookieHelper.SetCultureCookie(
context,
requestCultureFeature.RequestCulture
);
}
}
return Task.CompletedTask;
});
await middleware.Invoke(context);
}
}

@ -6,13 +6,26 @@ using Volo.Abp.Threading;
namespace Volo.Abp.AspNetCore.Threading
{
[Dependency(ReplaceServices = true)]
public class HttpContextCancellationTokenProvider : ICancellationTokenProvider, ITransientDependency
public class HttpContextCancellationTokenProvider : CancellationTokenProviderBase, ITransientDependency
{
public CancellationToken Token => _httpContextAccessor.HttpContext?.RequestAborted ?? CancellationToken.None;
private readonly IHttpContextAccessor _httpContextAccessor;
public HttpContextCancellationTokenProvider(IHttpContextAccessor httpContextAccessor)
public override CancellationToken Token
{
get
{
if (OverrideValue != null)
{
return OverrideValue.CancellationToken;
}
return _httpContextAccessor.HttpContext?.RequestAborted ?? CancellationToken.None;
}
}
public HttpContextCancellationTokenProvider(
IAmbientScopeProvider<CancellationTokenOverride> cancellationTokenOverrideScopeProvider,
IHttpContextAccessor httpContextAccessor)
: base(cancellationTokenOverrideScopeProvider)
{
_httpContextAccessor = httpContextAccessor;
}

@ -0,0 +1,10 @@
{
"culture": "ro-RO",
"texts": {
"Volo.Authorization:010001": "Autorizare eşuată! Politica dată nu a fost acordată.",
"Volo.Authorization:010002": "Autorizare eşuată! Politica dată nu a acordat: {PolicyName}",
"Volo.Authorization:010003": "Autorizare eşuată! Politica dată nu a fost acordată pentru resursa dată: {ResourceName}",
"Volo.Authorization:010004": "Autorizare eşuată! Cerinţa dată nu a fost acordată pentru resursa dată: {ResourceName}",
"Volo.Authorization:010005": "Autorizare eşuată! Cerinţele date nu au fost acordate pentru resursa dată: {ResourceName}"
}
}

@ -14,10 +14,10 @@
</ItemGroup>
<ItemGroup>
<PackageReference Include="Blazorise" Version="0.9.3.6" />
<PackageReference Include="Blazorise.DataGrid" Version="0.9.3.6" />
<PackageReference Include="Blazorise.Snackbar" Version="0.9.3.6" />
<PackageReference Include="Blazorise.Components" Version="0.9.3.6" />
<PackageReference Include="Blazorise" Version="0.9.4.1" />
<PackageReference Include="Blazorise.DataGrid" Version="0.9.4.1" />
<PackageReference Include="Blazorise.Snackbar" Version="0.9.4.1" />
<PackageReference Include="Blazorise.Components" Version="0.9.4.1" />
</ItemGroup>
</Project>

@ -10,9 +10,9 @@
public const string NuGetRootPath = NuGetRootPathDevelopment;
#else
public const string WwwAbpIo = WwwAbpIoProduction;
public const string AccountAbpIo = AccountAbpIoProduction;
public const string NuGetRootPath = NuGetRootPathProduction;
#endif

@ -70,9 +70,10 @@ namespace Volo.Abp.Cli.Commands
private string FindTenantDbContextName(string dbMigrationsFolder)
{
var tenantDbContext = Directory
.GetFiles(dbMigrationsFolder, "*TenantMigrationsDbContext.cs", SearchOption.AllDirectories)
.FirstOrDefault();
var tenantDbContext = Directory.GetFiles(dbMigrationsFolder, "*TenantMigrationsDbContext.cs", SearchOption.AllDirectories)
.FirstOrDefault() ??
Directory.GetFiles(dbMigrationsFolder, "*TenantDbContext.cs", SearchOption.AllDirectories)
.FirstOrDefault();
if (tenantDbContext == null)
{
@ -84,9 +85,10 @@ namespace Volo.Abp.Cli.Commands
private string FindDbContextName(string dbMigrationsFolder)
{
var dbContext = Directory
.GetFiles(dbMigrationsFolder, "*MigrationsDbContext.cs", SearchOption.AllDirectories)
.FirstOrDefault(fp => !fp.EndsWith("TenantMigrationsDbContext.cs"));
var dbContext = Directory.GetFiles(dbMigrationsFolder, "*MigrationsDbContext.cs", SearchOption.AllDirectories)
.FirstOrDefault(fp => !fp.EndsWith("TenantMigrationsDbContext.cs")) ??
Directory.GetFiles(dbMigrationsFolder, "*DbContext.cs", SearchOption.AllDirectories)
.FirstOrDefault(fp => !fp.EndsWith("TenantDbContext.cs"));
if (dbContext == null)
{

@ -26,24 +26,20 @@ using Volo.Abp.Threading;
namespace Volo.Abp.Cli.Commands
{
public class NewCommand : IConsoleCommand, ITransientDependency
public class NewCommand : ProjectCreationCommandBase, IConsoleCommand, ITransientDependency
{
private readonly EfCoreMigrationManager _efCoreMigrationManager;
public ILogger<NewCommand> Logger { get; set; }
protected TemplateProjectBuilder TemplateProjectBuilder { get; }
public ITemplateInfoProvider TemplateInfoProvider { get; }
public ConnectionStringProvider ConnectionStringProvider { get; }
public NewCommand(TemplateProjectBuilder templateProjectBuilder
, ITemplateInfoProvider templateInfoProvider,
EfCoreMigrationManager efCoreMigrationManager,
ConnectionStringProvider connectionStringProvider)
: base(connectionStringProvider)
{
_efCoreMigrationManager = efCoreMigrationManager;
TemplateProjectBuilder = templateProjectBuilder;
TemplateInfoProvider = templateInfoProvider;
ConnectionStringProvider = connectionStringProvider;
Logger = NullLogger<NewCommand>.Instance;
}
@ -79,257 +75,23 @@ namespace Volo.Abp.Cli.Commands
template = (await TemplateInfoProvider.GetDefaultAsync()).Name;
}
var version = commandLineArgs.Options.GetOrNull(Options.Version.Short, Options.Version.Long);
if (version != null)
{
Logger.LogInformation("Version: " + version);
}
var isTiered = commandLineArgs.Options.ContainsKey(Options.Tiered.Long);
if (isTiered)
{
Logger.LogInformation("Tiered: yes");
}
var preview = commandLineArgs.Options.ContainsKey(Options.Preview.Long);
if (preview)
{
Logger.LogInformation("Preview: yes");
}
var databaseProvider = GetDatabaseProvider(commandLineArgs);
if (databaseProvider != DatabaseProvider.NotSpecified)
{
Logger.LogInformation("Database provider: " + databaseProvider);
}
var connectionString = GetConnectionString(commandLineArgs);
if (connectionString != null)
{
Logger.LogInformation("Connection string: " + connectionString);
}
var databaseManagementSystem = GetDatabaseManagementSystem(commandLineArgs);
if (databaseManagementSystem != DatabaseManagementSystem.NotSpecified)
{
Logger.LogInformation("DBMS: " + databaseManagementSystem);
}
var uiFramework = GetUiFramework(commandLineArgs);
if (uiFramework != UiFramework.NotSpecified)
{
Logger.LogInformation("UI Framework: " + uiFramework);
}
var publicWebSite = uiFramework != UiFramework.None && commandLineArgs.Options.ContainsKey(Options.PublicWebSite.Long);
if (publicWebSite)
{
Logger.LogInformation("Public Web Site: yes");
}
var mobileApp = GetMobilePreference(commandLineArgs);
if (mobileApp != MobileApp.None)
{
Logger.LogInformation("Mobile App: " + mobileApp);
}
var gitHubAbpLocalRepositoryPath = commandLineArgs.Options.GetOrNull(Options.GitHubAbpLocalRepositoryPath.Long);
if (gitHubAbpLocalRepositoryPath != null)
{
Logger.LogInformation("GitHub Abp Local Repository Path: " + gitHubAbpLocalRepositoryPath);
}
var gitHubVoloLocalRepositoryPath = commandLineArgs.Options.GetOrNull(Options.GitHubVoloLocalRepositoryPath.Long);
if (gitHubVoloLocalRepositoryPath != null)
{
Logger.LogInformation("GitHub Volo Local Repository Path: " + gitHubVoloLocalRepositoryPath);
}
var templateSource = commandLineArgs.Options.GetOrNull(Options.TemplateSource.Short, Options.TemplateSource.Long);
if (templateSource != null)
{
Logger.LogInformation("Template Source: " + templateSource);
}
var createSolutionFolder = GetCreateSolutionFolderPreference(commandLineArgs);
var outputFolder = commandLineArgs.Options.GetOrNull(Options.OutputFolder.Short, Options.OutputFolder.Long);
var outputFolderRoot =
outputFolder != null ? Path.GetFullPath(outputFolder) : Directory.GetCurrentDirectory();
SolutionName solutionName;
if (MicroserviceServiceTemplateBase.IsMicroserviceServiceTemplate(template))
{
var microserviceSolutionName = FindMicroserviceSolutionName(outputFolderRoot);
if (microserviceSolutionName == null)
{
throw new CliUsageException("This command should be run inside a folder that contains a microservice solution!");
}
solutionName = SolutionName.Parse(microserviceSolutionName, projectName);
outputFolder = MicroserviceServiceTemplateBase.CalculateTargetFolder(outputFolderRoot, projectName);
uiFramework = uiFramework == UiFramework.NotSpecified ? FindMicroserviceSolutionUiFramework(outputFolderRoot) : uiFramework;
}
else
{
solutionName = SolutionName.Parse(projectName);
outputFolder = createSolutionFolder ?
Path.Combine(outputFolderRoot, SolutionName.Parse(projectName).FullName) :
outputFolderRoot;
}
Volo.Abp.IO.DirectoryHelper.CreateIfNotExists(outputFolder);
Logger.LogInformation("Output folder: " + outputFolder);
if (connectionString == null &&
databaseManagementSystem != DatabaseManagementSystem.NotSpecified &&
databaseManagementSystem != DatabaseManagementSystem.SQLServer)
{
connectionString = ConnectionStringProvider.GetByDbms(databaseManagementSystem, outputFolder);
}
commandLineArgs.Options.Add(CliConsts.Command, commandLineArgs.Command);
var projectArgs = GetProjectBuildArgs(commandLineArgs, template, projectName);
var result = await TemplateProjectBuilder.BuildAsync(
new ProjectBuildArgs(
solutionName,
template,
version,
databaseProvider,
databaseManagementSystem,
uiFramework,
mobileApp,
publicWebSite,
gitHubAbpLocalRepositoryPath,
gitHubVoloLocalRepositoryPath,
templateSource,
commandLineArgs.Options,
connectionString
)
projectArgs
);
using (var templateFileStream = new MemoryStream(result.ZipContent))
{
using (var zipInputStream = new ZipInputStream(templateFileStream))
{
var zipEntry = zipInputStream.GetNextEntry();
while (zipEntry != null)
{
if (string.IsNullOrWhiteSpace(zipEntry.Name))
{
zipEntry = zipInputStream.GetNextEntry();
continue;
}
var fullZipToPath = Path.Combine(outputFolder, zipEntry.Name);
var directoryName = Path.GetDirectoryName(fullZipToPath);
if (!string.IsNullOrEmpty(directoryName))
{
Directory.CreateDirectory(directoryName);
}
var fileName = Path.GetFileName(fullZipToPath);
if (fileName.Length == 0)
{
zipEntry = zipInputStream.GetNextEntry();
continue;
}
var buffer = new byte[4096]; // 4K is optimum
using (var streamWriter = File.Create(fullZipToPath))
{
StreamUtils.Copy(zipInputStream, streamWriter, buffer);
}
zipEntry = zipInputStream.GetNextEntry();
}
}
}
Logger.LogInformation($"'{projectName}' has been successfully created to '{outputFolder}'");
if (AppTemplateBase.IsAppTemplate(template))
{
var isCommercial = template == AppProTemplate.TemplateName;
OpenThanksPage(uiFramework, databaseProvider, isTiered || commandLineArgs.Options.ContainsKey("separate-identity-server"), isCommercial);
}
else if (MicroserviceTemplateBase.IsMicroserviceTemplate(template))
{
OpenMicroserviceDocumentPage();
}
}
private string FindMicroserviceSolutionName(string outputFolderRoot)
{
var slnFile = Directory.GetFiles(outputFolderRoot, "*.sln").FirstOrDefault();
if (slnFile == null)
{
return null;
}
return Path.GetFileName(slnFile).RemovePostFix(".sln");
}
private UiFramework FindMicroserviceSolutionUiFramework(string outputFolderRoot)
{
if (Directory.Exists(Path.Combine(outputFolderRoot, "apps", "blazor")))
{
return UiFramework.Blazor;
}
if (Directory.Exists(Path.Combine(outputFolderRoot, "apps", "web")))
{
return UiFramework.Mvc;
}
if (Directory.Exists(Path.Combine(outputFolderRoot, "apps", "angular")))
{
return UiFramework.Angular;
}
return UiFramework.None;
}
private void OpenThanksPage(UiFramework uiFramework, DatabaseProvider databaseProvider, bool tiered, bool commercial)
{
uiFramework = uiFramework == UiFramework.NotSpecified || uiFramework == UiFramework.None ? UiFramework.Mvc : uiFramework;
databaseProvider = databaseProvider == DatabaseProvider.NotSpecified ? DatabaseProvider.EntityFrameworkCore : databaseProvider;
var urlPrefix = commercial ? "commercial" : "www";
var tieredYesNo = tiered ? "yes" : "no";
var url = $"https://{urlPrefix}.abp.io/project-created-success?ui={uiFramework:g}&db={databaseProvider:g}&tiered={tieredYesNo}";
CmdHelper.OpenWebPage(url);
}
private void OpenMicroserviceDocumentPage()
{
var url = "https://docs.abp.io/en/commercial/latest/startup-templates/microservice/index";
ExtractProjectZip(result, projectArgs.OutputFolder);
CmdHelper.OpenWebPage(url);
}
private bool GetCreateSolutionFolderPreference(CommandLineArgs commandLineArgs)
{
var longKey = commandLineArgs.Options.ContainsKey(Options.CreateSolutionFolder.Long);
if (longKey == false)
{
return commandLineArgs.Options.ContainsKey(Options.CreateSolutionFolder.Short);
}
return longKey;
}
Logger.LogInformation($"'{projectName}' has been successfully created to '{projectArgs.OutputFolder}'");
private static string GetConnectionString(CommandLineArgs commandLineArgs)
{
var connectionString = commandLineArgs.Options.GetOrNull(Options.ConnectionString.Short, Options.ConnectionString.Long);
return string.IsNullOrWhiteSpace(connectionString) ? null : connectionString;
OpenRelatedWebPage(projectArgs, template, isTiered, commandLineArgs);
}
public string GetUsageInfo()
@ -389,174 +151,5 @@ namespace Volo.Abp.Cli.Commands
return "Generate a new solution based on the ABP startup templates.";
}
protected virtual DatabaseProvider GetDatabaseProvider(CommandLineArgs commandLineArgs)
{
var optionValue = commandLineArgs.Options.GetOrNull(Options.DatabaseProvider.Short, Options.DatabaseProvider.Long);
switch (optionValue)
{
case "ef":
return DatabaseProvider.EntityFrameworkCore;
case "mongodb":
return DatabaseProvider.MongoDb;
default:
return DatabaseProvider.NotSpecified;
}
}
protected virtual DatabaseManagementSystem GetDatabaseManagementSystem(CommandLineArgs commandLineArgs)
{
var optionValue = commandLineArgs.Options.GetOrNull(Options.DatabaseManagementSystem.Short, Options.DatabaseManagementSystem.Long);
if (optionValue == null)
{
return DatabaseManagementSystem.NotSpecified;
}
switch (optionValue.ToLowerInvariant())
{
case "sqlserver":
return DatabaseManagementSystem.SQLServer;
case "mysql":
return DatabaseManagementSystem.MySQL;
case "postgresql":
return DatabaseManagementSystem.PostgreSQL;
case "oracle-devart":
return DatabaseManagementSystem.OracleDevart;
case "sqlite":
return DatabaseManagementSystem.SQLite;
case "oracle":
return DatabaseManagementSystem.Oracle;
default:
return DatabaseManagementSystem.NotSpecified;
}
}
protected virtual UiFramework GetUiFramework(CommandLineArgs commandLineArgs)
{
if (commandLineArgs.Options.ContainsKey("no-ui"))
{
return UiFramework.None;
}
var optionValue = commandLineArgs.Options.GetOrNull(Options.UiFramework.Short, Options.UiFramework.Long);
switch (optionValue)
{
case "none":
return UiFramework.None;
case "mvc":
return UiFramework.Mvc;
case "angular":
return UiFramework.Angular;
case "blazor":
return UiFramework.Blazor;
case "blazor-server":
return UiFramework.BlazorServer;
default:
return UiFramework.NotSpecified;
}
}
protected virtual MobileApp GetMobilePreference(CommandLineArgs commandLineArgs)
{
var optionValue = commandLineArgs.Options.GetOrNull(Options.Mobile.Short, Options.Mobile.Long);
switch (optionValue)
{
case "none":
return MobileApp.None;
case "react-native":
return MobileApp.ReactNative;
default:
return MobileApp.None;
}
}
public static class Options
{
public static class Template
{
public const string Short = "t";
public const string Long = "template";
}
public static class DatabaseProvider
{
public const string Short = "d";
public const string Long = "database-provider";
}
public static class DatabaseManagementSystem
{
public const string Short = "dbms";
public const string Long = "database-management-system";
}
public static class OutputFolder
{
public const string Short = "o";
public const string Long = "output-folder";
}
public static class GitHubAbpLocalRepositoryPath
{
public const string Long = "abp-path";
}
public static class GitHubVoloLocalRepositoryPath
{
public const string Long = "volo-path";
}
public static class Version
{
public const string Short = "v";
public const string Long = "version";
}
public static class UiFramework
{
public const string Short = "u";
public const string Long = "ui";
}
public static class Mobile
{
public const string Short = "m";
public const string Long = "mobile";
}
public static class PublicWebSite
{
public const string Long = "with-public-website";
}
public static class TemplateSource
{
public const string Short = "ts";
public const string Long = "template-source";
}
public static class ConnectionString
{
public const string Short = "cs";
public const string Long = "connection-string";
}
public static class CreateSolutionFolder
{
public const string Short = "csf";
public const string Long = "create-solution-folder";
}
public static class Tiered
{
public const string Long = "tiered";
}
public static class Preview
{
public const string Long = "preview";
}
}
}
}
}

@ -0,0 +1,449 @@
using System;
using System.IO;
using System.Linq;
using ICSharpCode.SharpZipLib.Core;
using ICSharpCode.SharpZipLib.Zip;
using Microsoft.Extensions.Logging;
using Microsoft.Extensions.Logging.Abstractions;
using Volo.Abp.Cli.Args;
using Volo.Abp.Cli.Commands.Services;
using Volo.Abp.Cli.ProjectBuilding;
using Volo.Abp.Cli.ProjectBuilding.Building;
using Volo.Abp.Cli.ProjectBuilding.Templates.App;
using Volo.Abp.Cli.ProjectBuilding.Templates.Microservice;
using Volo.Abp.Cli.Utils;
namespace Volo.Abp.Cli.Commands
{
public abstract class ProjectCreationCommandBase
{
public ConnectionStringProvider ConnectionStringProvider { get; }
public ILogger<NewCommand> Logger { get; set; }
public ProjectCreationCommandBase(ConnectionStringProvider connectionStringProvider)
{
ConnectionStringProvider = connectionStringProvider;
Logger = NullLogger<NewCommand>.Instance;
}
protected ProjectBuildArgs GetProjectBuildArgs(CommandLineArgs commandLineArgs, string template, string projectName)
{
var version = commandLineArgs.Options.GetOrNull(Options.Version.Short, Options.Version.Long);
if (version != null)
{
Logger.LogInformation("Version: " + version);
}
var preview = commandLineArgs.Options.ContainsKey(Options.Preview.Long);
if (preview)
{
Logger.LogInformation("Preview: yes");
}
var databaseProvider = GetDatabaseProvider(commandLineArgs);
if (databaseProvider != DatabaseProvider.NotSpecified)
{
Logger.LogInformation("Database provider: " + databaseProvider);
}
var connectionString = GetConnectionString(commandLineArgs);
if (connectionString != null)
{
Logger.LogInformation("Connection string: " + connectionString);
}
var databaseManagementSystem = GetDatabaseManagementSystem(commandLineArgs);
if (databaseManagementSystem != DatabaseManagementSystem.NotSpecified)
{
Logger.LogInformation("DBMS: " + databaseManagementSystem);
}
var uiFramework = GetUiFramework(commandLineArgs);
if (uiFramework != UiFramework.NotSpecified)
{
Logger.LogInformation("UI Framework: " + uiFramework);
}
var publicWebSite = uiFramework != UiFramework.None && commandLineArgs.Options.ContainsKey(Options.PublicWebSite.Long);
if (publicWebSite)
{
Logger.LogInformation("Public Web Site: yes");
}
var mobileApp = GetMobilePreference(commandLineArgs);
if (mobileApp != MobileApp.None)
{
Logger.LogInformation("Mobile App: " + mobileApp);
}
var gitHubAbpLocalRepositoryPath = commandLineArgs.Options.GetOrNull(Options.GitHubAbpLocalRepositoryPath.Long);
if (gitHubAbpLocalRepositoryPath != null)
{
Logger.LogInformation("GitHub Abp Local Repository Path: " + gitHubAbpLocalRepositoryPath);
}
var gitHubVoloLocalRepositoryPath = commandLineArgs.Options.GetOrNull(Options.GitHubVoloLocalRepositoryPath.Long);
if (gitHubVoloLocalRepositoryPath != null)
{
Logger.LogInformation("GitHub Volo Local Repository Path: " + gitHubVoloLocalRepositoryPath);
}
var templateSource = commandLineArgs.Options.GetOrNull(Options.TemplateSource.Short, Options.TemplateSource.Long);
if (templateSource != null)
{
Logger.LogInformation("Template Source: " + templateSource);
}
var createSolutionFolder = GetCreateSolutionFolderPreference(commandLineArgs);
var outputFolder = commandLineArgs.Options.GetOrNull(Options.OutputFolder.Short, Options.OutputFolder.Long);
var outputFolderRoot =
outputFolder != null ? Path.GetFullPath(outputFolder) : Directory.GetCurrentDirectory();
SolutionName solutionName;
if (MicroserviceServiceTemplateBase.IsMicroserviceServiceTemplate(template))
{
var microserviceSolutionName = FindMicroserviceSolutionName(outputFolderRoot);
if (microserviceSolutionName == null)
{
throw new CliUsageException("This command should be run inside a folder that contains a microservice solution!");
}
solutionName = SolutionName.Parse(microserviceSolutionName, projectName);
outputFolder = MicroserviceServiceTemplateBase.CalculateTargetFolder(outputFolderRoot, projectName);
uiFramework = uiFramework == UiFramework.NotSpecified ? FindMicroserviceSolutionUiFramework(outputFolderRoot) : uiFramework;
}
else
{
solutionName = SolutionName.Parse(projectName);
outputFolder = createSolutionFolder ?
Path.Combine(outputFolderRoot, SolutionName.Parse(projectName).FullName) :
outputFolderRoot;
}
IO.DirectoryHelper.CreateIfNotExists(outputFolder);
Logger.LogInformation("Output folder: " + outputFolder);
if (connectionString == null &&
databaseManagementSystem != DatabaseManagementSystem.NotSpecified &&
databaseManagementSystem != DatabaseManagementSystem.SQLServer)
{
connectionString = ConnectionStringProvider.GetByDbms(databaseManagementSystem, outputFolder);
}
commandLineArgs.Options.Add(CliConsts.Command, commandLineArgs.Command);
return new ProjectBuildArgs(
solutionName,
template,
version,
outputFolder,
databaseProvider,
databaseManagementSystem,
uiFramework,
mobileApp,
publicWebSite,
gitHubAbpLocalRepositoryPath,
gitHubVoloLocalRepositoryPath,
templateSource,
commandLineArgs.Options,
connectionString
);
}
protected void ExtractProjectZip(ProjectBuildResult project, string outputFolder)
{
using (var templateFileStream = new MemoryStream(project.ZipContent))
{
using (var zipInputStream = new ZipInputStream(templateFileStream))
{
var zipEntry = zipInputStream.GetNextEntry();
while (zipEntry != null)
{
if (string.IsNullOrWhiteSpace(zipEntry.Name))
{
zipEntry = zipInputStream.GetNextEntry();
continue;
}
var fullZipToPath = Path.Combine(outputFolder, zipEntry.Name);
var directoryName = Path.GetDirectoryName(fullZipToPath);
if (!string.IsNullOrEmpty(directoryName))
{
Directory.CreateDirectory(directoryName);
}
var fileName = Path.GetFileName(fullZipToPath);
if (fileName.Length == 0)
{
zipEntry = zipInputStream.GetNextEntry();
continue;
}
var buffer = new byte[4096]; // 4K is optimum
using (var streamWriter = File.Create(fullZipToPath))
{
StreamUtils.Copy(zipInputStream, streamWriter, buffer);
}
zipEntry = zipInputStream.GetNextEntry();
}
}
}
}
protected void OpenRelatedWebPage(ProjectBuildArgs projectArgs,
string template,
bool isTiered,
CommandLineArgs commandLineArgs)
{
if (AppTemplateBase.IsAppTemplate(template))
{
var isCommercial = template == AppProTemplate.TemplateName;
OpenThanksPage(projectArgs.UiFramework, projectArgs.DatabaseProvider, isTiered || commandLineArgs.Options.ContainsKey("separate-identity-server"), isCommercial);
}
else if (MicroserviceTemplateBase.IsMicroserviceTemplate(template))
{
OpenMicroserviceDocumentPage();
}
}
protected string FindMicroserviceSolutionName(string outputFolderRoot)
{
var slnFile = Directory.GetFiles(outputFolderRoot, "*.sln").FirstOrDefault();
if (slnFile == null)
{
return null;
}
return Path.GetFileName(slnFile).RemovePostFix(".sln");
}
protected UiFramework FindMicroserviceSolutionUiFramework(string outputFolderRoot)
{
if (Directory.Exists(Path.Combine(outputFolderRoot, "apps", "blazor")))
{
return UiFramework.Blazor;
}
if (Directory.Exists(Path.Combine(outputFolderRoot, "apps", "web")))
{
return UiFramework.Mvc;
}
if (Directory.Exists(Path.Combine(outputFolderRoot, "apps", "angular")))
{
return UiFramework.Angular;
}
return UiFramework.None;
}
protected void OpenThanksPage(UiFramework uiFramework, DatabaseProvider databaseProvider, bool tiered, bool commercial)
{
uiFramework = uiFramework == UiFramework.NotSpecified || uiFramework == UiFramework.None ? UiFramework.Mvc : uiFramework;
databaseProvider = databaseProvider == DatabaseProvider.NotSpecified ? DatabaseProvider.EntityFrameworkCore : databaseProvider;
var urlPrefix = commercial ? "commercial" : "www";
var tieredYesNo = tiered ? "yes" : "no";
var url = $"https://{urlPrefix}.abp.io/project-created-success?ui={uiFramework:g}&db={databaseProvider:g}&tiered={tieredYesNo}";
CmdHelper.OpenWebPage(url);
}
protected void OpenMicroserviceDocumentPage()
{
var url = "https://docs.abp.io/en/commercial/latest/startup-templates/microservice/index";
CmdHelper.OpenWebPage(url);
}
protected bool GetCreateSolutionFolderPreference(CommandLineArgs commandLineArgs)
{
return commandLineArgs.Options.ContainsKey(Options.CreateSolutionFolder.Long)
|| commandLineArgs.Options.ContainsKey(Options.CreateSolutionFolder.Short);
}
protected static string GetConnectionString(CommandLineArgs commandLineArgs)
{
var connectionString = commandLineArgs.Options.GetOrNull(Options.ConnectionString.Short, Options.ConnectionString.Long);
return string.IsNullOrWhiteSpace(connectionString) ? null : connectionString;
}
protected virtual DatabaseProvider GetDatabaseProvider(CommandLineArgs commandLineArgs)
{
var optionValue = commandLineArgs.Options.GetOrNull(Options.DatabaseProvider.Short, Options.DatabaseProvider.Long);
switch (optionValue)
{
case "ef":
return DatabaseProvider.EntityFrameworkCore;
case "mongodb":
return DatabaseProvider.MongoDb;
default:
return DatabaseProvider.NotSpecified;
}
}
protected virtual DatabaseManagementSystem GetDatabaseManagementSystem(CommandLineArgs commandLineArgs)
{
var optionValue = commandLineArgs.Options.GetOrNull(Options.DatabaseManagementSystem.Short, Options.DatabaseManagementSystem.Long);
if (optionValue == null)
{
return DatabaseManagementSystem.NotSpecified;
}
switch (optionValue.ToLowerInvariant())
{
case "sqlserver":
return DatabaseManagementSystem.SQLServer;
case "mysql":
return DatabaseManagementSystem.MySQL;
case "postgresql":
return DatabaseManagementSystem.PostgreSQL;
case "oracle-devart":
return DatabaseManagementSystem.OracleDevart;
case "sqlite":
return DatabaseManagementSystem.SQLite;
case "oracle":
return DatabaseManagementSystem.Oracle;
default:
return DatabaseManagementSystem.NotSpecified;
}
}
protected virtual MobileApp GetMobilePreference(CommandLineArgs commandLineArgs)
{
var optionValue = commandLineArgs.Options.GetOrNull(Options.Mobile.Short, Options.Mobile.Long);
switch (optionValue)
{
case "none":
return MobileApp.None;
case "react-native":
return MobileApp.ReactNative;
default:
return MobileApp.None;
}
}
protected virtual UiFramework GetUiFramework(CommandLineArgs commandLineArgs)
{
if (commandLineArgs.Options.ContainsKey("no-ui"))
{
return UiFramework.None;
}
var optionValue = commandLineArgs.Options.GetOrNull(Options.UiFramework.Short, Options.UiFramework.Long);
switch (optionValue)
{
case "none":
return UiFramework.None;
case "mvc":
return UiFramework.Mvc;
case "angular":
return UiFramework.Angular;
case "blazor":
return UiFramework.Blazor;
case "blazor-server":
return UiFramework.BlazorServer;
default:
return UiFramework.NotSpecified;
}
}
public static class Options
{
public static class Template
{
public const string Short = "t";
public const string Long = "template";
}
public static class DatabaseProvider
{
public const string Short = "d";
public const string Long = "database-provider";
}
public static class DatabaseManagementSystem
{
public const string Short = "dbms";
public const string Long = "database-management-system";
}
public static class OutputFolder
{
public const string Short = "o";
public const string Long = "output-folder";
}
public static class GitHubAbpLocalRepositoryPath
{
public const string Long = "abp-path";
}
public static class GitHubVoloLocalRepositoryPath
{
public const string Long = "volo-path";
}
public static class Version
{
public const string Short = "v";
public const string Long = "version";
}
public static class UiFramework
{
public const string Short = "u";
public const string Long = "ui";
}
public static class Mobile
{
public const string Short = "m";
public const string Long = "mobile";
}
public static class PublicWebSite
{
public const string Long = "with-public-website";
}
public static class TemplateSource
{
public const string Short = "ts";
public const string Long = "template-source";
}
public static class ConnectionString
{
public const string Short = "cs";
public const string Long = "connection-string";
}
public static class CreateSolutionFolder
{
public const string Short = "csf";
public const string Long = "create-solution-folder";
}
public static class Tiered
{
public const string Long = "tiered";
}
public static class Preview
{
public const string Long = "preview";
}
}
}
}

@ -42,6 +42,7 @@ namespace Volo.Abp.Cli.Commands.Services
SolutionName.Parse(moduleName),
moduleName,
version,
outputFolder,
DatabaseProvider.NotSpecified,
DatabaseManagementSystem.NotSpecified,
UiFramework.NotSpecified,
@ -106,7 +107,8 @@ namespace Volo.Abp.Cli.Commands.Services
new ProjectBuildArgs(
SolutionName.Parse(packageName),
packageName,
version
version,
outputFolder
)
);
@ -156,7 +158,8 @@ namespace Volo.Abp.Cli.Commands.Services
new ProjectBuildArgs(
SolutionName.Parse(packageName),
packageName,
version
version,
outputFolder
)
);

@ -59,7 +59,7 @@ namespace Volo.Abp.Cli.Commands
Logger.LogInformation("Target culture: " + targetCulture);
Logger.LogInformation("Reference culture: " + referenceCulture);
Logger.LogInformation("Output file: " + outputFile);
if (allValues)
{
Logger.LogInformation("Include all keys");
@ -195,8 +195,11 @@ namespace Volo.Abp.Cli.Commands
var targetText = targetLocalizationInfo.Texts.FirstOrDefault(x => x.Name == text.LocalizationKey);
if (targetText != null)
{
Logger.LogInformation($"Update translation: {targetText.Name} => " + text.Target);
targetText.Value = text.Target;
if (!text.Target.IsNullOrEmpty())
{
Logger.LogInformation($"Update translation: {targetText.Name} => " + text.Target);
targetText.Value = text.Target;
}
}
else
{

@ -60,7 +60,9 @@ namespace Volo.Abp.Cli.ProjectBuilding.Building.Steps
foreach (XmlNode node in importNodes)
{
if (!(node.Attributes?["Project"]?.Value?.EndsWith("\\common.props") ?? false))
var value = node.Attributes?["Project"]?.Value;
if (value == null || (!value.EndsWith("\\common.props") && !value.EndsWith("\\common.test.props")))
{
continue;
}

@ -1,4 +1,5 @@
using System;
using NuGet.Versioning;
using Volo.Abp.Cli.ProjectBuilding.Building.Steps;
using Volo.Abp.Cli.ProjectBuilding.Templates.App;
using Volo.Abp.Cli.ProjectBuilding.Templates.Microservice;
@ -13,7 +14,11 @@ namespace Volo.Abp.Cli.ProjectBuilding.Building
var pipeline = new ProjectBuildPipeline(context);
pipeline.Steps.Add(new FileEntryListReadStep());
pipeline.Steps.Add(new CreateAppSettingsSecretsStep());
if (SemanticVersion.Parse(context.TemplateFile.Version) > new SemanticVersion(4, 3, 99))
{
pipeline.Steps.Add(new CreateAppSettingsSecretsStep());
}
pipeline.Steps.AddRange(context.Template.GetCustomSteps(context));

@ -37,6 +37,9 @@ namespace Volo.Abp.Cli.ProjectBuilding
[CanBeNull]
public string ConnectionString { get; set; }
[NotNull]
public string OutputFolder { get; set; }
[NotNull]
public Dictionary<string, string> ExtraProperties { get; set; }
@ -44,6 +47,7 @@ namespace Volo.Abp.Cli.ProjectBuilding
[NotNull] SolutionName solutionName,
[CanBeNull] string templateName = null,
[CanBeNull] string version = null,
string outputFolder = null,
DatabaseProvider databaseProvider = DatabaseProvider.NotSpecified,
DatabaseManagementSystem databaseManagementSystem = DatabaseManagementSystem.NotSpecified,
UiFramework uiFramework = UiFramework.NotSpecified,
@ -58,6 +62,7 @@ namespace Volo.Abp.Cli.ProjectBuilding
SolutionName = Check.NotNull(solutionName, nameof(solutionName));
TemplateName = templateName;
Version = version;
OutputFolder = outputFolder;
DatabaseProvider = databaseProvider;
DatabaseManagementSystem = databaseManagementSystem;
UiFramework = uiFramework;

@ -53,6 +53,11 @@ namespace Volo.Abp.Cli.ProjectBuilding.Templates.Module
projectFolderPath: "/aspnet-core/host/MyCompanyName.MyProjectName.Blazor.Host"
));
steps.Add(new RemoveProjectFromSolutionStep(
"MyCompanyName.MyProjectName.Blazor.Server.Host",
projectFolderPath: "/aspnet-core/host/MyCompanyName.MyProjectName.Blazor.Server.Host"
));
steps.Add(new RemoveProjectFromSolutionStep(
"MyCompanyName.MyProjectName.Web.Host",
projectFolderPath: "/aspnet-core/host/MyCompanyName.MyProjectName.Web.Host"

@ -69,9 +69,10 @@ namespace Volo.Abp.Cli.ProjectModification
protected virtual string FindDbContextName(string dbMigrationsFolder)
{
var dbContext = Directory
.GetFiles(dbMigrationsFolder, "*MigrationsDbContext.cs", SearchOption.AllDirectories)
.FirstOrDefault(fp => !fp.EndsWith("TenantMigrationsDbContext.cs"));
var dbContext = Directory.GetFiles(dbMigrationsFolder, "*MigrationsDbContext.cs", SearchOption.AllDirectories)
.FirstOrDefault(fp => !fp.EndsWith("TenantMigrationsDbContext.cs")) ??
Directory.GetFiles(dbMigrationsFolder, "*DbContext.cs", SearchOption.AllDirectories)
.FirstOrDefault(fp => !fp.EndsWith("TenantDbContext.cs"));
if (dbContext == null)
{
@ -83,9 +84,10 @@ namespace Volo.Abp.Cli.ProjectModification
protected virtual string FindTenantDbContextName(string dbMigrationsFolder)
{
var tenantDbContext = Directory
.GetFiles(dbMigrationsFolder, "*TenantMigrationsDbContext.cs", SearchOption.AllDirectories)
.FirstOrDefault();
var tenantDbContext = Directory.GetFiles(dbMigrationsFolder, "*TenantMigrationsDbContext.cs", SearchOption.AllDirectories)
.FirstOrDefault() ??
Directory.GetFiles(dbMigrationsFolder, "*TenantDbContext.cs", SearchOption.AllDirectories)
.FirstOrDefault();
if (tenantDbContext == null)
{

@ -67,7 +67,8 @@ namespace Volo.Abp.Cli.ProjectModification
if (oldNodeIncludeValue.EndsWith(".test", StringComparison.InvariantCultureIgnoreCase) ||
oldNodeIncludeValue.EndsWith(".tests", StringComparison.InvariantCultureIgnoreCase) ||
oldNodeIncludeValue.EndsWith(".testbase", StringComparison.InvariantCultureIgnoreCase))
oldNodeIncludeValue.EndsWith(".testbase", StringComparison.InvariantCultureIgnoreCase)||
oldNodeIncludeValue.EndsWith(".Demo", StringComparison.InvariantCultureIgnoreCase))
{
tempSourceFile = "test";
}

@ -576,11 +576,12 @@ namespace Volo.Abp.Cli.ProjectModification
return;
}
var dbMigrationsProject = projectFiles.FirstOrDefault(p => p.EndsWith(".DbMigrations.csproj"));
var dbMigrationsProject = projectFiles.FirstOrDefault(p => p.EndsWith(".DbMigrations.csproj"))
?? projectFiles.FirstOrDefault(p => p.EndsWith(".EntityFrameworkCore.csproj")) ;
if (dbMigrationsProject == null)
{
Logger.LogDebug("Solution doesn't have a \".DbMigrations\" project.");
Logger.LogDebug("Solution doesn't have a Migrations project.");
if (!skipDbMigrations)
{

@ -25,6 +25,7 @@
<PackageReference Include="Microsoft.Extensions.Options.ConfigurationExtensions" Version="$(MicrosoftPackageVersion)" />
<PackageReference Include="Microsoft.Extensions.Logging" Version="$(MicrosoftPackageVersion)" />
<PackageReference Include="Microsoft.Extensions.Hosting.Abstractions" Version="$(MicrosoftPackageVersion)" />
<PackageReference Include="System.Text.Encodings.Web" Version="5.0.1" />
<PackageReference Include="System.Runtime.Loader" Version="4.3.0" />
<PackageReference Include="System.Linq.Dynamic.Core" Version="1.2.9" />
<PackageReference Include="System.Linq.Queryable" Version="4.3.0" />

@ -0,0 +1,6 @@
{
"culture": "ro-RO",
"texts": {
"MaxResultCountExceededExceptionMessage": "{0} nu poate fi mai mare decât {1}! Măriţi {2}.{3} pe partea de server pentru a permite mai multe rezultate."
}
}

@ -1,6 +1,4 @@
using System;
using System.Collections.Generic;
using System.Linq.Expressions;
using System.Collections.Generic;
using System.Threading;
using System.Threading.Tasks;
using JetBrains.Annotations;
@ -18,19 +16,7 @@ namespace Volo.Abp.Domain.Repositories
/// <param name="cancellationToken">A <see cref="T:System.Threading.CancellationToken" /> to observe while waiting for the task to complete.</param>
/// <returns>Entity</returns>
Task<List<TEntity>> GetListAsync(bool includeDetails = false, CancellationToken cancellationToken = default);
/// <summary>
/// Gets a list entities by the given <paramref name="predicate"/>.
/// </summary>
/// <param name="predicate">A condition to find the entity</param>
/// <param name="includeDetails">Set true to include all children of this entity</param>
/// <param name="cancellationToken">A <see cref="T:System.Threading.CancellationToken" /> to observe while waiting for the task to complete.</param>
/// <returns>Entity</returns>
Task<List<TEntity>> GetListAsync(
[NotNull] Expression<Func<TEntity, bool>> predicate,
bool includeDetails = false,
CancellationToken cancellationToken = default);
/// <summary>
/// Gets total count of all entities.
/// </summary>

@ -1,7 +1,10 @@
using System;
using System.Collections.Generic;
using System.Linq;
using System.Linq.Expressions;
using System.Threading;
using System.Threading.Tasks;
using JetBrains.Annotations;
using Volo.Abp.Domain.Entities;
using Volo.Abp.Linq;
@ -23,6 +26,17 @@ namespace Volo.Abp.Domain.Repositories
Task<IQueryable<TEntity>> WithDetailsAsync(params Expression<Func<TEntity, object>>[] propertySelectors); //TODO: CancellationToken
Task<IQueryable<TEntity>> GetQueryableAsync(); //TODO: CancellationToken
/// <summary>
/// Gets a list of entities by the given <paramref name="predicate"/>.
/// </summary>
/// <param name="predicate">A condition to filter the entities</param>
/// <param name="includeDetails">Set true to include details (sub-collections) of this entity</param>
/// <param name="cancellationToken">A <see cref="T:System.Threading.CancellationToken" /> to observe while waiting for the task to complete.</param>
Task<List<TEntity>> GetListAsync(
[NotNull] Expression<Func<TEntity, bool>> predicate,
bool includeDetails = false,
CancellationToken cancellationToken = default);
}
public interface IReadOnlyRepository<TEntity, TKey> : IReadOnlyRepository<TEntity>, IReadOnlyBasicRepository<TEntity, TKey>

@ -0,0 +1,25 @@
{
"culture": "ro-RO",
"texts": {
"DisplayName:Abp.Mailing.DefaultFromAddress": "Implicit de la adresa",
"DisplayName:Abp.Mailing.DefaultFromDisplayName": "Implicit de la numele afişat",
"DisplayName:Abp.Mailing.Smtp.Host": "Host",
"DisplayName:Abp.Mailing.Smtp.Port": "Port",
"DisplayName:Abp.Mailing.Smtp.UserName": "Nume de utilizator",
"DisplayName:Abp.Mailing.Smtp.Password": "Parola",
"DisplayName:Abp.Mailing.Smtp.Domain": "Domeniu",
"DisplayName:Abp.Mailing.Smtp.EnableSsl": "Activează SSL",
"DisplayName:Abp.Mailing.Smtp.UseDefaultCredentials": "Folosiţi credenţialele implicite",
"Description:Abp.Mailing.DefaultFromAddress": "Implicit de la adresa",
"Description:Abp.Mailing.DefaultFromDisplayName": "Implicit de la numele afişat",
"Description:Abp.Mailing.Smtp.Host": "Numele sau adresa IP al hostului folosit pentru tranzacţiile SMTP.",
"Description:Abp.Mailing.Smtp.Port": "Portul folosit pentru tranzacţiile SMTP.",
"Description:Abp.Mailing.Smtp.UserName": "Numele de utilizator asociat credenţialelor.",
"Description:Abp.Mailing.Smtp.Password": "Parola folosită pentru numele de utilizator asociat credenţialelor.",
"Description:Abp.Mailing.Smtp.Domain": "Domeniul sau numele calculatorului care verifică credenţialele.",
"Description:Abp.Mailing.Smtp.EnableSsl": "Dacă SmtpClient foloseste Secure Sockets Layer (SSL) pentru a cripta conexiunea.",
"Description:Abp.Mailing.Smtp.UseDefaultCredentials": "Dacă DefaultCredentials sunt trimise odată cu solicitările.",
"TextTemplate:StandardEmailTemplates.Layout": "Şablon aspect implicit pentru email",
"TextTemplate:StandardEmailTemplates.Message": "Şablon de mesaj simplu pentru emailuri"
}
}

@ -30,17 +30,13 @@ namespace Volo.Abp.EventBus.Local
protected ConcurrentDictionary<Type, List<IEventHandlerFactory>> HandlerFactories { get; }
protected IJsonSerializer Serializer { get; }
public LocalEventBus(
IOptions<AbpLocalEventBusOptions> options,
IServiceScopeFactory serviceScopeFactory,
ICurrentTenant currentTenant,
IEventErrorHandler errorHandler,
IJsonSerializer serializer)
IEventErrorHandler errorHandler)
: base(serviceScopeFactory, currentTenant, errorHandler)
{
Serializer = serializer;
Options = options.Value;
Logger = NullLogger<LocalEventBus>.Instance;
@ -131,10 +127,9 @@ namespace Volo.Abp.EventBus.Local
public virtual async Task PublishAsync(LocalEventMessage localEventMessage)
{
var rawEventData = Serializer.Serialize(localEventMessage.EventData);
await TriggerHandlersAsync(localEventMessage.EventType, localEventMessage.EventData, errorContext =>
{
errorContext.EventData = Serializer.Deserialize(localEventMessage.EventType, rawEventData);
errorContext.EventData = localEventMessage.EventData;
errorContext.SetProperty(nameof(LocalEventMessage.MessageId), localEventMessage.MessageId);
});
}

@ -0,0 +1,26 @@
{
"culture": "ro-RO",
"texts": {
"InternalServerErrorMessage": "A intervenit o eroare internă în timpul solicitării dumneavoastră!",
"ValidationErrorMessage": "Solicitarea dumneavoastră este invalidă!",
"ValidationNarrativeErrorMessageTitle": "Următoarele erori au fost detectate în timpul validării",
"DefaultErrorMessage": "A intervenit o eroare!",
"DefaultErrorMessageDetail": "Detaliile erorii nu au fost trimise de către server.",
"DefaultErrorMessage401": "Nu sunteţi autentificat!",
"DefaultErrorMessage401Detail": "Ar trebui să vă autentificaţi pentru a efectua această operaţie.",
"DefaultErrorMessage403": "Nu sunteţi autorizat!",
"DefaultErrorMessage403Detail": "Nu sunteţi autorizat să efectuaţi această operaţie!",
"DefaultErrorMessage404": "Resursa nu a fost găsită!",
"DefaultErrorMessage404Detail": "Resursa solicitată nu a fost găsită pe server!",
"EntityNotFoundErrorMessage": "Nu există entitatea {0} cu id-ul {1}!",
"AbpDbConcurrencyErrorMessage": "Datele pe care le-aţi trimis au fost modificate deja de către alt utilizator/client. Vă rugăm să renunţaţi la modificările pe care le-aţi făcut şi să încercaţi de la început.",
"Error": "Eroare",
"UnhandledException": "Excepţie netratată!",
"401Message": "Neautorizat",
"403Message": "Interzis",
"404Message": "Pagina nu a fost găsită",
"500Message": "Eroare internă a serverului",
"403MessageDetail": "Nu sunteţi autorizat să efectuaţi această operaţie!",
"404MessageDetail": "Ne cerem scuze, nu este nimic la această adresă."
}
}

@ -13,6 +13,7 @@
"DefaultErrorMessage404": "未找到资源!",
"DefaultErrorMessage404Detail": "未在服务中找到请求的资源!",
"EntityNotFoundErrorMessage": "实体 {0} 不存在,id = {1}!",
"AbpDbConcurrencyErrorMessage": "你提交的数据已经被其他用户/客户端修改.请放弃你所做的修改并再次尝试.",
"Error": "错误",
"UnhandledException": "未处理的异常!",
"401Message": "未授权",
@ -22,4 +23,4 @@
"403MessageDetail": "您没有权限执行此操作",
"404MessageDetail": "抱歉, 这个地址是空的"
}
}
}

@ -13,6 +13,7 @@
"DefaultErrorMessage404": "未找到資源!",
"DefaultErrorMessage404Detail": "未在服務中找到請求的資源!",
"EntityNotFoundErrorMessage": "實體 {0} 不存在,id = {1}!",
"AbpDbConcurrencyErrorMessage": "你提交的數據已經被其他用戶/客戶端修改.請放棄你所做的修改並再次嘗試.",
"Error": "錯誤",
"UnhandledException": "未處理的異常!",
"401Message": "未授權",
@ -22,4 +23,4 @@
"403MessageDetail": "你不被授權執行此操作",
"404MessageDetail": "對不起,地址是空的"
}
}
}

@ -0,0 +1,8 @@
{
"culture": "ro-RO",
"texts": {
"Volo.Feature:010001": "Caracteristica nu este activată: {FeatureName}",
"Volo.Feature:010002": "Caracteristicile necesare nu sunt activate. Toate aceste caracteristici trebuie activate: {FeatureNames}",
"Volo.Feature:010003": "Caracteristicile necesare nu sunt activate. Cel puţin una din aceste caracteristici trebuie activată: {FeatureNames}"
}
}

@ -2,6 +2,7 @@ using FluentValidation;
using System;
using System.ComponentModel.DataAnnotations;
using System.Linq;
using System.Threading.Tasks;
using Volo.Abp.DependencyInjection;
using Volo.Abp.Validation;
@ -17,7 +18,7 @@ namespace Volo.Abp.FluentValidation
_serviceProvider = serviceProvider;
}
public void AddErrors(ObjectValidationContext context)
public virtual async Task AddErrorsAsync(ObjectValidationContext context)
{
var serviceType = typeof(IValidator<>).MakeGenericType(context.ValidatingObject.GetType());
var validator = _serviceProvider.GetService(serviceType) as IValidator;
@ -26,7 +27,7 @@ namespace Volo.Abp.FluentValidation
return;
}
var result = validator.Validate((IValidationContext) Activator.CreateInstance(
var result = await validator.ValidateAsync((IValidationContext) Activator.CreateInstance(
typeof(ValidationContext<>).MakeGenericType(context.ValidatingObject.GetType()),
context.ValidatingObject));

@ -0,0 +1,6 @@
{
"culture": "ro-RO",
"texts": {
"Volo.GlobalFeature:010001": "Serviciul '{ServiceName}' trebuie să activeze caracteristica '{GlobalFeatureName}'."
}
}

@ -19,6 +19,7 @@
</ItemGroup>
<ItemGroup>
<ProjectReference Include="..\Volo.Abp.Authorization.Abstractions\Volo.Abp.Authorization.Abstractions.csproj" />
<ProjectReference Include="..\Volo.Abp.Core\Volo.Abp.Core.csproj" />
</ItemGroup>

@ -0,0 +1,46 @@
using System;
using System.Threading.Tasks;
using Hangfire.Dashboard;
using Microsoft.Extensions.DependencyInjection;
using Volo.Abp.Authorization.Permissions;
using Volo.Abp.Users;
namespace Volo.Abp.Hangfire
{
public class AbpHangfireAuthorizationFilter : IDashboardAsyncAuthorizationFilter
{
private readonly string _requiredPermissionName;
public AbpHangfireAuthorizationFilter(string requiredPermissionName = null)
{
_requiredPermissionName = requiredPermissionName;
}
public async Task<bool> AuthorizeAsync(DashboardContext context)
{
if (!IsLoggedIn(context))
{
return false;
}
if (_requiredPermissionName.IsNullOrEmpty())
{
return true;
}
return await IsPermissionGrantedAsync(context, _requiredPermissionName);
}
private static bool IsLoggedIn(DashboardContext context)
{
var currentUser = context.GetHttpContext().RequestServices.GetRequiredService<ICurrentUser>();
return currentUser.IsAuthenticated;
}
private static async Task<bool> IsPermissionGrantedAsync(DashboardContext context, string requiredPermissionName)
{
var permissionChecker = context.GetHttpContext().RequestServices.GetRequiredService<IPermissionChecker>();
return await permissionChecker.IsGrantedAsync(requiredPermissionName);
}
}
}

@ -1,10 +1,12 @@
using Hangfire;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Options;
using Volo.Abp.Authorization;
using Volo.Abp.Modularity;
namespace Volo.Abp.Hangfire
{
[DependsOn(typeof(AbpAuthorizationAbstractionsModule))]
public class AbpHangfireModule : AbpModule
{
private BackgroundJobServer _backgroundJobServer;

@ -173,7 +173,7 @@ namespace Volo.Abp.Http.Client.DynamicProxying
var response = await client.SendAsync(
requestMessage,
HttpCompletionOption.ResponseHeadersRead /*this will buffer only the headers, the content will be used as a stream*/,
GetCancellationToken()
GetCancellationToken(invocation)
);
if (!response.IsSuccessStatusCode)
@ -305,8 +305,18 @@ namespace Volo.Abp.Http.Client.DynamicProxying
return input;
}
protected virtual CancellationToken GetCancellationToken()
protected virtual CancellationToken GetCancellationToken(IAbpMethodInvocation invocation)
{
var cancellationTokenArg = invocation.Arguments.LastOrDefault(x => x is CancellationToken);
if (cancellationTokenArg != null)
{
var cancellationToken = (CancellationToken) cancellationTokenArg;
if (cancellationToken != default)
{
return cancellationToken;
}
}
return CancellationTokenProvider.Token;
}
}

@ -1,4 +1,5 @@
using System;
using System.Text.Encodings.Web;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Options;
using Volo.Abp.Json.SystemTextJson.JsonConverters;
@ -24,6 +25,9 @@ namespace Volo.Abp.Json.SystemTextJson
options.JsonSerializerOptions.Converters.Add(new ObjectToInferredTypesConverter());
options.JsonSerializerOptions.Converters.Add(new AbpHasExtraPropertiesJsonConverterFactory());
// If the user hasn't explicitly configured the encoder, use the less strict encoder that does not encode all non-ASCII characters.
options.JsonSerializerOptions.Encoder ??= JavaScriptEncoder.UnsafeRelaxedJsonEscaping;
}
}
}

@ -11,9 +11,18 @@ namespace Volo.Abp.Json.SystemTextJson.JsonConverters
{
public override T Read(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options)
{
var newOptions = JsonSerializerOptionsHelper.Create(options, x =>
x == this ||
x.GetType() == typeof(AbpHasExtraPropertiesJsonConverterFactory));
var newOptions = JsonSerializerOptionsHelper.Create(options, x => x == this);
var converterFactory = newOptions.Converters.FirstOrDefault(x => x is AbpHasExtraPropertiesJsonConverterFactory).As<AbpHasExtraPropertiesJsonConverterFactory>();
var newConverterFactory = new AbpHasExtraPropertiesJsonConverterFactory();
if (converterFactory != null)
{
newOptions.Converters.Remove(converterFactory);
newConverterFactory.AddExcludeTypes(converterFactory.GetExcludeTypes().ToArray());
}
newConverterFactory.AddExcludeTypes(typeToConvert);
newOptions.Converters.Add(newConverterFactory);
var rootElement = JsonDocument.ParseValue(ref reader).RootElement;
if (rootElement.ValueKind == JsonValueKind.Object)

@ -1,5 +1,7 @@
using System;
using System.Collections.Concurrent;
using System.Collections.Generic;
using System.Collections.Immutable;
using System.Reflection;
using System.Text.Json;
using System.Text.Json.Serialization;
@ -11,8 +13,26 @@ namespace Volo.Abp.Json.SystemTextJson.JsonConverters
{
private static readonly ConcurrentDictionary<Type, bool> CachedTypes = new ConcurrentDictionary<Type, bool>();
private readonly List<Type> _excludeTypes = new List<Type>();
public virtual AbpHasExtraPropertiesJsonConverterFactory AddExcludeTypes(params Type[] excludeTypes)
{
_excludeTypes.AddIfNotContains(excludeTypes);
return this;
}
public virtual IReadOnlyList<Type> GetExcludeTypes()
{
return _excludeTypes.ToImmutableList();
}
public override bool CanConvert(Type typeToConvert)
{
if (_excludeTypes.Contains(typeToConvert))
{
return false;
}
//Only for private or protected ExtraProperties.
if (typeof(IHasExtraProperties).IsAssignableFrom(typeToConvert))
{

@ -0,0 +1,19 @@
{
"culture": "ro-RO",
"texts": {
"DisplayName:Abp.Ldap.ServerHost": "Server host",
"Description:Abp.Ldap.ServerHost": "Server host",
"DisplayName:Abp.Ldap.ServerPort": "Server port",
"Description:Abp.Ldap.ServerPort": "Server port",
"DisplayName:Abp.Ldap.BaseDc": "Componenta domeniului de bază",
"Description:Abp.Ldap.BaseDc": "Componenta domeniului de bază",
"DisplayName:Abp.Ldap.UserName": "Nume de utilizator",
"Description:Abp.Ldap.UserName": "Nume de utilizator",
"DisplayName:Abp.Ldap.Password": "Parola",
"Description:Abp.Ldap.Password": "Parola"
}
}

@ -0,0 +1,7 @@
{
"culture": "ro-RO",
"texts": {
"DisplayName:Abp.Localization.DefaultLanguage": "Limba implicită",
"Description:Abp.Localization.DefaultLanguage": "Limba implicită a aplicaţiei."
}
}

@ -18,6 +18,7 @@
</ItemGroup>
<ItemGroup>
<PackageReference Include="AlibabaCloud.OpenApiUtil" Version="1.0.10" />
<PackageReference Include="AlibabaCloud.SDK.Dysmsapi20170525" Version="1.0.2"/>
</ItemGroup>

@ -0,0 +1,14 @@
using System.Threading;
namespace Volo.Abp.Threading
{
public class CancellationTokenOverride
{
public CancellationToken CancellationToken { get; }
public CancellationTokenOverride(CancellationToken cancellationToken)
{
CancellationToken = cancellationToken;
}
}
}

@ -0,0 +1,26 @@
using System;
using System.Threading;
namespace Volo.Abp.Threading
{
public abstract class CancellationTokenProviderBase : ICancellationTokenProvider
{
public const string CancellationTokenOverrideContextKey = "Volo.Abp.Threading.CancellationToken.Override";
public abstract CancellationToken Token { get; }
protected IAmbientScopeProvider<CancellationTokenOverride> CancellationTokenOverrideScopeProvider { get; }
protected CancellationTokenOverride OverrideValue => CancellationTokenOverrideScopeProvider.GetValue(CancellationTokenOverrideContextKey);
protected CancellationTokenProviderBase(IAmbientScopeProvider<CancellationTokenOverride> cancellationTokenOverrideScopeProvider)
{
CancellationTokenOverrideScopeProvider = cancellationTokenOverrideScopeProvider;
}
public IDisposable Use(CancellationToken cancellationToken)
{
return CancellationTokenOverrideScopeProvider.BeginScope(CancellationTokenOverrideContextKey, new CancellationTokenOverride(cancellationToken));
}
}
}

@ -1,9 +1,12 @@
using System.Threading;
using System;
using System.Threading;
namespace Volo.Abp.Threading
{
public interface ICancellationTokenProvider
{
CancellationToken Token { get; }
IDisposable Use(CancellationToken cancellationToken);
}
}

@ -2,15 +2,15 @@
namespace Volo.Abp.Threading
{
public class NullCancellationTokenProvider : ICancellationTokenProvider
public class NullCancellationTokenProvider : CancellationTokenProviderBase
{
public static NullCancellationTokenProvider Instance { get; } = new NullCancellationTokenProvider();
public CancellationToken Token { get; } = CancellationToken.None;
public override CancellationToken Token => OverrideValue?.CancellationToken ?? CancellationToken.None;
private NullCancellationTokenProvider()
: base(new AmbientDataContextAmbientScopeProvider<CancellationTokenOverride>(new AsyncLocalAmbientDataContext()))
{
}
}
}
}

@ -0,0 +1,7 @@
{
"culture": "ro-RO",
"texts": {
"DisplayName:Abp.Timing.Timezone": "Fus orar",
"Description:Abp.Timing.Timezone": "Fusul orar al aplicaţiei"
}
}

@ -0,0 +1,6 @@
{
"culture": "ro-RO",
"texts": {
"Menu:Administration": "Administrare"
}
}

@ -0,0 +1,52 @@
{
"culture": "ro-RO",
"texts": {
"Languages": "Limbi",
"AreYouSure": "Sunteţi sigur(ă)?",
"Cancel": "Anulează",
"Clear": "Goleşte",
"Yes": "Da",
"No": "Nu",
"Ok": "Ok",
"Close": "Închide",
"Save": "Salvează",
"SavingWithThreeDot": "Se salvează...",
"Actions": "Acţiuni",
"Delete": "Şterge",
"Edit": "Editează",
"Refresh": "Reîncarcă",
"Language": "Limba",
"LoadMore": "Încarcă mai multe",
"ProcessingWithThreeDot": "Se procesează...",
"LoadingWithThreeDot": "Se încarcă...",
"Welcome": "Bun venit",
"Login": "Autentificare",
"Register": "Înregistrare",
"Logout": "Delogare",
"Submit": "Trimite",
"Back": "Înapoi",
"PagerSearch": "Caută",
"PagerNext": "Următorul",
"PagerPrevious": "Anterior",
"PagerFirst": "Primul",
"PagerLast": "Ultimul",
"PagerInfo": "Afişate _START_ to _END_ of _TOTAL_ înregistrări",
"PagerInfo{0}{1}{2}": "Afişate {0} to {1} of {2} înregistrări",
"PagerInfoEmpty": "Afişate 0 to 0 of 0 înregistrări",
"PagerInfoFiltered": "(filtrate de la _MAX_ înregistrări totale)",
"NoDataAvailableInDatatable": "Nicio informaţie disponibilă",
"Total": "total",
"Selected": "selectate",
"PagerShowMenuEntries": "Arată _MENU_ intrări",
"DatatableActionDropdownDefaultText": "Acţiuni",
"ChangePassword": "Schimbă parola",
"PersonalInfo": "Profilul meu",
"AreYouSureYouWantToCancelEditingWarningMessage": "Aveţi modificări nesalvate.",
"GoHomePage": "Pagina principală",
"GoBack": "Înapoi",
"Search": "Caută",
"ItemWillBeDeletedMessageWithFormat": "{0} va fi şters!",
"ItemWillBeDeletedMessage": "Acest articol va fi şters!",
"ManageYourAccount": "Administraţi-vă contul"
}
}

@ -4,6 +4,7 @@ using System.Collections.Generic;
using System.ComponentModel;
using System.ComponentModel.DataAnnotations;
using System.Linq;
using System.Threading.Tasks;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Options;
using Volo.Abp.DependencyInjection;
@ -26,9 +27,10 @@ namespace Volo.Abp.Validation
Options = options.Value;
}
public void AddErrors(ObjectValidationContext context)
public Task AddErrorsAsync(ObjectValidationContext context)
{
ValidateObjectRecursively(context.Errors, context.ValidatingObject, currentDepth: 1);
return Task.CompletedTask;
}
protected virtual void ValidateObjectRecursively(List<ValidationResult> errors, object validatingObject, int currentDepth)

@ -1,7 +1,9 @@
namespace Volo.Abp.Validation
using System.Threading.Tasks;
namespace Volo.Abp.Validation
{
public interface IMethodInvocationValidator
{
void Validate(MethodInvocationValidationContext context);
Task ValidateAsync(MethodInvocationValidationContext context);
}
}

@ -1,7 +1,9 @@
namespace Volo.Abp.Validation
using System.Threading.Tasks;
namespace Volo.Abp.Validation
{
public interface IObjectValidationContributor
{
void AddErrors(ObjectValidationContext context);
Task AddErrorsAsync(ObjectValidationContext context);
}
}
}

@ -1,20 +1,21 @@
using System.Collections.Generic;
using System.ComponentModel.DataAnnotations;
using System.Threading.Tasks;
namespace Volo.Abp.Validation
{
public interface IObjectValidator
{
void Validate(
Task ValidateAsync(
object validatingObject,
string name = null,
bool allowNull = false
);
List<ValidationResult> GetErrors(
Task<List<ValidationResult>> GetErrorsAsync(
object validatingObject,
string name = null,
bool allowNull = false
);
}
}
}

@ -0,0 +1,34 @@
{
"culture": "ro-RO",
"texts": {
"'{0}' and '{1}' do not match.": "'{0}' şi '{1}' nu se potrivesc.",
"The {0} field is not a valid credit card number.": "Câmpul {0} nu este un număr valid de card de credit.",
"{0} is not valid.": "{0} nu este valid.",
"The {0} field is not a valid e-mail address.": "Câmpul {0} nu este o adresa de e-mail validă.",
"The {0} field only accepts files with the following extensions: {1}": "Câmpul {0} acceptă doar fişiere cu următoarele extensii: {1}",
"The field {0} must be a string or array type with a maximum length of '{1}'.": "Câmpul field {0} trebuie să fie de tipul string sau array cu lungimea maximă de '{1}'.",
"The field {0} must be a string or array type with a minimum length of '{1}'.": "Câmpul field {0} trebuie să fie de tipul string sau array cu lungimea minimă de '{1}'.",
"The {0} field is not a valid phone number.": "Câmpul {0} nu este un număr de telefon valid.",
"The field {0} must be between {1} and {2}.": "Câmpul {0} trebuie să fie între {1} şi {2}.",
"The field {0} must match the regular expression '{1}'.": "Câmpul {0} nu corespunde formatului solicitat.",
"The {0} field is required.": "Câmpul {0} este obligatoriu.",
"The field {0} must be a string with a maximum length of {1}.": "Câmpul {0} trebuie să fie un string cu lungimea maximă de {1}.",
"The field {0} must be a string with a minimum length of {2} and a maximum length of {1}.": "Câmpul {0} trebuie să fie un string cu lungimea minimă de {2} şi lungimea maximă de {1}.",
"The {0} field is not a valid fully-qualified http, https, or ftp URL.": "Câmpul {0} nu este o adresă validă complet http, https sau ftp.",
"The field {0} is invalid.": "Câmpul {0} este invalid.",
"ThisFieldIsNotAValidCreditCardNumber.": "Acest câmp nu este un număr de card de credit valid.",
"ThisFieldIsNotValid.": "Acest câmp nu este valid.",
"ThisFieldIsNotAValidEmailAddress.": "Acest câmp nu este o adresă de e-mail validă.",
"ThisFieldOnlyAcceptsFilesWithTheFollowingExtensions:{0}": "Acest câmp acceptă doar fişiere cu următoarele extensii: {0}",
"ThisFieldMustBeAStringOrArrayTypeWithAMaximumLengthOf{0}": "Acest câmp trebuie să fie de tipul string sau array cu lungimea maximă de '{0}'.",
"ThisFieldMustBeAStringOrArrayTypeWithAMinimumLengthOf{0}": "Acest câmp trebuie să fie de tipul string sau arraz cu lungimea minimă de '{0}'.",
"ThisFieldIsNotAValidPhoneNumber.": "Acest câmp nu este un număr de telefon valid.",
"ThisFieldMustBeBetween{0}And{1}": "Acest câmp trebuie să fie între {0} şi {1}.",
"ThisFieldMustMatchTheRegularExpression{0}": "Acest câmp trebuie să se potrivească cu expresia uzuală '{0}'.",
"ThisFieldIsRequired.": "Acest câmp este obligatoriu.",
"ThisFieldMustBeAStringWithAMaximumLengthOf{0}": "Acest câmp trebuie să fie un string cu lungimea maximă de {0}.",
"ThisFieldMustBeAStringWithAMinimumLengthOf{1}AndAMaximumLengthOf{0}": "Acest câmp trebuie să fie un string cu lungimea minimă de {1} şi lungimea maximă de {0}.",
"ThisFieldIsNotAValidFullyQualifiedHttpHttpsOrFtpUrl": "Acest câmp nu este o adresă validă http, https sau ftp.",
"ThisFieldIsInvalid.": "Acest câmp este invalid."
}
}

@ -2,6 +2,7 @@ using System;
using System.Collections.Generic;
using System.Linq;
using System.Reflection;
using System.Threading.Tasks;
using Volo.Abp.DependencyInjection;
using Volo.Abp.Reflection;
@ -16,7 +17,7 @@ namespace Volo.Abp.Validation
_objectValidator = objectValidator;
}
public virtual void Validate(MethodInvocationValidationContext context)
public virtual async Task ValidateAsync(MethodInvocationValidationContext context)
{
Check.NotNull(context, nameof(context));
@ -46,7 +47,7 @@ namespace Volo.Abp.Validation
ThrowValidationError(context);
}
AddMethodParameterValidationErrors(context);
await AddMethodParameterValidationErrorsAsync(context);
if (context.Errors.Any())
{
@ -60,7 +61,7 @@ namespace Volo.Abp.Validation
{
return false;
}
if (ReflectionHelper.GetSingleAttributeOfMemberOrDeclaringTypeOrDefault<DisableValidationAttribute>(context.Method) != null)
{
return true;
@ -82,22 +83,22 @@ namespace Volo.Abp.Validation
);
}
protected virtual void AddMethodParameterValidationErrors(MethodInvocationValidationContext context)
protected virtual async Task AddMethodParameterValidationErrorsAsync(MethodInvocationValidationContext context)
{
for (var i = 0; i < context.Parameters.Length; i++)
{
AddMethodParameterValidationErrors(context, context.Parameters[i], context.ParameterValues[i]);
await AddMethodParameterValidationErrorsAsync(context, context.Parameters[i], context.ParameterValues[i]);
}
}
protected virtual void AddMethodParameterValidationErrors(IAbpValidationResult context, ParameterInfo parameterInfo, object parameterValue)
protected virtual async Task AddMethodParameterValidationErrorsAsync(IAbpValidationResult context, ParameterInfo parameterInfo, object parameterValue)
{
var allowNulls = parameterInfo.IsOptional ||
parameterInfo.IsOut ||
TypeHelper.IsPrimitiveExtended(parameterInfo.ParameterType, includeEnums: true);
context.Errors.AddRange(
_objectValidator.GetErrors(
await _objectValidator.GetErrorsAsync(
parameterValue,
parameterInfo.Name,
allowNulls
@ -105,4 +106,4 @@ namespace Volo.Abp.Validation
);
}
}
}
}

@ -2,6 +2,7 @@ using Microsoft.Extensions.Options;
using System.Collections.Generic;
using System.ComponentModel.DataAnnotations;
using System.Linq;
using System.Threading.Tasks;
using Microsoft.Extensions.DependencyInjection;
using Volo.Abp.DependencyInjection;
@ -18,9 +19,9 @@ namespace Volo.Abp.Validation
Options = options.Value;
}
public virtual void Validate(object validatingObject, string name = null, bool allowNull = false)
public virtual async Task ValidateAsync(object validatingObject, string name = null, bool allowNull = false)
{
var errors = GetErrors(validatingObject, name, allowNull);
var errors = await GetErrorsAsync(validatingObject, name, allowNull);
if (errors.Any())
{
@ -31,7 +32,7 @@ namespace Volo.Abp.Validation
}
}
public virtual List<ValidationResult> GetErrors(object validatingObject, string name = null, bool allowNull = false)
public virtual async Task<List<ValidationResult>> GetErrorsAsync(object validatingObject, string name = null, bool allowNull = false)
{
if (validatingObject == null)
{
@ -58,7 +59,7 @@ namespace Volo.Abp.Validation
{
var contributor = (IObjectValidationContributor)
scope.ServiceProvider.GetRequiredService(contributorType);
contributor.AddErrors(context);
await contributor.AddErrorsAsync(context);
}
}

@ -15,13 +15,13 @@ namespace Volo.Abp.Validation
public override async Task InterceptAsync(IAbpMethodInvocation invocation)
{
Validate(invocation);
await ValidateAsync(invocation);
await invocation.ProceedAsync();
}
protected virtual void Validate(IAbpMethodInvocation invocation)
protected virtual async Task ValidateAsync(IAbpMethodInvocation invocation)
{
_methodInvocationValidator.Validate(
await _methodInvocationValidator.ValidateAsync(
new MethodInvocationValidationContext(
invocation.TargetObject,
invocation.Method,

@ -88,6 +88,7 @@ namespace Volo.Abp.AspNetCore.Mvc
options.Languages.Add(new LanguageInfo("en", "en", "English"));
options.Languages.Add(new LanguageInfo("hu", "hu", "Magyar"));
options.Languages.Add(new LanguageInfo("ro-RO", "ro-RO", "Română"));
options.Languages.Add(new LanguageInfo("sk", "sk", "Slovak"));
options.Languages.Add(new LanguageInfo("tr", "tr", "Türkçe"));
});

@ -33,9 +33,7 @@ namespace Volo.Abp.EventBus.Local
LocalEventBus.Subscribe<MyExceptionHandleEventData>(eventData =>
{
eventData.Value.ShouldBe(0);
retryAttempt++;
eventData.Value++;
if (retryAttempt < 2)
{
throw new Exception("This exception is intentionally thrown!");
@ -58,7 +56,6 @@ namespace Volo.Abp.EventBus.Local
eventData.Value.ShouldBe(0);
retryAttempt++;
eventData.Value++;
throw new Exception("This exception is intentionally thrown!");
});

@ -39,7 +39,8 @@ namespace Volo.Abp.FluentValidation
},
MyMethodInput3 = new MyMethodInput3
{
MyStringValue3 = "ccc"
MyStringValue3 = "ccc",
MyBoolValue3 = true
}
});
@ -62,12 +63,13 @@ namespace Volo.Abp.FluentValidation
},
MyMethodInput3 = new MyMethodInput3
{
MyStringValue3 = "c"
MyStringValue3 = "c",
MyBoolValue3 = false
}
}
)
);
exception.ValidationErrors.ShouldContain(x => x.MemberNames.Contains("MyStringValue"));
exception.ValidationErrors.ShouldContain(x => x.MemberNames.Contains("MyMethodInput2.MyStringValue2"));
exception.ValidationErrors.ShouldContain(x => x.MemberNames.Contains("MyMethodInput3.MyStringValue3"));
@ -100,7 +102,7 @@ namespace Volo.Abp.FluentValidation
output.ShouldBe("444");
}
[DependsOn(typeof(AbpAutofacModule))]
[DependsOn(typeof(AbpFluentValidationModule))]
public class TestModule : AbpModule
@ -162,6 +164,8 @@ namespace Volo.Abp.FluentValidation
{
public string MyStringValue3 { get; set; }
public bool MyBoolValue3 { get; set; }
}
public class MyMethodInput4
@ -175,7 +179,8 @@ namespace Volo.Abp.FluentValidation
{
RuleFor(x => x.MyStringValue).Equal("aaa");
RuleFor(x => x.MyMethodInput2.MyStringValue2).Equal("bbb");
RuleFor(customer => customer.MyMethodInput3).SetValidator(new MyMethodInput3Validator());
RuleFor(x => x.MyMethodInput3).SetValidator(new MyMethodInput3Validator());
RuleFor(x => x.MyMethodInput3).SetValidator(new MyMethodInput3AsyncValidator());
}
}
@ -194,5 +199,15 @@ namespace Volo.Abp.FluentValidation
RuleFor(x => x.MyStringValue3).Equal("ccc");
}
}
public class MyMethodInput3AsyncValidator : MethodInputBaseValidator
{
public MyMethodInput3AsyncValidator()
{
RuleFor(x => x.MyStringValue3).Equal("ccc");
RuleFor(x => x.MyBoolValue3).MustAsync((myBookValue3, cancellation) => Task.FromResult(myBookValue3));
}
}
}
}
}

@ -1,4 +1,5 @@
using System;
using System.Threading;
using System.Threading.Tasks;
namespace Volo.Abp.Http.DynamicProxying
@ -36,5 +37,7 @@ namespace Volo.Abp.Http.DynamicProxying
Task<string> PatchValueWithHeaderAndQueryStringAsync(string headerValue, string qsValue);
Task<int> DeleteByIdAsync(int id);
Task<string> AbortRequestAsync(CancellationToken cancellationToken = default);
}
}

@ -1,11 +1,10 @@
using System;
using System.Collections.Generic;
using System.Globalization;
using System.Threading;
using System.Threading.Tasks;
using Microsoft.AspNetCore.Mvc;
using Volo.Abp.Application.Services;
using Volo.Abp.AspNetCore.Mvc;
using Volo.Abp.UI;
namespace Volo.Abp.Http.DynamicProxying
{
@ -129,6 +128,14 @@ namespace Volo.Abp.Http.DynamicProxying
{
return Task.FromResult(id + 1);
}
[HttpGet]
[Route("abort-request")]
public async Task<string> AbortRequestAsync(CancellationToken cancellationToken = default)
{
await Task.Delay(100, cancellationToken);
return "AbortRequestAsync";
}
}
public class Car

@ -1,10 +1,10 @@
using System;
using System.Net.Http;
using System.Threading;
using System.Threading.Tasks;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Localization;
using Shouldly;
using Volo.Abp.Http.Client;
using Volo.Abp.Http.Localization;
using Volo.Abp.Localization;
using Xunit;
@ -159,5 +159,17 @@ namespace Volo.Abp.Http.DynamicProxying
(await _controller.DeleteByIdAsync(42)).ShouldBe(43);
}
[Fact]
public async Task AbortRequestAsync()
{
var cts = new CancellationTokenSource();
cts.CancelAfter(10);
var result = await _controller.AbortRequestAsync(default);
result.ShouldBe("AbortRequestAsync");
var exception = await Assert.ThrowsAsync<HttpRequestException>(async () => await _controller.AbortRequestAsync(cts.Token));
exception.InnerException.InnerException.Message.ShouldBe("The client aborted the request.");
}
}
}

@ -0,0 +1,76 @@
using System.Collections.Generic;
using System.Linq;
using Shouldly;
using Volo.Abp.Data;
using Volo.Abp.ObjectExtending;
using Xunit;
namespace Volo.Abp.Json
{
public class AbpHasExtraPropertiesJsonConverter_Tests: AbpJsonTestBase
{
private readonly IJsonSerializer _jsonSerializer;
public AbpHasExtraPropertiesJsonConverter_Tests()
{
_jsonSerializer = GetRequiredService<IJsonSerializer>();
}
[Fact]
public void JsonConverter_Test()
{
var fooDto = new FooDto
{
Name = "foo-dto",
BarDtos = new List<BarDto>()
};
fooDto.SetProperty("foo", "foo-value");
var barDto = new BarDto
{
Name = "bar-dto"
};
barDto.SetProperty("bar", "bar-value");
fooDto.BarDtos.Add(barDto);
var json = _jsonSerializer.Serialize(fooDto);
fooDto = _jsonSerializer.Deserialize<FooDto>(json);
fooDto.ShouldNotBeNull();
fooDto.Name.ShouldBe("foo-dto");
fooDto.GetProperty("foo").ShouldBe("foo-value");
fooDto.BarDtos.Count.ShouldBe(1);
fooDto.BarDtos.First().Name.ShouldBe("bar-dto");
fooDto.BarDtos.First().GetProperty("bar").ShouldBe("bar-value");
fooDto.Name = "new-foo-dto";
fooDto.SetProperty("foo", "new-foo-value");
fooDto.BarDtos.First().Name = "new-bar-dto";
fooDto.BarDtos.First().SetProperty("bar", "new-bar-value");
json = _jsonSerializer.Serialize(fooDto);
fooDto = _jsonSerializer.Deserialize<FooDto>(json);
fooDto.ShouldNotBeNull();
fooDto.Name.ShouldBe("new-foo-dto");
fooDto.GetProperty("foo").ShouldBe("new-foo-value");
fooDto.BarDtos.Count.ShouldBe(1);
fooDto.BarDtos.First().Name.ShouldBe("new-bar-dto");
fooDto.BarDtos.First().GetProperty("bar").ShouldBe("new-bar-value");
}
}
public class FooDto : ExtensibleObject
{
public string Name { get; set; }
public List<BarDto> BarDtos { get; set; }
}
public class BarDto : ExtensibleObject
{
public string Name { get; set; }
}
}

@ -0,0 +1,16 @@
<Project Sdk="Microsoft.NET.Sdk">
<Import Project="..\..\..\common.test.props" />
<PropertyGroup>
<TargetFramework>net5.0</TargetFramework>
<RootNamespace />
</PropertyGroup>
<ItemGroup>
<ProjectReference Include="..\AbpTestBase\AbpTestBase.csproj" />
<ProjectReference Include="..\..\src\Volo.Abp.Threading\Volo.Abp.Threading.csproj" />
<PackageReference Include="Microsoft.NET.Test.Sdk" Version="$(MicrosoftNETTestSdkPackageVersion)" />
</ItemGroup>
</Project>

Some files were not shown because too many files have changed in this diff Show More

Loading…
Cancel
Save