First commit
This commit is contained in:
25
.dockerignore
Normal file
25
.dockerignore
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
**/.dockerignore
|
||||||
|
**/.env
|
||||||
|
**/.git
|
||||||
|
**/.gitignore
|
||||||
|
**/.project
|
||||||
|
**/.settings
|
||||||
|
**/.toolstarget
|
||||||
|
**/.vs
|
||||||
|
**/.vscode
|
||||||
|
**/.idea
|
||||||
|
**/*.*proj.user
|
||||||
|
**/*.dbmdl
|
||||||
|
**/*.jfm
|
||||||
|
**/azds.yaml
|
||||||
|
**/bin
|
||||||
|
**/charts
|
||||||
|
**/docker-compose*
|
||||||
|
**/Dockerfile*
|
||||||
|
**/node_modules
|
||||||
|
**/npm-debug.log
|
||||||
|
**/obj
|
||||||
|
**/secrets.dev.yaml
|
||||||
|
**/values.dev.yaml
|
||||||
|
LICENSE
|
||||||
|
README.md
|
||||||
133
.gitignore
vendored
Normal file
133
.gitignore
vendored
Normal file
@@ -0,0 +1,133 @@
|
|||||||
|
## Ignore Visual Studio temporary files, build results, and
|
||||||
|
## files generated by popular Visual Studio add-ons.
|
||||||
|
|
||||||
|
# User-specific files
|
||||||
|
*.suo
|
||||||
|
*.user
|
||||||
|
*.sln.docstates
|
||||||
|
|
||||||
|
# Build results
|
||||||
|
|
||||||
|
[Dd]ebug/
|
||||||
|
[Rr]elease/
|
||||||
|
x64/
|
||||||
|
[Bb]in/
|
||||||
|
[Oo]bj/
|
||||||
|
|
||||||
|
# MSTest test Results
|
||||||
|
[Tt]est[Rr]esult*/
|
||||||
|
[Bb]uild[Ll]og.*
|
||||||
|
|
||||||
|
*_i.c
|
||||||
|
*_p.c
|
||||||
|
*_i.h
|
||||||
|
*.ilk
|
||||||
|
*.meta
|
||||||
|
*.obj
|
||||||
|
*.pch
|
||||||
|
*.pdb
|
||||||
|
*.pgc
|
||||||
|
*.pgd
|
||||||
|
*.rsp
|
||||||
|
*.sbr
|
||||||
|
*.tlb
|
||||||
|
*.tli
|
||||||
|
*.tlh
|
||||||
|
*.tmp
|
||||||
|
*.tmp_proj
|
||||||
|
*.log
|
||||||
|
*.vspscc
|
||||||
|
*.vssscc
|
||||||
|
.builds
|
||||||
|
*.pidb
|
||||||
|
*.log
|
||||||
|
*.svclog
|
||||||
|
*.scc
|
||||||
|
|
||||||
|
# Visual C++ cache files
|
||||||
|
ipch/
|
||||||
|
*.aps
|
||||||
|
*.ncb
|
||||||
|
*.opensdf
|
||||||
|
*.sdf
|
||||||
|
*.cachefile
|
||||||
|
|
||||||
|
# Visual Studio profiler
|
||||||
|
*.psess
|
||||||
|
*.vsp
|
||||||
|
*.vspx
|
||||||
|
|
||||||
|
# Guidance Automation Toolkit
|
||||||
|
*.gpState
|
||||||
|
|
||||||
|
# ReSharper is a .NET coding add-in
|
||||||
|
_ReSharper*/
|
||||||
|
*.[Rr]e[Ss]harper
|
||||||
|
*.DotSettings.user
|
||||||
|
|
||||||
|
# Click-Once directory
|
||||||
|
publish/
|
||||||
|
|
||||||
|
# Publish Web Output
|
||||||
|
*.Publish.xml
|
||||||
|
*.pubxml
|
||||||
|
*.azurePubxml
|
||||||
|
|
||||||
|
# NuGet Packages Directory
|
||||||
|
## TODO: If you have NuGet Package Restore enabled, uncomment the next line
|
||||||
|
packages/
|
||||||
|
## TODO: If the tool you use requires repositories.config, also uncomment the next line
|
||||||
|
!packages/repositories.config
|
||||||
|
|
||||||
|
# Windows Azure Build Output
|
||||||
|
csx/
|
||||||
|
*.build.csdef
|
||||||
|
|
||||||
|
# Windows Store app package directory
|
||||||
|
AppPackages/
|
||||||
|
|
||||||
|
# Others
|
||||||
|
sql/
|
||||||
|
*.Cache
|
||||||
|
ClientBin/
|
||||||
|
[Ss]tyle[Cc]op.*
|
||||||
|
![Ss]tyle[Cc]op.targets
|
||||||
|
~$*
|
||||||
|
*~
|
||||||
|
*.dbmdl
|
||||||
|
*.[Pp]ublish.xml
|
||||||
|
|
||||||
|
*.publishsettings
|
||||||
|
|
||||||
|
# RIA/Silverlight projects
|
||||||
|
Generated_Code/
|
||||||
|
|
||||||
|
# Backup & report files from converting an old project file to a newer
|
||||||
|
# Visual Studio version. Backup files are not needed, because we have git ;-)
|
||||||
|
_UpgradeReport_Files/
|
||||||
|
Backup*/
|
||||||
|
UpgradeLog*.XML
|
||||||
|
UpgradeLog*.htm
|
||||||
|
|
||||||
|
# SQL Server files
|
||||||
|
App_Data/*.mdf
|
||||||
|
App_Data/*.ldf
|
||||||
|
|
||||||
|
# =========================
|
||||||
|
# Windows detritus
|
||||||
|
# =========================
|
||||||
|
|
||||||
|
# Windows image file caches
|
||||||
|
Thumbs.db
|
||||||
|
ehthumbs.db
|
||||||
|
|
||||||
|
# Folder config file
|
||||||
|
Desktop.ini
|
||||||
|
|
||||||
|
# Recycle Bin used on file shares
|
||||||
|
$RECYCLE.BIN/
|
||||||
|
|
||||||
|
# Mac desktop service store files
|
||||||
|
.DS_Store
|
||||||
|
|
||||||
|
_NCrunch*
|
||||||
13
.idea/.idea.ITS.Cled.Ripasso/.idea/.gitignore
generated
vendored
Normal file
13
.idea/.idea.ITS.Cled.Ripasso/.idea/.gitignore
generated
vendored
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
# Default ignored files
|
||||||
|
/shelf/
|
||||||
|
/workspace.xml
|
||||||
|
# Rider ignored files
|
||||||
|
/modules.xml
|
||||||
|
/.idea.ITS.Cled.Ripasso.iml
|
||||||
|
/projectSettingsUpdater.xml
|
||||||
|
/contentModel.xml
|
||||||
|
# Editor-based HTTP Client requests
|
||||||
|
/httpRequests/
|
||||||
|
# Datasource local storage ignored files
|
||||||
|
/dataSources/
|
||||||
|
/dataSources.local.xml
|
||||||
7
.idea/.idea.ITS.Cled.Ripasso/.idea/CSharpierPlugin.xml
generated
Normal file
7
.idea/.idea.ITS.Cled.Ripasso/.idea/CSharpierPlugin.xml
generated
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<project version="4">
|
||||||
|
<component name="com.intellij.csharpier">
|
||||||
|
<option name="customPath" value="" />
|
||||||
|
<option name="runOnSave" value="true" />
|
||||||
|
</component>
|
||||||
|
</project>
|
||||||
12
.idea/.idea.ITS.Cled.Ripasso/.idea/dataSources.xml
generated
Normal file
12
.idea/.idea.ITS.Cled.Ripasso/.idea/dataSources.xml
generated
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<project version="4">
|
||||||
|
<component name="DataSourceManagerImpl" format="xml" multifile-model="true">
|
||||||
|
<data-source source="LOCAL" name="ITS@192.168.1.143" uuid="14a8e455-a937-4648-8c40-63fcfcc23e09">
|
||||||
|
<driver-ref>postgresql</driver-ref>
|
||||||
|
<synchronize>true</synchronize>
|
||||||
|
<jdbc-driver>org.postgresql.Driver</jdbc-driver>
|
||||||
|
<jdbc-url>jdbc:postgresql://192.168.1.143:5432/ITS</jdbc-url>
|
||||||
|
<working-dir>$ProjectFileDir$</working-dir>
|
||||||
|
</data-source>
|
||||||
|
</component>
|
||||||
|
</project>
|
||||||
8
.idea/.idea.ITS.Cled.Ripasso/.idea/indexLayout.xml
generated
Normal file
8
.idea/.idea.ITS.Cled.Ripasso/.idea/indexLayout.xml
generated
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<project version="4">
|
||||||
|
<component name="UserContentModel">
|
||||||
|
<attachedFolders />
|
||||||
|
<explicitIncludes />
|
||||||
|
<explicitExcludes />
|
||||||
|
</component>
|
||||||
|
</project>
|
||||||
12
.idea/.idea.ITS.Cled.Ripasso/.idea/inspectionProfiles/Project_Default.xml
generated
Normal file
12
.idea/.idea.ITS.Cled.Ripasso/.idea/inspectionProfiles/Project_Default.xml
generated
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
<component name="InspectionProjectProfileManager">
|
||||||
|
<profile version="1.0">
|
||||||
|
<option name="myName" value="Project Default" />
|
||||||
|
<inspection_tool class="PyPep8Inspection" enabled="true" level="WEAK WARNING" enabled_by_default="true">
|
||||||
|
<option name="ignoredErrors">
|
||||||
|
<list>
|
||||||
|
<option value="E305" />
|
||||||
|
</list>
|
||||||
|
</option>
|
||||||
|
</inspection_tool>
|
||||||
|
</profile>
|
||||||
|
</component>
|
||||||
13
.idea/.idea.ITS.Cled.Ripasso/.idea/misc.xml
generated
Normal file
13
.idea/.idea.ITS.Cled.Ripasso/.idea/misc.xml
generated
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<project version="4">
|
||||||
|
<component name="MaterialThemeProjectNewConfig">
|
||||||
|
<option name="metadata">
|
||||||
|
<MTProjectMetadataState>
|
||||||
|
<option name="migrated" value="true" />
|
||||||
|
<option name="pristineConfig" value="false" />
|
||||||
|
<option name="userId" value="-53a01828:18ed87172e0:-7ffe" />
|
||||||
|
<option name="version" value="8.13.2" />
|
||||||
|
</MTProjectMetadataState>
|
||||||
|
</option>
|
||||||
|
</component>
|
||||||
|
</project>
|
||||||
7
.idea/.idea.ITS.Cled.Ripasso/.idea/sqldialects.xml
generated
Normal file
7
.idea/.idea.ITS.Cled.Ripasso/.idea/sqldialects.xml
generated
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<project version="4">
|
||||||
|
<component name="SqlDialectMappings">
|
||||||
|
<file url="file://$PROJECT_DIR$/ITS.Cled.Ripasso/Services/ProductsDataService.cs" dialect="GenericSQL" />
|
||||||
|
<file url="PROJECT" dialect="PostgreSQL" />
|
||||||
|
</component>
|
||||||
|
</project>
|
||||||
6
.idea/.idea.ITS.Cled.Ripasso/.idea/vcs.xml
generated
Normal file
6
.idea/.idea.ITS.Cled.Ripasso/.idea/vcs.xml
generated
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<project version="4">
|
||||||
|
<component name="VcsDirectoryMappings">
|
||||||
|
<mapping directory="$PROJECT_DIR$" vcs="Git" />
|
||||||
|
</component>
|
||||||
|
</project>
|
||||||
22
ITS.Cled.Ripasso.sln
Normal file
22
ITS.Cled.Ripasso.sln
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
|
||||||
|
Microsoft Visual Studio Solution File, Format Version 12.00
|
||||||
|
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "ITS.Cled.Ripasso", "ITS.Cled.Ripasso\ITS.Cled.Ripasso.csproj", "{5C7DB573-A29E-46BE-8D37-B1E7201EEC86}"
|
||||||
|
EndProject
|
||||||
|
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Receiver", "Receiver\Receiver.csproj", "{397ACECB-96A0-492C-B161-93BF4C3831C9}"
|
||||||
|
EndProject
|
||||||
|
Global
|
||||||
|
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
||||||
|
Debug|Any CPU = Debug|Any CPU
|
||||||
|
Release|Any CPU = Release|Any CPU
|
||||||
|
EndGlobalSection
|
||||||
|
GlobalSection(ProjectConfigurationPlatforms) = postSolution
|
||||||
|
{5C7DB573-A29E-46BE-8D37-B1E7201EEC86}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||||
|
{5C7DB573-A29E-46BE-8D37-B1E7201EEC86}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||||
|
{5C7DB573-A29E-46BE-8D37-B1E7201EEC86}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||||
|
{5C7DB573-A29E-46BE-8D37-B1E7201EEC86}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||||
|
{397ACECB-96A0-492C-B161-93BF4C3831C9}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||||
|
{397ACECB-96A0-492C-B161-93BF4C3831C9}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||||
|
{397ACECB-96A0-492C-B161-93BF4C3831C9}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||||
|
{397ACECB-96A0-492C-B161-93BF4C3831C9}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||||
|
EndGlobalSection
|
||||||
|
EndGlobal
|
||||||
20
ITS.Cled.Ripasso/Components/App.razor
Normal file
20
ITS.Cled.Ripasso/Components/App.razor
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="en">
|
||||||
|
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8"/>
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1.0"/>
|
||||||
|
<base href="/"/>
|
||||||
|
<link rel="stylesheet" href="bootstrap/bootstrap.min.css"/>
|
||||||
|
<link rel="stylesheet" href="app.css"/>
|
||||||
|
<link rel="stylesheet" href="ITS.Cled.Ripasso.styles.css"/>
|
||||||
|
<link rel="icon" type="image/png" href="favicon.png"/>
|
||||||
|
<HeadOutlet @rendermode="InteractiveServer"/>
|
||||||
|
</head>
|
||||||
|
|
||||||
|
<body>
|
||||||
|
<Routes @rendermode="InteractiveServer"/>
|
||||||
|
<script src="_framework/blazor.web.js"></script>
|
||||||
|
</body>
|
||||||
|
|
||||||
|
</html>
|
||||||
23
ITS.Cled.Ripasso/Components/Layout/MainLayout.razor
Normal file
23
ITS.Cled.Ripasso/Components/Layout/MainLayout.razor
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
@inherits LayoutComponentBase
|
||||||
|
|
||||||
|
<div class="page">
|
||||||
|
<div class="sidebar">
|
||||||
|
<NavMenu/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<main>
|
||||||
|
<div class="top-row px-4">
|
||||||
|
<a href="https://learn.microsoft.com/aspnet/core/" target="_blank">About</a>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<article class="content px-4">
|
||||||
|
@Body
|
||||||
|
</article>
|
||||||
|
</main>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div id="blazor-error-ui">
|
||||||
|
An unhandled error has occurred.
|
||||||
|
<a href="" class="reload">Reload</a>
|
||||||
|
<a class="dismiss">🗙</a>
|
||||||
|
</div>
|
||||||
96
ITS.Cled.Ripasso/Components/Layout/MainLayout.razor.css
Normal file
96
ITS.Cled.Ripasso/Components/Layout/MainLayout.razor.css
Normal file
@@ -0,0 +1,96 @@
|
|||||||
|
.page {
|
||||||
|
position: relative;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
}
|
||||||
|
|
||||||
|
main {
|
||||||
|
flex: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.sidebar {
|
||||||
|
background-image: linear-gradient(180deg, rgb(5, 39, 103) 0%, #3a0647 70%);
|
||||||
|
}
|
||||||
|
|
||||||
|
.top-row {
|
||||||
|
background-color: #f7f7f7;
|
||||||
|
border-bottom: 1px solid #d6d5d5;
|
||||||
|
justify-content: flex-end;
|
||||||
|
height: 3.5rem;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.top-row ::deep a, .top-row ::deep .btn-link {
|
||||||
|
white-space: nowrap;
|
||||||
|
margin-left: 1.5rem;
|
||||||
|
text-decoration: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.top-row ::deep a:hover, .top-row ::deep .btn-link:hover {
|
||||||
|
text-decoration: underline;
|
||||||
|
}
|
||||||
|
|
||||||
|
.top-row ::deep a:first-child {
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (max-width: 640.98px) {
|
||||||
|
.top-row {
|
||||||
|
justify-content: space-between;
|
||||||
|
}
|
||||||
|
|
||||||
|
.top-row ::deep a, .top-row ::deep .btn-link {
|
||||||
|
margin-left: 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (min-width: 641px) {
|
||||||
|
.page {
|
||||||
|
flex-direction: row;
|
||||||
|
}
|
||||||
|
|
||||||
|
.sidebar {
|
||||||
|
width: 250px;
|
||||||
|
height: 100vh;
|
||||||
|
position: sticky;
|
||||||
|
top: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.top-row {
|
||||||
|
position: sticky;
|
||||||
|
top: 0;
|
||||||
|
z-index: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.top-row.auth ::deep a:first-child {
|
||||||
|
flex: 1;
|
||||||
|
text-align: right;
|
||||||
|
width: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.top-row, article {
|
||||||
|
padding-left: 2rem !important;
|
||||||
|
padding-right: 1.5rem !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#blazor-error-ui {
|
||||||
|
background: lightyellow;
|
||||||
|
bottom: 0;
|
||||||
|
box-shadow: 0 -1px 2px rgba(0, 0, 0, 0.2);
|
||||||
|
display: none;
|
||||||
|
left: 0;
|
||||||
|
padding: 0.6rem 1.25rem 0.7rem 1.25rem;
|
||||||
|
position: fixed;
|
||||||
|
width: 100%;
|
||||||
|
z-index: 1000;
|
||||||
|
}
|
||||||
|
|
||||||
|
#blazor-error-ui .dismiss {
|
||||||
|
cursor: pointer;
|
||||||
|
position: absolute;
|
||||||
|
right: 0.75rem;
|
||||||
|
top: 0.5rem;
|
||||||
|
}
|
||||||
29
ITS.Cled.Ripasso/Components/Layout/NavMenu.razor
Normal file
29
ITS.Cled.Ripasso/Components/Layout/NavMenu.razor
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
<div class="top-row ps-3 navbar navbar-dark">
|
||||||
|
<div class="container-fluid">
|
||||||
|
<a class="navbar-brand" href="">ITS.Cled.Ripasso</a>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<input type="checkbox" title="Navigation menu" class="navbar-toggler"/>
|
||||||
|
|
||||||
|
<div class="nav-scrollable" onclick="document.querySelector('.navbar-toggler').click()">
|
||||||
|
<nav class="flex-column">
|
||||||
|
<div class="nav-item px-3">
|
||||||
|
<NavLink class="nav-link" href="" Match="NavLinkMatch.All">
|
||||||
|
<span class="bi bi-house-door-fill-nav-menu" aria-hidden="true"></span> Home
|
||||||
|
</NavLink>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="nav-item px-3">
|
||||||
|
<NavLink class="nav-link" href="counter">
|
||||||
|
<span class="bi bi-plus-square-fill-nav-menu" aria-hidden="true"></span> Counter
|
||||||
|
</NavLink>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="nav-item px-3">
|
||||||
|
<NavLink class="nav-link" href="products">
|
||||||
|
<span class="bi bi-list-nested-nav-menu" aria-hidden="true"></span> Products
|
||||||
|
</NavLink>
|
||||||
|
</div>
|
||||||
|
</nav>
|
||||||
|
</div>
|
||||||
105
ITS.Cled.Ripasso/Components/Layout/NavMenu.razor.css
Normal file
105
ITS.Cled.Ripasso/Components/Layout/NavMenu.razor.css
Normal file
@@ -0,0 +1,105 @@
|
|||||||
|
.navbar-toggler {
|
||||||
|
appearance: none;
|
||||||
|
cursor: pointer;
|
||||||
|
width: 3.5rem;
|
||||||
|
height: 2.5rem;
|
||||||
|
color: white;
|
||||||
|
position: absolute;
|
||||||
|
top: 0.5rem;
|
||||||
|
right: 1rem;
|
||||||
|
border: 1px solid rgba(255, 255, 255, 0.1);
|
||||||
|
background: url("data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 30 30'%3e%3cpath stroke='rgba%28255, 255, 255, 0.55%29' stroke-linecap='round' stroke-miterlimit='10' stroke-width='2' d='M4 7h22M4 15h22M4 23h22'/%3e%3c/svg%3e") no-repeat center/1.75rem rgba(255, 255, 255, 0.1);
|
||||||
|
}
|
||||||
|
|
||||||
|
.navbar-toggler:checked {
|
||||||
|
background-color: rgba(255, 255, 255, 0.5);
|
||||||
|
}
|
||||||
|
|
||||||
|
.top-row {
|
||||||
|
height: 3.5rem;
|
||||||
|
background-color: rgba(0,0,0,0.4);
|
||||||
|
}
|
||||||
|
|
||||||
|
.navbar-brand {
|
||||||
|
font-size: 1.1rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.bi {
|
||||||
|
display: inline-block;
|
||||||
|
position: relative;
|
||||||
|
width: 1.25rem;
|
||||||
|
height: 1.25rem;
|
||||||
|
margin-right: 0.75rem;
|
||||||
|
top: -1px;
|
||||||
|
background-size: cover;
|
||||||
|
}
|
||||||
|
|
||||||
|
.bi-house-door-fill-nav-menu {
|
||||||
|
background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='16' height='16' fill='white' class='bi bi-house-door-fill' viewBox='0 0 16 16'%3E%3Cpath d='M6.5 14.5v-3.505c0-.245.25-.495.5-.495h2c.25 0 .5.25.5.5v3.5a.5.5 0 0 0 .5.5h4a.5.5 0 0 0 .5-.5v-7a.5.5 0 0 0-.146-.354L13 5.793V2.5a.5.5 0 0 0-.5-.5h-1a.5.5 0 0 0-.5.5v1.293L8.354 1.146a.5.5 0 0 0-.708 0l-6 6A.5.5 0 0 0 1.5 7.5v7a.5.5 0 0 0 .5.5h4a.5.5 0 0 0 .5-.5Z'/%3E%3C/svg%3E");
|
||||||
|
}
|
||||||
|
|
||||||
|
.bi-plus-square-fill-nav-menu {
|
||||||
|
background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='16' height='16' fill='white' class='bi bi-plus-square-fill' viewBox='0 0 16 16'%3E%3Cpath d='M2 0a2 2 0 0 0-2 2v12a2 2 0 0 0 2 2h12a2 2 0 0 0 2-2V2a2 2 0 0 0-2-2H2zm6.5 4.5v3h3a.5.5 0 0 1 0 1h-3v3a.5.5 0 0 1-1 0v-3h-3a.5.5 0 0 1 0-1h3v-3a.5.5 0 0 1 1 0z'/%3E%3C/svg%3E");
|
||||||
|
}
|
||||||
|
|
||||||
|
.bi-list-nested-nav-menu {
|
||||||
|
background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' width='16' height='16' fill='white' class='bi bi-list-nested' viewBox='0 0 16 16'%3E%3Cpath fill-rule='evenodd' d='M4.5 11.5A.5.5 0 0 1 5 11h10a.5.5 0 0 1 0 1H5a.5.5 0 0 1-.5-.5zm-2-4A.5.5 0 0 1 3 7h10a.5.5 0 0 1 0 1H3a.5.5 0 0 1-.5-.5zm-2-4A.5.5 0 0 1 1 3h10a.5.5 0 0 1 0 1H1a.5.5 0 0 1-.5-.5z'/%3E%3C/svg%3E");
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav-item {
|
||||||
|
font-size: 0.9rem;
|
||||||
|
padding-bottom: 0.5rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav-item:first-of-type {
|
||||||
|
padding-top: 1rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav-item:last-of-type {
|
||||||
|
padding-bottom: 1rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav-item ::deep .nav-link {
|
||||||
|
color: #d7d7d7;
|
||||||
|
background: none;
|
||||||
|
border: none;
|
||||||
|
border-radius: 4px;
|
||||||
|
height: 3rem;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
line-height: 3rem;
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav-item ::deep a.active {
|
||||||
|
background-color: rgba(255,255,255,0.37);
|
||||||
|
color: white;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav-item ::deep .nav-link:hover {
|
||||||
|
background-color: rgba(255,255,255,0.1);
|
||||||
|
color: white;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav-scrollable {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.navbar-toggler:checked ~ .nav-scrollable {
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (min-width: 641px) {
|
||||||
|
.navbar-toggler {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav-scrollable {
|
||||||
|
/* Never collapse the sidebar for wide screens */
|
||||||
|
display: block;
|
||||||
|
|
||||||
|
/* Allow sidebar to scroll for tall menus */
|
||||||
|
height: calc(100vh - 3.5rem);
|
||||||
|
overflow-y: auto;
|
||||||
|
}
|
||||||
|
}
|
||||||
19
ITS.Cled.Ripasso/Components/Pages/Counter.razor
Normal file
19
ITS.Cled.Ripasso/Components/Pages/Counter.razor
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
@page "/counter"
|
||||||
|
|
||||||
|
<PageTitle>Counter</PageTitle>
|
||||||
|
|
||||||
|
<h1>Counter</h1>
|
||||||
|
|
||||||
|
<p role="status">Current count: @currentCount</p>
|
||||||
|
|
||||||
|
<button class="btn btn-primary" @onclick="IncrementCount">Click me</button>
|
||||||
|
|
||||||
|
@code {
|
||||||
|
private int currentCount = 0;
|
||||||
|
|
||||||
|
private void IncrementCount()
|
||||||
|
{
|
||||||
|
currentCount++;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
36
ITS.Cled.Ripasso/Components/Pages/Error.razor
Normal file
36
ITS.Cled.Ripasso/Components/Pages/Error.razor
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
@page "/Error"
|
||||||
|
@using System.Diagnostics
|
||||||
|
|
||||||
|
<PageTitle>Error</PageTitle>
|
||||||
|
|
||||||
|
<h1 class="text-danger">Error.</h1>
|
||||||
|
<h2 class="text-danger">An error occurred while processing your request.</h2>
|
||||||
|
|
||||||
|
@if (ShowRequestId)
|
||||||
|
{
|
||||||
|
<p>
|
||||||
|
<strong>Request ID:</strong> <code>@RequestId</code>
|
||||||
|
</p>
|
||||||
|
}
|
||||||
|
|
||||||
|
<h3>Development Mode</h3>
|
||||||
|
<p>
|
||||||
|
Swapping to <strong>Development</strong> environment will display more detailed information about the error that occurred.
|
||||||
|
</p>
|
||||||
|
<p>
|
||||||
|
<strong>The Development environment shouldn't be enabled for deployed applications.</strong>
|
||||||
|
It can result in displaying sensitive information from exceptions to end users.
|
||||||
|
For local debugging, enable the <strong>Development</strong> environment by setting the <strong>ASPNETCORE_ENVIRONMENT</strong> environment variable to <strong>Development</strong>
|
||||||
|
and restarting the app.
|
||||||
|
</p>
|
||||||
|
|
||||||
|
@code{
|
||||||
|
[CascadingParameter] private HttpContext? HttpContext { get; set; }
|
||||||
|
|
||||||
|
private string? RequestId { get; set; }
|
||||||
|
private bool ShowRequestId => !string.IsNullOrEmpty(RequestId);
|
||||||
|
|
||||||
|
protected override void OnInitialized() =>
|
||||||
|
RequestId = Activity.Current?.Id ?? HttpContext?.TraceIdentifier;
|
||||||
|
|
||||||
|
}
|
||||||
7
ITS.Cled.Ripasso/Components/Pages/Home.razor
Normal file
7
ITS.Cled.Ripasso/Components/Pages/Home.razor
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
@page "/"
|
||||||
|
|
||||||
|
<PageTitle>Home</PageTitle>
|
||||||
|
|
||||||
|
<h1>Hello, world!</h1>
|
||||||
|
|
||||||
|
Welcome to your new app.
|
||||||
48
ITS.Cled.Ripasso/Components/Pages/Products/Create.razor
Normal file
48
ITS.Cled.Ripasso/Components/Pages/Products/Create.razor
Normal file
@@ -0,0 +1,48 @@
|
|||||||
|
@page "/products/create"
|
||||||
|
@using ITS.Cled.Ripasso.Services
|
||||||
|
@using ITS.Cled.Ripasso.Model
|
||||||
|
@using ITS.Cled.Ripasso.Messages
|
||||||
|
|
||||||
|
@inject IProductDataService Data
|
||||||
|
@inject MessageSender MS
|
||||||
|
@inject NavigationManager Nav
|
||||||
|
|
||||||
|
<h3>Create New Product</h3>
|
||||||
|
|
||||||
|
<EditForm Model="Input" OnValidSubmit="CreateProduct">
|
||||||
|
<DataAnnotationsValidator />
|
||||||
|
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="Name">Name</label>
|
||||||
|
<InputText id="Name" class="form-control" @bind-Value="Input.Name" />
|
||||||
|
<ValidationMessage For="@(() => Input.Name)" />
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="Code">Code</label>
|
||||||
|
<InputText id="Code" class="form-control" @bind-Value="Input.Code" />
|
||||||
|
<ValidationMessage For="@(() => Input.Code)" />
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="Price">Price</label>
|
||||||
|
<InputNumber id="Price" class="form-control" @bind-Value="Input.Price" />
|
||||||
|
<ValidationMessage For="@(() => Input.Price)" />
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<a href="/products" class="btn btn-secondary">Undo</a>
|
||||||
|
<button type="submit" class="btn btn-primary mx-3">Save</button>
|
||||||
|
|
||||||
|
</EditForm>
|
||||||
|
|
||||||
|
@code {
|
||||||
|
public Product Input {get; set;} = new();
|
||||||
|
|
||||||
|
private async Task CreateProduct()
|
||||||
|
{
|
||||||
|
await Data.CreateProduct(Input);
|
||||||
|
Input.Action = "Create";
|
||||||
|
await MS.Send(Input);
|
||||||
|
Nav.NavigateTo("/products");
|
||||||
|
}
|
||||||
|
}
|
||||||
68
ITS.Cled.Ripasso/Components/Pages/Products/Index.razor
Normal file
68
ITS.Cled.Ripasso/Components/Pages/Products/Index.razor
Normal file
@@ -0,0 +1,68 @@
|
|||||||
|
@page "/products"
|
||||||
|
@using ITS.Cled.Ripasso.Messages
|
||||||
|
@using ITS.Cled.Ripasso.Model
|
||||||
|
@using ITS.Cled.Ripasso.Services
|
||||||
|
|
||||||
|
@inject IProductDataService Data
|
||||||
|
@inject IJSRuntime Js
|
||||||
|
@inject NavigationManager Nav
|
||||||
|
@inject MessageSender MS
|
||||||
|
|
||||||
|
|
||||||
|
<h3>Prodotti</h3>
|
||||||
|
|
||||||
|
<a href="/products/create" class="btn btn-primary mb-3">Create New Product</a>
|
||||||
|
|
||||||
|
<table class="table table-striped">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th>Id</th>
|
||||||
|
<th>Nome</th>
|
||||||
|
<th>Codice</th>
|
||||||
|
<th>Prezzo</th>
|
||||||
|
<th></th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
@foreach (var product in _products)
|
||||||
|
{
|
||||||
|
<tr>
|
||||||
|
<td>@product.Id</td>
|
||||||
|
<td>@product.Name</td>
|
||||||
|
<td>@product.Code</td>
|
||||||
|
<td>@product.Price</td>
|
||||||
|
<td>
|
||||||
|
<a href="/products/update/@product.Id" class="btn btn-warning btn-sm">Edit</a>
|
||||||
|
<button class="btn btn-danger btn-sm ms-2"
|
||||||
|
@onclick="() => DeleteProduct(product)">Delete
|
||||||
|
</button>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
@code {
|
||||||
|
private IEnumerable<Product> _products = [];
|
||||||
|
|
||||||
|
protected override async Task OnInitializedAsync()
|
||||||
|
{
|
||||||
|
_products = await Data.GetProductsAsync();
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task DeleteProduct(Product product)
|
||||||
|
{
|
||||||
|
var result = await Js.InvokeAsync<bool>("confirm", "Are you sure you want to delete this product?");
|
||||||
|
if (result)
|
||||||
|
{
|
||||||
|
Product productdel = new Product();
|
||||||
|
productdel = product;
|
||||||
|
productdel.Action = "Delete";
|
||||||
|
await Data.DeleteProduct(product.Id);
|
||||||
|
await MS.Send(productdel);
|
||||||
|
Nav.Refresh(forceReload:true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
63
ITS.Cled.Ripasso/Components/Pages/Products/Update.razor
Normal file
63
ITS.Cled.Ripasso/Components/Pages/Products/Update.razor
Normal file
@@ -0,0 +1,63 @@
|
|||||||
|
@page "/products/update/{id:int}"
|
||||||
|
@using ITS.Cled.Ripasso.Services
|
||||||
|
@using ITS.Cled.Ripasso.Model
|
||||||
|
@using ITS.Cled.Ripasso.Messages
|
||||||
|
|
||||||
|
@inject IProductDataService Data
|
||||||
|
@inject NavigationManager Nav
|
||||||
|
@inject MessageSender MS
|
||||||
|
<h3>Update Product</h3>
|
||||||
|
|
||||||
|
<EditForm Model="Input" OnValidSubmit="SaveProduct">
|
||||||
|
<DataAnnotationsValidator />
|
||||||
|
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="Name">Name</label>
|
||||||
|
<InputText id="Name" class="form-control" @bind-Value="Input.Name" />
|
||||||
|
<ValidationMessage For="@(() => Input.Name)" />
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="Code">Code</label>
|
||||||
|
<InputText id="Code" class="form-control" @bind-Value="Input.Code" />
|
||||||
|
<ValidationMessage For="@(() => Input.Code)" />
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="form-group">
|
||||||
|
<label for="Price">Price</label>
|
||||||
|
<InputNumber id="Price" class="form-control" @bind-Value="Input.Price" />
|
||||||
|
<ValidationMessage For="@(() => Input.Price)" />
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<a href="/products" class="btn btn-secondary">Undo</a>
|
||||||
|
<button type="submit" class="btn btn-primary mx-3">Save</button>
|
||||||
|
|
||||||
|
</EditForm>
|
||||||
|
|
||||||
|
@code {
|
||||||
|
[Parameter]
|
||||||
|
public int Id {get; set;}
|
||||||
|
|
||||||
|
public Product Input {get; set;} = new();
|
||||||
|
|
||||||
|
protected override async Task OnInitializedAsync()
|
||||||
|
{
|
||||||
|
var product = await Data.GetProductById(Id);
|
||||||
|
if (product == null)
|
||||||
|
{
|
||||||
|
throw new Exception("Product Not Found");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
Input = product;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private async Task SaveProduct()
|
||||||
|
{
|
||||||
|
await Data.UpdateProduct(Input);
|
||||||
|
Input.Action = "Update";
|
||||||
|
await MS.Send(Input);
|
||||||
|
Nav.NavigateTo("/products");
|
||||||
|
}
|
||||||
|
}
|
||||||
66
ITS.Cled.Ripasso/Components/Pages/Weather.razor
Normal file
66
ITS.Cled.Ripasso/Components/Pages/Weather.razor
Normal file
@@ -0,0 +1,66 @@
|
|||||||
|
@page "/weather"
|
||||||
|
|
||||||
|
<PageTitle>Weather</PageTitle>
|
||||||
|
|
||||||
|
<h1>Weather</h1>
|
||||||
|
|
||||||
|
<p>This component demonstrates showing data.</p>
|
||||||
|
|
||||||
|
@if (forecasts == null)
|
||||||
|
{
|
||||||
|
<p>
|
||||||
|
<em>Loading...</em>
|
||||||
|
</p>
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
<table class="table">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th>Date</th>
|
||||||
|
<th>Temp. (C)</th>
|
||||||
|
<th>Temp. (F)</th>
|
||||||
|
<th>Summary</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
@foreach (var forecast in forecasts)
|
||||||
|
{
|
||||||
|
<tr>
|
||||||
|
<td>@forecast.Date.ToShortDateString()</td>
|
||||||
|
<td>@forecast.TemperatureC</td>
|
||||||
|
<td>@forecast.TemperatureF</td>
|
||||||
|
<td>@forecast.Summary</td>
|
||||||
|
</tr>
|
||||||
|
}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
}
|
||||||
|
|
||||||
|
@code {
|
||||||
|
private WeatherForecast[]? forecasts;
|
||||||
|
|
||||||
|
protected override async Task OnInitializedAsync()
|
||||||
|
{
|
||||||
|
// Simulate asynchronous loading to demonstrate a loading indicator
|
||||||
|
await Task.Delay(500);
|
||||||
|
|
||||||
|
var startDate = DateOnly.FromDateTime(DateTime.Now);
|
||||||
|
var summaries = new[] { "Freezing", "Bracing", "Chilly", "Cool", "Mild", "Warm", "Balmy", "Hot", "Sweltering", "Scorching" };
|
||||||
|
forecasts = Enumerable.Range(1, 5).Select(index => new WeatherForecast
|
||||||
|
{
|
||||||
|
Date = startDate.AddDays(index),
|
||||||
|
TemperatureC = Random.Shared.Next(-20, 55),
|
||||||
|
Summary = summaries[Random.Shared.Next(summaries.Length)]
|
||||||
|
}).ToArray();
|
||||||
|
}
|
||||||
|
|
||||||
|
private class WeatherForecast
|
||||||
|
{
|
||||||
|
public DateOnly Date { get; set; }
|
||||||
|
public int TemperatureC { get; set; }
|
||||||
|
public string? Summary { get; set; }
|
||||||
|
public int TemperatureF => 32 + (int)(TemperatureC / 0.5556);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
6
ITS.Cled.Ripasso/Components/Routes.razor
Normal file
6
ITS.Cled.Ripasso/Components/Routes.razor
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
<Router AppAssembly="typeof(Program).Assembly">
|
||||||
|
<Found Context="routeData">
|
||||||
|
<RouteView RouteData="routeData" DefaultLayout="typeof(Layout.MainLayout)"/>
|
||||||
|
<FocusOnNavigate RouteData="routeData" Selector="h1"/>
|
||||||
|
</Found>
|
||||||
|
</Router>
|
||||||
10
ITS.Cled.Ripasso/Components/_Imports.razor
Normal file
10
ITS.Cled.Ripasso/Components/_Imports.razor
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
@using System.Net.Http
|
||||||
|
@using System.Net.Http.Json
|
||||||
|
@using Microsoft.AspNetCore.Components.Forms
|
||||||
|
@using Microsoft.AspNetCore.Components.Routing
|
||||||
|
@using Microsoft.AspNetCore.Components.Web
|
||||||
|
@using static Microsoft.AspNetCore.Components.Web.RenderMode
|
||||||
|
@using Microsoft.AspNetCore.Components.Web.Virtualization
|
||||||
|
@using Microsoft.JSInterop
|
||||||
|
@using ITS.Cled.Ripasso
|
||||||
|
@using ITS.Cled.Ripasso.Components
|
||||||
23
ITS.Cled.Ripasso/Dockerfile
Normal file
23
ITS.Cled.Ripasso/Dockerfile
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
FROM mcr.microsoft.com/dotnet/aspnet:8.0 AS base
|
||||||
|
USER $APP_UID
|
||||||
|
WORKDIR /app
|
||||||
|
EXPOSE 8080
|
||||||
|
EXPOSE 8081
|
||||||
|
|
||||||
|
FROM mcr.microsoft.com/dotnet/sdk:8.0 AS build
|
||||||
|
ARG BUILD_CONFIGURATION=Release
|
||||||
|
WORKDIR /src
|
||||||
|
COPY ["ITS.Cled.Ripasso/ITS.Cled.Ripasso.csproj", "ITS.Cled.Ripasso/"]
|
||||||
|
RUN dotnet restore "ITS.Cled.Ripasso/ITS.Cled.Ripasso.csproj"
|
||||||
|
COPY . .
|
||||||
|
WORKDIR "/src/ITS.Cled.Ripasso"
|
||||||
|
RUN dotnet build "ITS.Cled.Ripasso.csproj" -c $BUILD_CONFIGURATION -o /app/build
|
||||||
|
|
||||||
|
FROM build AS publish
|
||||||
|
ARG BUILD_CONFIGURATION=Release
|
||||||
|
RUN dotnet publish "ITS.Cled.Ripasso.csproj" -c $BUILD_CONFIGURATION -o /app/publish /p:UseAppHost=false
|
||||||
|
|
||||||
|
FROM base AS final
|
||||||
|
WORKDIR /app
|
||||||
|
COPY --from=publish /app/publish .
|
||||||
|
ENTRYPOINT ["dotnet", "ITS.Cled.Ripasso.dll"]
|
||||||
72
ITS.Cled.Ripasso/Endpoints/ProductsEndpoints.cs
Normal file
72
ITS.Cled.Ripasso/Endpoints/ProductsEndpoints.cs
Normal file
@@ -0,0 +1,72 @@
|
|||||||
|
using ITS.Cled.Ripasso.Model;
|
||||||
|
using ITS.Cled.Ripasso.Services;
|
||||||
|
using Microsoft.AspNetCore.Http.HttpResults;
|
||||||
|
|
||||||
|
namespace ITS.Cled.Ripasso.Endpoints;
|
||||||
|
|
||||||
|
public static class ProductsEndpoints
|
||||||
|
{
|
||||||
|
public static IEndpointRouteBuilder MapProductsEndpoint(this IEndpointRouteBuilder app)
|
||||||
|
{
|
||||||
|
var group = app.MapGroup("api/products").WithOpenApi().WithTags("Products");
|
||||||
|
|
||||||
|
group.MapGet("/", GetProductsAsync);
|
||||||
|
group.MapGet("/{id:int}", GetProductAsync);
|
||||||
|
group.MapPost("/", InsertProductAsync);
|
||||||
|
group.MapPut("/{id:int}", UpdateProductAsync);
|
||||||
|
group.MapDelete("/{id:int}", DeleteProductAsync);
|
||||||
|
|
||||||
|
return app;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static async Task<Ok<IEnumerable<Product>>> GetProductsAsync(IProductDataService data)
|
||||||
|
{
|
||||||
|
var list = await data.GetProductsAsync();
|
||||||
|
|
||||||
|
return TypedResults.Ok(list);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static async Task<Results<Ok<Product>, NotFound>> GetProductAsync(
|
||||||
|
int id,
|
||||||
|
IProductDataService data
|
||||||
|
)
|
||||||
|
{
|
||||||
|
var product = await data.GetProductById(id);
|
||||||
|
|
||||||
|
if (product == null)
|
||||||
|
{
|
||||||
|
return TypedResults.NotFound();
|
||||||
|
}
|
||||||
|
|
||||||
|
return TypedResults.Ok(product);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static async Task<Created<Product>> InsertProductAsync(
|
||||||
|
Product product,
|
||||||
|
IProductDataService data
|
||||||
|
)
|
||||||
|
{
|
||||||
|
var newProduct = await data.CreateProduct(product);
|
||||||
|
|
||||||
|
return TypedResults.Created($"/api/products/{newProduct.Id}", newProduct);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static async Task<NoContent> UpdateProductAsync(
|
||||||
|
int id,
|
||||||
|
Product product,
|
||||||
|
IProductDataService data
|
||||||
|
)
|
||||||
|
{
|
||||||
|
product.Id = id;
|
||||||
|
await data.UpdateProduct(product);
|
||||||
|
|
||||||
|
return TypedResults.NoContent();
|
||||||
|
}
|
||||||
|
|
||||||
|
private static async Task<NoContent> DeleteProductAsync(int id, IProductDataService data)
|
||||||
|
{
|
||||||
|
await data.DeleteProduct(id);
|
||||||
|
|
||||||
|
return TypedResults.NoContent();
|
||||||
|
}
|
||||||
|
}
|
||||||
27
ITS.Cled.Ripasso/ITS.Cled.Ripasso.csproj
Normal file
27
ITS.Cled.Ripasso/ITS.Cled.Ripasso.csproj
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
<Project Sdk="Microsoft.NET.Sdk.Web">
|
||||||
|
|
||||||
|
<PropertyGroup>
|
||||||
|
<TargetFramework>net8.0</TargetFramework>
|
||||||
|
<Nullable>enable</Nullable>
|
||||||
|
<ImplicitUsings>enable</ImplicitUsings>
|
||||||
|
<DockerDefaultTargetOS>Linux</DockerDefaultTargetOS>
|
||||||
|
<UserSecretsId>bb183bb1-4918-4fb7-869c-58c8be18f6a6</UserSecretsId>
|
||||||
|
</PropertyGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<Content Include="..\.dockerignore">
|
||||||
|
<Link>.dockerignore</Link>
|
||||||
|
</Content>
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<PackageReference Include="Azure.Messaging.ServiceBus" Version="7.18.2" />
|
||||||
|
<PackageReference Include="Dapper" Version="2.1.35" />
|
||||||
|
<PackageReference Include="Dapper.Contrib" Version="2.0.78" />
|
||||||
|
<PackageReference Include="Microsoft.AspNetCore.OpenApi" Version="8.0.10" />
|
||||||
|
<PackageReference Include="Microsoft.Data.SqlClient" Version="5.2.2" />
|
||||||
|
<PackageReference Include="Npgsql" Version="8.0.5" />
|
||||||
|
<PackageReference Include="Swashbuckle.AspNetCore" Version="6.9.0" />
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
|
</Project>
|
||||||
66
ITS.Cled.Ripasso/Messages/MessageSender.cs
Normal file
66
ITS.Cled.Ripasso/Messages/MessageSender.cs
Normal file
@@ -0,0 +1,66 @@
|
|||||||
|
using System.Text.Json;
|
||||||
|
|
||||||
|
namespace ITS.Cled.Ripasso.Messages;
|
||||||
|
|
||||||
|
using Azure.Messaging.ServiceBus;
|
||||||
|
|
||||||
|
public class MessageSender
|
||||||
|
{
|
||||||
|
private readonly string _connectionstring =
|
||||||
|
"";
|
||||||
|
|
||||||
|
private readonly string _queue = "test";
|
||||||
|
public async Task Send(Object objectToSend)
|
||||||
|
{
|
||||||
|
// the client that owns the connection and can be used to create senders and receivers
|
||||||
|
ServiceBusClient client;
|
||||||
|
|
||||||
|
// the sender used to publish messages to the queue
|
||||||
|
ServiceBusSender sender;
|
||||||
|
|
||||||
|
// number of messages to be sent to the queue
|
||||||
|
const int numOfMessages = 1;
|
||||||
|
|
||||||
|
// The Service Bus client types are safe to cache and use as a singleton for the lifetime
|
||||||
|
// of the application, which is best practice when messages are being published or read
|
||||||
|
// regularly.
|
||||||
|
//
|
||||||
|
// set the transport type to AmqpWebSockets so that the ServiceBusClient uses the port 443.
|
||||||
|
// If you use the default AmqpTcp, you will need to make sure that the ports 5671 and 5672 are open
|
||||||
|
|
||||||
|
// TODO: Replace the <NAMESPACE-CONNECTION-STRING> and <QUEUE-NAME> placeholders
|
||||||
|
var clientOptions = new ServiceBusClientOptions()
|
||||||
|
{
|
||||||
|
TransportType = ServiceBusTransportType.AmqpWebSockets
|
||||||
|
};
|
||||||
|
client = new ServiceBusClient( _connectionstring, clientOptions);
|
||||||
|
sender = client.CreateSender(_queue);
|
||||||
|
|
||||||
|
// create a batch
|
||||||
|
using ServiceBusMessageBatch messageBatch = await sender.CreateMessageBatchAsync();
|
||||||
|
|
||||||
|
for (int i = 1; i <= numOfMessages; i++)
|
||||||
|
{
|
||||||
|
// try adding a message to the batch
|
||||||
|
if (!messageBatch.TryAddMessage(new ServiceBusMessage(JsonSerializer.Serialize(objectToSend))))
|
||||||
|
{
|
||||||
|
// if it is too large for the batch
|
||||||
|
throw new Exception($"The message {i} is too large to fit in the batch.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
// Use the producer client to send the batch of messages to the Service Bus queue
|
||||||
|
await sender.SendMessagesAsync(messageBatch);
|
||||||
|
Console.WriteLine($"A batch of {numOfMessages} messages has been published to the queue.");
|
||||||
|
}
|
||||||
|
finally
|
||||||
|
{
|
||||||
|
// Calling DisposeAsync on client types is required to ensure that network
|
||||||
|
// resources and other unmanaged objects are properly cleaned up.
|
||||||
|
await sender.DisposeAsync();
|
||||||
|
await client.DisposeAsync();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
26
ITS.Cled.Ripasso/Model/Product.cs
Normal file
26
ITS.Cled.Ripasso/Model/Product.cs
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
using System.ComponentModel.DataAnnotations;
|
||||||
|
|
||||||
|
namespace ITS.Cled.Ripasso.Model;
|
||||||
|
|
||||||
|
using Dapper.Contrib.Extensions;
|
||||||
|
|
||||||
|
[Table("Products")]
|
||||||
|
public class Product
|
||||||
|
{
|
||||||
|
[Key]
|
||||||
|
public int Id { get; set; }
|
||||||
|
|
||||||
|
[Required]
|
||||||
|
[StringLength(50)]
|
||||||
|
public string Name { get; set; } = default!;
|
||||||
|
|
||||||
|
[Required]
|
||||||
|
[StringLength(10)]
|
||||||
|
public string Code { get; set; } = default!;
|
||||||
|
|
||||||
|
[Required]
|
||||||
|
[Range(0, double.MaxValue)]
|
||||||
|
public decimal Price { get; set; }
|
||||||
|
|
||||||
|
public string Action { get; set; } = default!;
|
||||||
|
}
|
||||||
39
ITS.Cled.Ripasso/Program.cs
Normal file
39
ITS.Cled.Ripasso/Program.cs
Normal file
@@ -0,0 +1,39 @@
|
|||||||
|
using ITS.Cled.Ripasso.Components;
|
||||||
|
using ITS.Cled.Ripasso.Endpoints;
|
||||||
|
using ITS.Cled.Ripasso.Messages;
|
||||||
|
using ITS.Cled.Ripasso.Services;
|
||||||
|
|
||||||
|
var builder = WebApplication.CreateBuilder(args);
|
||||||
|
|
||||||
|
builder.Services.AddEndpointsApiExplorer();
|
||||||
|
builder.Services.AddSwaggerGen();
|
||||||
|
|
||||||
|
// Add services to the container.
|
||||||
|
builder.Services.AddRazorComponents().AddInteractiveServerComponents();
|
||||||
|
|
||||||
|
builder.Services.AddScoped<IProductDataService, ProductsDataService>();
|
||||||
|
builder.Services.AddScoped<MessageSender>();
|
||||||
|
|
||||||
|
var app = builder.Build();
|
||||||
|
|
||||||
|
// Configure the HTTP request pipeline.
|
||||||
|
if (!app.Environment.IsDevelopment())
|
||||||
|
{
|
||||||
|
app.UseExceptionHandler("/Error", createScopeForErrors: true);
|
||||||
|
// The default HSTS value is 30 days. You may want to change this for production scenarios, see https://aka.ms/aspnetcore-hsts.
|
||||||
|
app.UseHsts();
|
||||||
|
}
|
||||||
|
|
||||||
|
app.UseSwagger();
|
||||||
|
app.UseSwaggerUI();
|
||||||
|
|
||||||
|
app.UseHttpsRedirection();
|
||||||
|
|
||||||
|
app.UseStaticFiles();
|
||||||
|
app.UseAntiforgery();
|
||||||
|
|
||||||
|
app.MapProductsEndpoint();
|
||||||
|
|
||||||
|
app.MapRazorComponents<App>().AddInteractiveServerRenderMode();
|
||||||
|
|
||||||
|
app.Run();
|
||||||
38
ITS.Cled.Ripasso/Properties/launchSettings.json
Normal file
38
ITS.Cled.Ripasso/Properties/launchSettings.json
Normal file
@@ -0,0 +1,38 @@
|
|||||||
|
{
|
||||||
|
"$schema": "http://json.schemastore.org/launchsettings.json",
|
||||||
|
"iisSettings": {
|
||||||
|
"windowsAuthentication": false,
|
||||||
|
"anonymousAuthentication": true,
|
||||||
|
"iisExpress": {
|
||||||
|
"applicationUrl": "http://localhost:45424",
|
||||||
|
"sslPort": 44302
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"profiles": {
|
||||||
|
"http": {
|
||||||
|
"commandName": "Project",
|
||||||
|
"dotnetRunMessages": true,
|
||||||
|
"launchBrowser": true,
|
||||||
|
"applicationUrl": "http://localhost:5017",
|
||||||
|
"environmentVariables": {
|
||||||
|
"ASPNETCORE_ENVIRONMENT": "Development"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"https": {
|
||||||
|
"commandName": "Project",
|
||||||
|
"dotnetRunMessages": true,
|
||||||
|
"launchBrowser": true,
|
||||||
|
"applicationUrl": "https://localhost:7181;http://localhost:5017",
|
||||||
|
"environmentVariables": {
|
||||||
|
"ASPNETCORE_ENVIRONMENT": "Development"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"IIS Express": {
|
||||||
|
"commandName": "IISExpress",
|
||||||
|
"launchBrowser": true,
|
||||||
|
"environmentVariables": {
|
||||||
|
"ASPNETCORE_ENVIRONMENT": "Development"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
12
ITS.Cled.Ripasso/Services/IProductDataService.cs
Normal file
12
ITS.Cled.Ripasso/Services/IProductDataService.cs
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
using ITS.Cled.Ripasso.Model;
|
||||||
|
|
||||||
|
namespace ITS.Cled.Ripasso.Services;
|
||||||
|
|
||||||
|
public interface IProductDataService
|
||||||
|
{
|
||||||
|
Task<IEnumerable<Product>> GetProductsAsync();
|
||||||
|
Task<Product?> GetProductById(int id);
|
||||||
|
Task<Product> CreateProduct(Product product);
|
||||||
|
Task UpdateProduct(Product product);
|
||||||
|
Task DeleteProduct(int id);
|
||||||
|
}
|
||||||
83
ITS.Cled.Ripasso/Services/ProductsDataService.cs
Normal file
83
ITS.Cled.Ripasso/Services/ProductsDataService.cs
Normal file
@@ -0,0 +1,83 @@
|
|||||||
|
using Microsoft.Data.SqlClient;
|
||||||
|
|
||||||
|
namespace ITS.Cled.Ripasso.Services;
|
||||||
|
|
||||||
|
using Dapper;
|
||||||
|
using Dapper.Contrib.Extensions;
|
||||||
|
using ITS.Cled.Ripasso.Model;
|
||||||
|
using Npgsql;
|
||||||
|
|
||||||
|
public class ProductsDataService : IProductDataService
|
||||||
|
{
|
||||||
|
private readonly string _connectionString;
|
||||||
|
|
||||||
|
public ProductsDataService(IConfiguration configuration)
|
||||||
|
{
|
||||||
|
_connectionString =
|
||||||
|
configuration.GetConnectionString("db")
|
||||||
|
?? throw new Exception("Missing connectionString 'db'.");
|
||||||
|
}
|
||||||
|
|
||||||
|
public async Task<IEnumerable<Product>> GetProductsAsync()
|
||||||
|
{
|
||||||
|
await using var connection = new SqlConnection(_connectionString);
|
||||||
|
const string query = """
|
||||||
|
SELECT
|
||||||
|
id,
|
||||||
|
name,
|
||||||
|
code,
|
||||||
|
price
|
||||||
|
FROM products;
|
||||||
|
""";
|
||||||
|
return await connection.QueryAsync<Product>(query);
|
||||||
|
}
|
||||||
|
|
||||||
|
public async Task<Product?> GetProductById(int id)
|
||||||
|
{
|
||||||
|
await using var connection = new SqlConnection(_connectionString);
|
||||||
|
const string query = """
|
||||||
|
SELECT
|
||||||
|
id,
|
||||||
|
name,
|
||||||
|
code,
|
||||||
|
price
|
||||||
|
FROM products
|
||||||
|
WHERE id = @id;
|
||||||
|
""";
|
||||||
|
return await connection.QueryFirstOrDefaultAsync<Product>(query, new { id });
|
||||||
|
}
|
||||||
|
|
||||||
|
public async Task<Product> CreateProduct(Product product)
|
||||||
|
{
|
||||||
|
await using var connection = new SqlConnection(_connectionString);
|
||||||
|
const string query = """
|
||||||
|
INSERT INTO products (name, code, price)
|
||||||
|
VALUES (@Name, @Code, @Price)
|
||||||
|
""";
|
||||||
|
await connection.ExecuteAsync(query, product);
|
||||||
|
|
||||||
|
return product;
|
||||||
|
}
|
||||||
|
|
||||||
|
public async Task UpdateProduct(Product product)
|
||||||
|
{
|
||||||
|
await using var connection = new SqlConnection(_connectionString);
|
||||||
|
const string query = """
|
||||||
|
UPDATE products
|
||||||
|
SET
|
||||||
|
name = @Name,
|
||||||
|
code = @Code,
|
||||||
|
price = @Price
|
||||||
|
WHERE id = @Id
|
||||||
|
""";
|
||||||
|
await connection.ExecuteAsync(query, product);
|
||||||
|
}
|
||||||
|
|
||||||
|
public async Task DeleteProduct(int id)
|
||||||
|
{
|
||||||
|
await using var connection = new SqlConnection(_connectionString);
|
||||||
|
|
||||||
|
const string query = "DELETE FROM products WHERE id = @id;";
|
||||||
|
await connection.ExecuteAsync(query, new { id });
|
||||||
|
}
|
||||||
|
}
|
||||||
8
ITS.Cled.Ripasso/appsettings.Development.json
Normal file
8
ITS.Cled.Ripasso/appsettings.Development.json
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
{
|
||||||
|
"Logging": {
|
||||||
|
"LogLevel": {
|
||||||
|
"Default": "Information",
|
||||||
|
"Microsoft.AspNetCore": "Warning"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
9
ITS.Cled.Ripasso/appsettings.json
Normal file
9
ITS.Cled.Ripasso/appsettings.json
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
{
|
||||||
|
"Logging": {
|
||||||
|
"LogLevel": {
|
||||||
|
"Default": "Information",
|
||||||
|
"Microsoft.AspNetCore": "Warning"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"AllowedHosts": "*"
|
||||||
|
}
|
||||||
51
ITS.Cled.Ripasso/wwwroot/app.css
Normal file
51
ITS.Cled.Ripasso/wwwroot/app.css
Normal file
@@ -0,0 +1,51 @@
|
|||||||
|
html, body {
|
||||||
|
font-family: 'Helvetica Neue', Helvetica, Arial, sans-serif;
|
||||||
|
}
|
||||||
|
|
||||||
|
a, .btn-link {
|
||||||
|
color: #006bb7;
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-primary {
|
||||||
|
color: #fff;
|
||||||
|
background-color: #1b6ec2;
|
||||||
|
border-color: #1861ac;
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn:focus, .btn:active:focus, .btn-link.nav-link:focus, .form-control:focus, .form-check-input:focus {
|
||||||
|
box-shadow: 0 0 0 0.1rem white, 0 0 0 0.25rem #258cfb;
|
||||||
|
}
|
||||||
|
|
||||||
|
.content {
|
||||||
|
padding-top: 1.1rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
h1:focus {
|
||||||
|
outline: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.valid.modified:not([type=checkbox]) {
|
||||||
|
outline: 1px solid #26b050;
|
||||||
|
}
|
||||||
|
|
||||||
|
.invalid {
|
||||||
|
outline: 1px solid #e50000;
|
||||||
|
}
|
||||||
|
|
||||||
|
.validation-message {
|
||||||
|
color: #e50000;
|
||||||
|
}
|
||||||
|
|
||||||
|
.blazor-error-boundary {
|
||||||
|
background: url() no-repeat 1rem/1.8rem, #b32121;
|
||||||
|
padding: 1rem 1rem 1rem 3.7rem;
|
||||||
|
color: white;
|
||||||
|
}
|
||||||
|
|
||||||
|
.blazor-error-boundary::after {
|
||||||
|
content: "An error has occurred."
|
||||||
|
}
|
||||||
|
|
||||||
|
.darker-border-checkbox.form-check-input {
|
||||||
|
border-color: #929292;
|
||||||
|
}
|
||||||
7
ITS.Cled.Ripasso/wwwroot/bootstrap/bootstrap.min.css
vendored
Normal file
7
ITS.Cled.Ripasso/wwwroot/bootstrap/bootstrap.min.css
vendored
Normal file
File diff suppressed because one or more lines are too long
1
ITS.Cled.Ripasso/wwwroot/bootstrap/bootstrap.min.css.map
Normal file
1
ITS.Cled.Ripasso/wwwroot/bootstrap/bootstrap.min.css.map
Normal file
File diff suppressed because one or more lines are too long
BIN
ITS.Cled.Ripasso/wwwroot/favicon.png
Normal file
BIN
ITS.Cled.Ripasso/wwwroot/favicon.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 1.1 KiB |
14
ITS.ServiceBus/ITS.ServiceBus.csproj
Normal file
14
ITS.ServiceBus/ITS.ServiceBus.csproj
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
<Project Sdk="Microsoft.NET.Sdk">
|
||||||
|
|
||||||
|
<PropertyGroup>
|
||||||
|
<OutputType>Exe</OutputType>
|
||||||
|
<TargetFramework>net8.0</TargetFramework>
|
||||||
|
<ImplicitUsings>enable</ImplicitUsings>
|
||||||
|
<Nullable>enable</Nullable>
|
||||||
|
</PropertyGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<PackageReference Include="Azure.Messaging.ServiceBus" Version="7.18.2" />
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
|
</Project>
|
||||||
15
ITS.ServiceBus/Program.cs
Normal file
15
ITS.ServiceBus/Program.cs
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
namespace ITS.ServiceBus;
|
||||||
|
|
||||||
|
class Program
|
||||||
|
{
|
||||||
|
static async Task Main(string[] args)
|
||||||
|
{
|
||||||
|
MessageSender sender = new MessageSender();
|
||||||
|
|
||||||
|
MessageReceiver receiver = new MessageReceiver();
|
||||||
|
|
||||||
|
await sender.Send();
|
||||||
|
|
||||||
|
await receiver.Receive();
|
||||||
|
}
|
||||||
|
}
|
||||||
108
Receiver/MessageReceiver.cs
Normal file
108
Receiver/MessageReceiver.cs
Normal file
@@ -0,0 +1,108 @@
|
|||||||
|
using System.Text.Json;
|
||||||
|
using QuestPDF.Fluent;
|
||||||
|
|
||||||
|
namespace Receiver;
|
||||||
|
|
||||||
|
using Azure.Messaging.ServiceBus;
|
||||||
|
using System;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
|
||||||
|
public class MessageReceiver
|
||||||
|
{
|
||||||
|
private readonly string _connectionstring =
|
||||||
|
"";
|
||||||
|
|
||||||
|
private readonly string _queue = "test";
|
||||||
|
|
||||||
|
public async Task Receive()
|
||||||
|
{
|
||||||
|
// the client that owns the connection and can be used to create senders and receivers
|
||||||
|
ServiceBusClient client;
|
||||||
|
|
||||||
|
// the processor that reads and processes messages from the queue
|
||||||
|
ServiceBusProcessor processor;
|
||||||
|
|
||||||
|
// The Service Bus client types are safe to cache and use as a singleton for the lifetime
|
||||||
|
// of the application, which is best practice when messages are being published or read
|
||||||
|
// regularly.
|
||||||
|
//
|
||||||
|
// Set the transport type to AmqpWebSockets so that the ServiceBusClient uses port 443.
|
||||||
|
// If you use the default AmqpTcp, make sure that ports 5671 and 5672 are open.
|
||||||
|
|
||||||
|
// TODO: Replace the <NAMESPACE-CONNECTION-STRING> and <QUEUE-NAME> placeholders
|
||||||
|
var clientOptions = new ServiceBusClientOptions()
|
||||||
|
{
|
||||||
|
TransportType = ServiceBusTransportType.AmqpWebSockets
|
||||||
|
};
|
||||||
|
client = new ServiceBusClient(_connectionstring, clientOptions);
|
||||||
|
|
||||||
|
// create a processor that we can use to process the messages
|
||||||
|
// TODO: Replace the <QUEUE-NAME> placeholder
|
||||||
|
processor = client.CreateProcessor(_queue, new ServiceBusProcessorOptions());
|
||||||
|
|
||||||
|
try
|
||||||
|
{
|
||||||
|
// add handler to process messages
|
||||||
|
processor.ProcessMessageAsync += MessageHandler;
|
||||||
|
|
||||||
|
// add handler to process any errors
|
||||||
|
processor.ProcessErrorAsync += ErrorHandler;
|
||||||
|
|
||||||
|
// start processing
|
||||||
|
await processor.StartProcessingAsync();
|
||||||
|
|
||||||
|
Console.WriteLine("Wait for a minute and then press any key to end the processing");
|
||||||
|
Console.ReadKey();
|
||||||
|
|
||||||
|
// stop processing
|
||||||
|
Console.WriteLine("\nStopping the receiver...");
|
||||||
|
await processor.StopProcessingAsync();
|
||||||
|
Console.WriteLine("Stopped receiving messages");
|
||||||
|
}
|
||||||
|
finally
|
||||||
|
{
|
||||||
|
// Calling DisposeAsync on client types is required to ensure that network
|
||||||
|
// resources and other unmanaged objects are properly cleaned up.
|
||||||
|
await processor.DisposeAsync();
|
||||||
|
await client.DisposeAsync();
|
||||||
|
}
|
||||||
|
|
||||||
|
// handle received messages
|
||||||
|
async Task MessageHandler(ProcessMessageEventArgs args)
|
||||||
|
{
|
||||||
|
string body = args.Message.Body.ToString();
|
||||||
|
|
||||||
|
var bodyDeserialized = JsonSerializer.Deserialize<Product>(body);
|
||||||
|
Console.WriteLine($"Received: {bodyDeserialized.Name}");
|
||||||
|
if (bodyDeserialized.Action == "Create")
|
||||||
|
{
|
||||||
|
Document.Create(container =>
|
||||||
|
{
|
||||||
|
container.Page(page =>
|
||||||
|
{
|
||||||
|
page.Header()
|
||||||
|
.Text("Prodotto")
|
||||||
|
.AlignCenter();
|
||||||
|
page.Content()
|
||||||
|
.Text(
|
||||||
|
$"Product: {bodyDeserialized.Name} Code: {bodyDeserialized.Code} Price: {bodyDeserialized.Price}");
|
||||||
|
});
|
||||||
|
}).GeneratePdf($"{bodyDeserialized.Name}{bodyDeserialized.Code}{bodyDeserialized.Price}.pdf");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (bodyDeserialized.Action == "Delete")
|
||||||
|
{
|
||||||
|
System.IO.File.Delete($"{bodyDeserialized.Name}{bodyDeserialized.Code}{bodyDeserialized.Price}.pdf");
|
||||||
|
}
|
||||||
|
// complete the message. message is deleted from the queue.
|
||||||
|
await args.CompleteMessageAsync(args.Message);
|
||||||
|
}
|
||||||
|
|
||||||
|
// handle any errors when receiving messages
|
||||||
|
Task ErrorHandler(ProcessErrorEventArgs args)
|
||||||
|
{
|
||||||
|
Console.WriteLine(args.Exception.ToString());
|
||||||
|
return Task.CompletedTask;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
23
Receiver/Product.cs
Normal file
23
Receiver/Product.cs
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
using System.ComponentModel.DataAnnotations;
|
||||||
|
|
||||||
|
namespace Receiver;
|
||||||
|
|
||||||
|
public class Product
|
||||||
|
{
|
||||||
|
[Key]
|
||||||
|
public int Id { get; set; }
|
||||||
|
|
||||||
|
[Required]
|
||||||
|
[StringLength(50)]
|
||||||
|
public string Name { get; set; } = default!;
|
||||||
|
|
||||||
|
[Required]
|
||||||
|
[StringLength(10)]
|
||||||
|
public string Code { get; set; } = default!;
|
||||||
|
|
||||||
|
[Required]
|
||||||
|
[Range(0, double.MaxValue)]
|
||||||
|
public decimal Price { get; set; }
|
||||||
|
|
||||||
|
public string Action { get; set; } = default!;
|
||||||
|
}
|
||||||
15
Receiver/Program.cs
Normal file
15
Receiver/Program.cs
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
using QuestPDF.Infrastructure;
|
||||||
|
|
||||||
|
namespace Receiver;
|
||||||
|
|
||||||
|
class Program
|
||||||
|
{
|
||||||
|
static async Task Main(string[] args)
|
||||||
|
{
|
||||||
|
QuestPDF.Settings.License = LicenseType.Community;
|
||||||
|
|
||||||
|
MessageReceiver receiver = new MessageReceiver();
|
||||||
|
|
||||||
|
await receiver.Receive();
|
||||||
|
}
|
||||||
|
}
|
||||||
15
Receiver/Receiver.csproj
Normal file
15
Receiver/Receiver.csproj
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
<Project Sdk="Microsoft.NET.Sdk">
|
||||||
|
|
||||||
|
<PropertyGroup>
|
||||||
|
<OutputType>Exe</OutputType>
|
||||||
|
<TargetFramework>net8.0</TargetFramework>
|
||||||
|
<ImplicitUsings>enable</ImplicitUsings>
|
||||||
|
<Nullable>enable</Nullable>
|
||||||
|
</PropertyGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<PackageReference Include="Azure.Messaging.ServiceBus" Version="7.18.2" />
|
||||||
|
<PackageReference Include="QuestPDF" Version="2024.10.3" />
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
|
</Project>
|
||||||
Reference in New Issue
Block a user