First commit

This commit is contained in:
2024-11-21 16:53:30 +01:00
commit 5cbd7b9d6e
46 changed files with 1506 additions and 0 deletions

25
.dockerignore Normal file
View 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
View 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
View 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

View 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>

View 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>

View File

@@ -0,0 +1,8 @@
<?xml version="1.0" encoding="UTF-8"?>
<project version="4">
<component name="UserContentModel">
<attachedFolders />
<explicitIncludes />
<explicitExcludes />
</component>
</project>

View 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>

View 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>

View 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>

View 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
View 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

View 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>

View 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>

View 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;
}

View 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>

View 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;
}
}

View 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++;
}
}

View 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;
}

View File

@@ -0,0 +1,7 @@
@page "/"
<PageTitle>Home</PageTitle>
<h1>Hello, world!</h1>
Welcome to your new app.

View 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");
}
}

View 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);
}
}
}

View 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");
}
}

View 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);
}
}

View 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>

View 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

View 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"]

View 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();
}
}

View 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>

View 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();
}
}
}

View 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!;
}

View 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();

View 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"
}
}
}
}

View 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);
}

View 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 });
}
}

View File

@@ -0,0 +1,8 @@
{
"Logging": {
"LogLevel": {
"Default": "Information",
"Microsoft.AspNetCore": "Warning"
}
}
}

View File

@@ -0,0 +1,9 @@
{
"Logging": {
"LogLevel": {
"Default": "Information",
"Microsoft.AspNetCore": "Warning"
}
},
"AllowedHosts": "*"
}

View 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;
}

File diff suppressed because one or more lines are too long

File diff suppressed because one or more lines are too long

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.1 KiB

View 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
View 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
View 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
View 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
View 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
View 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>