Velkommen til .NET-skolen!
Dette er et kurs hvor du blir tatt gjennom prosessen av å sette opp, og implementere, en .NET-løsning fra bunnen av, steg for steg. Målet med kurset er å vise hvordan man kan utføre oppgaver som er vanlige i etableringsfasen av et system, som å:
- Opprette prosjekter og mappestruktur
- Sette opp pakkehåndtering
- Sette opp tester
Som en eksempel-applikasjon skal vi lage et web-API i F# for å hente ut elektronisk programguide (EPG) for NRK TV, med tilhørende enhets- og integrasjonstester. Tanken er at API-et kunne levert datagrunnlaget til en programguide - f.eks. den som vises her: https://info.nrk.no/presse/tvguide/
Et sekundært mål med dette repoet er at den ferdige eksempel-applikasjonen (som du finner i branchen ferdig) kan fungere som et referanse-repo for hvordan å sette opp et .NET-prosjekt.
Vi skal bruke .NET CLI til å opprette prosjekter, samt kjøre koden og testene. I tillegg skal vi dokumentere web-API-et vårt ved hjelp av OpenAPI.
Overordnet kommer mappestrukturen til løsningen vår til å se slik ut:
└── docs (kontrakt for web-API-et)
└── src (kildekode til web-API-et)
└── test (kildekode til enhets- og integrasjonstestene)
Det anbefales å følge denne veiledningen på GitHub, da visningen der støtter lenkene som er lagt inn, og har innholdsfortegnelse som alltid er synlig oppe til venstre når man blar i veiledningen.
For å gjennomføre dette kurset trenger du .NET 6 SDK, en teksteditor og en terminal. Når du har dette, gå til Steg 1 - Opprette API og følg veiledningen. For alternative startpunkter se alternative startpunkter.
Stegene i kurset gir veiledning, steg for steg, med anvisninger for kommandoer du kan kjøre og referanseimplementasjon av kode du kan kopiere. Enkelte steder er implementasjonen av koden imidlertid utelatt slik at du kan forsøke å implementere den selv. Disse stedene er markert med ☑️. Les mer om hvordan du kan se fullstendig løsningsforslag for hvert steg her.
Hvis du trenger mer detaljer om hvordan du gjør klar maskinen din til å gjennomføre kurset, se Detaljer om oppsett på maskinen din.
Dersom du er helt ny til .NET kan det være nyttig å begynne med å lese:
Denne workshopen dekker en del ulike temaer, og det kan ta litt tid å fullføre alle stegene. Heldigvis finnes det løsningsforslag for hvert steg i workshopen, som betyr at du kan starte på et hvilket som helst steg ved å sjekke ut branchen med løsningsforslaget til steget før du ønsker å begynne på, og fortsette derfra. Les mer om hvordan du kan klone dette repoet og sjekke ut løsningsforslag.
Under følger noen anbefalinger for alternative startpunkter, avhengig av hvilke temaer du ønsker å lære mer om.
NB! Dersom du begynner på steg 5 eller senere, må du kjøre
dotnet tool restore
før du fortsetter å følge veiledningen.
Dersom du er interessert i å lære mer om hvordan du kan bruke .NET CLI til å opprette prosjekter, solutions, og sette opp håndtering av NuGet-pakker med paket
, kan følge disse stegene:
- Steg 1 - Opprette API
- Steg 2 - Opprette testprosjekter
- Steg 3 - Opprette solution
- Steg 4 - Pakkehåndtering
Vil du lære mer om domenemodellering i F# og tilhørende enhetstester, kan følge disse stegene:
Hvis du vil lære mer om hvordan du kan dokumentere API-et ditt vha. Open API, og modellere kontraktstyper, kan følge disse stegene:
Om du er interessert i .NET 6 sin hosting modell, "minimal APIs", og hvordan du kan teste API-et ditt med integrasjonstester, kan følge disse stegene:
Til slutt finnes det noen ekstraoppgaver, hvis du vil ha mer å bryne deg på:
Lurer du på noe knyttet til kurset? Opprett gjerne en tråd under "Discussions" i dette repoet:
Nyttige tips og triks finner du her
- Microsoft's offisielle dokumentasjon for .NET - https://docs.microsoft.com/en-us/dotnet/
- F# cheat sheet - http://dungpa.github.io/fsharp-cheatsheet/
- Innføring i F# - https://fsharpforfunandprofit.com/site-contents/#understanding-f
- Andre kurs i NRK
- F#-skolen, kurs i F# laget av ansatte i NRK TV - https://github.com/nrkno/fsharpskolen
- GitHub Actions 101, laget av @teodor-elstad https://github.com/teodor-elstad/github-actions-101
Har du tilbakemeldinger til kurset? Opprett gjerne en tråd for det her:
- @heidisu - Idé og kvalitetssikring
- @thomaswolff - Primus motor og forfatter
- Takk til alle som har kommet med konstruktiv kritikk og nyttige tilbakemeldinger til dette kurset.
All dokumentasjon (inkludert denne veiledningen) og kildekoden i dette repoet er åpent tilgjengelig under MIT-lisensen.
- Steg
- Steg 1 - Opprette API
- Steg 2 - Opprette testprosjekter
- Steg 3 - Opprette solution
- Steg 4 - Pakkehåndtering
- Steg 5 - Definere domenemodell
- Steg 6 - Enhetstester for domenemodell
- Steg 7 - Definere API-kontrakt
- Steg 8 - Implementere kontraktstyper
- Steg 9 - Sette opp skall for API
- Steg 10 - Implementere web-API
- Ekstraoppgaver
Nå som du har installert alle verktøyene du trenger er du klar til å begynne på selve kurset!
Steg 1 av 10 - 🔝 Gå til toppen ⬇ Neste steg
I dette steget starter vi med en mappe helt uten kode, og bruker .NET CLI til å opprette vårt første prosjekt NRK.Dotnetskolen.Api
.
Siden denne veiledningen er skrevet for .NET 6, og det er mulig at du har flere .NET-versjoner installert på maskinen din, må vi instruere .NET CLI til å benytte .NET 6 når vi kjører kommandoene i veiledningen. Dette gjør vi ved å opprette en konfigurasjonsfil global.json
i roten av repoet med følgende innhold:
{
"sdk": {
"version": "6.0.0",
"rollForward": "latestMinor"
}
}
Her oppgir vi at vi i utgangspunktet ønsker å bruke version 6.0.0
av .NET SDK. I tillegg sier vi gjennom rollForward: latestMinor
at vi ønsker at den høyeste tilgjengelige versjonen av .NET 6 på maskinen din skal brukes.
Du kan lese mer om
global.json
her: https://docs.microsoft.com/en-us/dotnet/core/tools/global-json
For å kunne organisere kode i .NET bruker man prosjekter. Et prosjekt er en samling med kildekodefiler, og eventuelle andre ressursfiler, og alle filene som inngår i prosjektet er referert til i en prosjektfil. For F#-prosjekter har slike prosjektfiler filendelsen .fsproj
.
Når man kompilerer .NET-prosjekter kan man velge mellom to typer output:
- Kjørbar fil (executable) - et program som kan kjøres
- Klassebibliotek (dynamically linked library) - en samling med funksjonalitet som kan benyttes av andre programmer
Som nevnt i innledningen er .NET CLI et kommandolinjeverktøy laget for å utvikle, bygge, kjøre og publisere .NET-applikasjoner. .NET CLI kjøres fra kommandolinjen med kommandoen dotnet
, og har mange underkommandoer og valg. For å se alle kan du kjøre kommandoen under, eller lese mer her: https://docs.microsoft.com/en-us/dotnet/core/tools/dotnet
dotnet --help
.NET SDK (6.0.101)
Usage: dotnet [runtime-options] [path-to-application] [arguments]
Execute a .NET application.
runtime-options:
--additionalprobingpath <path> Path containing probing policy and assemblies to probe for.
--additional-deps <path> Path to additional deps.json file.
--depsfile Path to <application>.deps.json file.
--fx-version <version> Version of the installed Shared Framework to use to run the application.
--roll-forward <setting> Roll forward to framework version (LatestPatch, Minor, LatestMinor, Major, LatestMajor, Disable).
--runtimeconfig Path to <application>.runtimeconfig.json file.
path-to-application:
The path to an application .dll file to execute.
Usage: dotnet [sdk-options] [command] [command-options] [arguments]
Execute a .NET SDK command.
sdk-options:
-d|--diagnostics Enable diagnostic output.
-h|--help Show command line help.
--info Display .NET information.
--list-runtimes Display the installed runtimes.
--list-sdks Display the installed SDKs.
--version Display .NET SDK version in use.
SDK commands:
add Add a package or reference to a .NET project.
build Build a .NET project.
build-server Interact with servers started by a build.
clean Clean build outputs of a .NET project.
format Apply style preferences to a project or solution.
help Show command line help.
list List project references of a .NET project.
msbuild Run Microsoft Build Engine (MSBuild) commands.
new Create a new .NET project or file.
nuget Provides additional NuGet commands.
pack Create a NuGet package.
publish Publish a .NET project for deployment.
remove Remove a package or reference from a .NET project.
restore Restore dependencies specified in a .NET project.
run Build and run a .NET project output.
sdk Manage .NET SDK installation.
sln Modify Visual Studio solution files.
store Store the specified assemblies in the runtime package store.
test Run unit tests using the test runner specified in a .NET project.
tool Install or manage tools that extend the .NET experience.
vstest Run Microsoft Test Engine (VSTest) commands.
workload Manage optional workloads.
Additional commands from bundled tools:
dev-certs Create and manage development certificates.
fsi Start F# Interactive / execute F# scripts.
sql-cache SQL Server cache command-line tools.
user-secrets Manage development user secrets.
watch Start a file watcher that runs a command when files change.
Run 'dotnet [command] --help' for more information on a command.
For å opprette API-prosjektet skal vi bruke new
-kommandoen i .NET CLI. dotnet new
oppretter .NET-prosjekter, og som første parameter tar new
-kommandoen inn hva slags mal prosjektet man oppretter skal følge. Når man installerer .NET SDK får man med et sett med forhåndsdefinerte prosjektmaler for vanlige formål. For å se malene som er installert på din maskin kan du kjøre dotnet new --list
slik:
dotnet new --list
These templates matched your input:
Template Name Short Name Language Tags
-------------------------------------------- ------------------- ---------- -------------------------------------
ASP.NET Core Empty web [C#],F# Web/Empty
ASP.NET Core gRPC Service grpc [C#] Web/gRPC
ASP.NET Core Web API webapi [C#],F# Web/WebAPI
ASP.NET Core Web App razor,webapp [C#] Web/MVC/Razor Pages
ASP.NET Core Web App (Model-View-Controller) mvc [C#],F# Web/MVC
ASP.NET Core with Angular angular [C#] Web/MVC/SPA
ASP.NET Core with React.js react [C#] Web/MVC/SPA
ASP.NET Core with React.js and Redux reactredux [C#] Web/MVC/SPA
Blazor Server App blazorserver [C#] Web/Blazor
Blazor WebAssembly App blazorwasm [C#] Web/Blazor/WebAssembly/PWA
Class Library classlib [C#],F#,VB Common/Library
Console App console [C#],F#,VB Common/Console
dotnet gitignore file gitignore Config
Dotnet local tool manifest file tool-manifest Config
EditorConfig file editorconfig Config
global.json file globaljson Config
MSTest Test Project mstest [C#],F#,VB Test/MSTest
MVC ViewImports viewimports [C#] Web/ASP.NET
MVC ViewStart viewstart [C#] Web/ASP.NET
NuGet Config nugetconfig Config
NUnit 3 Test Item nunit-test [C#],F#,VB Test/NUnit
NUnit 3 Test Project nunit [C#],F#,VB Test/NUnit
Protocol Buffer File proto Web/gRPC
Razor Class Library razorclasslib [C#] Web/Razor/Library/Razor Class Library
Razor Component razorcomponent [C#] Web/ASP.NET
Razor Page page [C#] Web/ASP.NET
Solution File sln Solution
Web Config webconfig Config
Windows Forms App winforms [C#],VB Common/WinForms
Windows Forms Class Library winformslib [C#],VB Common/WinForms
Windows Forms Control Library winformscontrollib [C#],VB Common/WinForms
Worker Service worker [C#],F# Common/Worker/Web
WPF Application wpf [C#],VB Common/WPF
WPF Class library wpflib [C#],VB Common/WPF
WPF Custom Control Library wpfcustomcontrollib [C#],VB Common/WPF
WPF User Control Library wpfusercontrollib [C#],VB Common/WPF
xUnit Test Project xunit [C#],F#,VB Test/xUnit
I tillegg til å styre hva slags type prosjekt man vil opprette med new
-kommandoen, har man mulighet til å styre ting som hvilket språk man ønsker prosjektet skal opprettes for, og i hvilken mappe prosjektet opprettes i. For å se alle valgene man har i dotnet new
kan du kjøre følgende kommando
dotnet new --help
Usage: new [options]
Options:
-h, --help Displays help for this command.
-l, --list Lists templates containing the specified name. If no name is specified, lists all templates.
-n, --name The name for the output being created. If no name is specified, the name of the current directory is used.
-o, --output Location to place the generated output.
-i, --install Installs a source or a template pack.
-u, --uninstall Uninstalls a source or a template pack.
--interactive Allows the internal dotnet restore command to stop and wait for user input or action (for example to complete authentication).
--nuget-source Specifies a NuGet source to use during install.
--type Filters templates based on available types. Predefined values are "project", "item" or "other".
--dry-run Displays a summary of what would happen if the given command line were run if it would result in a template creation.
--force Forces content to be generated even if it would change existing files.
-lang, --language Filters templates based on language and specifies the language of the template to create.
--update-check Check the currently installed template packs for updates.
--update-apply Check the currently installed template packs for update, and install the updates.
Som du ser av malene som er listet ut over, er det en innebygget mal for web-API som heter webapi
. For å komme raskt i gang med et prosjekt, eller se hvordan et default .NET API er satt opp, kan man bruke webapi
som mal. Vi kommer imidlertid til å opprette API-et vårt fra bunnen av ved å bruke malen console
for å lære mest mulig om de ulike bestanddelene.
Kjør følgende kommando for å opprette API-prosjektet
dotnet new console --language F# --output src/api --name NRK.Dotnetskolen.Api
The template "Console Application" was created successfully.
Processing post-creation actions...
Running 'dotnet restore' on src\api\NRK.Dotnetskolen.Api.fsproj...
Determining projects to restore...
Restored C:\Dev\n[email protected]\dotnetskolen\src\api\NRK.Dotnetskolen.Api.fsproj (in 101 ms).
Restore succeeded.
I kommandoen over brukte vi --language
-argumentet for å oppgi at vi ønsket et F#-prosjekt. I tillegg brukte vi --output
for å oppgi hvor vi ønsket at prosjektet skulle ligge relativt til der vi kjører kommandoen fra, og --name
til å styre navnet på prosjektet.
Merk at istedenfor
--language
,--output
og--name
, kunne vi brukt forkortelsene-lang
,-o
og-n
.
Du skal nå ha en mappestruktur som ser slik ut
src
└── api
└── NRK.Dotnetskolen.Api.fsproj
└── Program.fs
Som vi ser av diagrammet over opprettet .NET CLI mappene src
og src/api
, med NRK.Dotnetskolen.Api.fsproj
og Program.fs
i src/api
.
Merk at med mindre noe annet er spesifisert, er alle kommandoene i veiledningen skrevet med forutsetning om at du står i samme mappe når du kjører dem. Dersom du har klonet Git-repoet til kurset er det rotmappen til repoet. Dersom du følger kurset uten å bruke Git er det mappen du bestemmer deg for å kjøre kommandoene i.
Åpne NRK.Dotnetskolen.Api.fsproj
for å se innholdet til prosjektfilen til prosjektet du nettopp opprettet:
<Project Sdk="Microsoft.NET.Sdk">
<PropertyGroup>
<OutputType>Exe</OutputType>
<TargetFramework>net6.0</TargetFramework>
</PropertyGroup>
<ItemGroup>
<Compile Include="Program.fs" />
</ItemGroup>
</Project>
Her ser vi at prosjektet:
- Har outputtypen
exe
- prosjektet kompileres til å bli en kjørbar fil - Skal kompileres til .NET 6
- Består av én fil
Program.fs
For å se hva programmet gjør kan vi åpne Program.fs
og se på koden:
// For more information see https://aka.ms/fsharp-console-apps
printfn "Hello from F#"
Malen la inn kun én linje i Program.fs
som skriver tekststrengen Hello world from F#
til output. Fra andre programmeringsspråk er du kanskje vant til å se en main
-funksjon eller liknende, men det ser vi ikke her. Grunnen til det er at F# bruker et implisitt startpunkt som er på toppen av filen. Deretter utføres koden linje for linje slik som spesifisert i filen. Det er også mulig å bruke eksplisitte startpunkter i F#-programmer. Les mer om det her: https://docs.microsoft.com/en-us/dotnet/fsharp/language-reference/functions/entry-point#implicit-entry-point
Navnet til prosjektet
NRK.Dotnetskolen.Api.fsproj
følger Microsoft sin navnekonvensjon for programmer og biblioteker i .NET. For å lese mer om denne, og andre navnekonvensjoner i .NET, kan du se her: https://docs.microsoft.com/en-us/dotnet/standard/design-guidelines/names-of-assemblies-and-dllsMappestrukturen over er ment som et forslag, og de videre stegene i kurset bygger på denne. Hvis du bruker kurset som inspirasjon eller veiledning til å opprette ditt eget prosjekt, trenger du ikke følge denne mappestrukturen. Hvordan du strukturerer mappene i ditt system er opp til deg, og er avhengig av aspekter som størrelse på systemet, antall prosjekter, og personlige preferanser.
For å kjøre prosjektet som ble opprettet over kan du kjøre følgende kommando
dotnet run --project src/api/NRK.Dotnetskolen.Api.fsproj
Hello world from F#
Alternativt kan du gå til mappen hvor prosjektet ligger, og kjøre dotnet run
derfra, slik som vist under
cd src/api
dotnet run
Hello world from F#
Nå som du har fullført det første steget i kurset er det en fin anledning til å lagre endringene du har gjort så langt i Git.
Gitt at du fulgte veiledningen for å sette opp koden lokalt før du begynte å kode, kan du kjøre følgende kommando for å se hvilke endringer som er gjort i repoet:
git status
On branch <branchnavn>
Untracked files:
(use "git add <file>..." to include in what will be committed)
global.json
src/
nothing added to commit but untracked files present (use "git add" to track)
I outputen over ser vi at Git har oppdaget at det er opprettet en mappe src
og innhold i den, men Git overvåker ikke disse per nå (filene er "untracked").
For å få Git til å overvåke filene vi har opprettet, og deretter se status i Git kan du kjøre følgende kommandoer:
git add .
git status
Changes to be committed:
(use "git restore --staged <file>..." to unstage)
new file: global.json
new file: src/api/NRK.Dotnetskolen.Api.fsproj
new file: src/api/Program.fs
Nå overvåker Git filene.
For å lagre nåværende tilstand av filene i en "commit" i Git kan du kjøre følgende kommando:
git commit -m "Opprettet API-prosjekt"
[<branchnavn> 00d11c8] Opprettet API-prosjekt
2 files changed, 25 insertions(+)
create mode 100644 src/api/NRK.Dotnetskolen.Api.fsproj
create mode 100644 src/api/Program.fs
For å se alle commits i nåværende branch i Git, kan du kjøre følgende kommando:
git log
commit 00d11c82d0179f41883a55ce88e147a73ae60ee2 (HEAD -> <branchnavn>)
Author: Thomas Wolff <[email protected]>
Date: Fri Apr 16 13:43:40 2021 +0200
Opprettet API-prosjekt
...
💡 Tips! Gjenta de tre stegene over med å se endringer, legge dem til, og lagre dem etter å ha fullført hvert steg for å ha bedre oversikt over hva du har vært gjennom i kurset.
Dersom du ønsker å se den forventede tilstanden til repoet etter å ha utført de ulike stegene i kurset, kan du sjekke ut branchen med korresponderende navn som seksjonen du ønsker å se på. F.eks. hvis du vil se hvordan repoet ser ut etter første steg, kan du sjekke ut branchen steg-1
slik:
git checkout steg-1
Switched to branch 'steg-1'
Steg 2 av 10 - 🔝 Gå til toppen ⬆ Forrige steg ⬇ Neste steg
Tester er en viktig del av systemutvikling fordi de hjelper oss med å verifisere at systemet fungerer slik det skal. Når man skriver tester for kode opererer man ofte med to typer tester:
- Enhetstester
- Integrasjonstester
Enhetstester verifiserer at små, isolerte deler av koden fungerer slik den skal. Gjerne én og én funksjon. I dette kurset skal vi bruke enhetstester til å verifisere valideringsregler i domenet vårt.
Integrasjonstester verifiserer at større deler av systemet fungerer slik det skal, og kan til og med dekke samspill med andre systemer. I dette kurset skal vi bruke integrasjonstester til å verifisere at web-API-et oppfører seg i henhold til kontrakten vi definerer i steg 7.
I dette steget skal vi opprette to testprosjekter
- Ett for enhetstester -
NRK.Dotnetskolen.UnitTests
- Ett for integrasjonstester -
NRK.Dotnetskolen.IntegrationTests
For å opprette testprosjektene skal vi igjen bruke dotnet new
-kommandoen, men denne gangen velger vi en annen mal enn da vi opprettet API-prosjektet. Når man installerer .NET SDK følger det med flere maler for testprosjekter som korresponderer til ulike rammeverk som finnes for å detektere og kjøre tester:
- xUnit
- nUnit
- MSTest
I dette kurset kommer vi til å bruke xUnit. Dette valget er litt vilkårlig ettersom alle rammeverkene over vil være tilstrekkelig til formålet vårt, som er å vise hvordan man kan sette opp testprosjekter og komme i gang med å skrive tester. Dersom du ønsker å vite mer om de ulike testrammeverkene, kan du lese mer om dem her: https://docs.microsoft.com/en-us/dotnet/core/testing/#testing-tools
Kjør følgende kommando for å opprette enhetstestprosjektet
dotnet new xunit -lang F# -o test/unit -n NRK.Dotnetskolen.UnitTests
The template "xUnit Test Project" was created successfully.
Processing post-creation actions...
Running 'dotnet restore' on test/unit/NRK.Dotnetskolen.UnitTests.fsproj...
Determining projects to restore...
Restored C:\Dev\n[email protected]\dotnetskolen\test\unit\NRK.Dotnetskolen.UnitTests.fsproj (in 1.31 sec).
Restore succeeded.
Du skal nå ha følgende mappestruktur
src
└── api
└── NRK.Dotnetskolen.Api.fsproj
└── Program.fs
test
└── unit
└── NRK.Dotnetskolen.UnitTests.fsproj
└── Program.fs
└── Tests.fs
Åpne filen NRK.Dotnetskolen.UnitTests.fsproj
:
<Project Sdk="Microsoft.NET.Sdk">
<PropertyGroup>
<TargetFramework>net6.0</TargetFramework>
<IsPackable>false</IsPackable>
<GenerateProgramFile>false</GenerateProgramFile>
</PropertyGroup>
<ItemGroup>
<Compile Include="Tests.fs" />
<Compile Include="Program.fs" />
</ItemGroup>
<ItemGroup>
<PackageReference Include="Microsoft.NET.Test.Sdk" Version="16.11.0" />
<PackageReference Include="xunit" Version="2.4.1" />
<PackageReference Include="xunit.runner.visualstudio" Version="2.4.3">
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
<PrivateAssets>all</PrivateAssets>
</PackageReference>
<PackageReference Include="coverlet.collector" Version="3.1.0">
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
<PrivateAssets>all</PrivateAssets>
</PackageReference>
</ItemGroup>
</Project>
I prosjektfilen kan vi se at enhetstestprosjektet:
- Skal kompileres til .NET 6
- Inneholder to kildekodefiler
Tests.fs
Program.fs
- Har referanser til fire NuGet-pakker
Microsoft.NET.Test.Sdk
- Pakke for å bygge .NET testprosjekterxunit
- Bibliotek for å skrive enhetstesterxunit.runner.visualstudio
- Pakke for å kjøre Xunit-tester i "Test explorer" i Visual Studio https://docs.microsoft.com/en-us/visualstudio/test/run-unit-tests-with-test-explorer?view=vs-2019coverlet.collector
- bibliotek for å få code coverage statistikk for prosjekter https://docs.microsoft.com/en-us/dotnet/core/testing/unit-testing-code-coverage?tabs=windows
Vi ser nærmere på hva NuGet-pakker er i steg 4.
Åpne filen Tests.fs
:
module Tests
open System
open Xunit
[<Fact>]
let ``My test`` () =
Assert.True(true)
Øverst i filen blir det definert en F#-modul Tests
. I tillegg blir modulene System
og Xunit
åpnet, som kommer fra hhv. basebiblioteket til Microsoft, og biblioteket Xunit. Videre blir det definert en test ``My test``
. Måten vi ser at det er en test på er ved å se at den er annotert med [<Fact>]
. Xunit opererer med to annotasjoner for tester:
[<Fact>]
[<Theory>]
Forskjellen på disse blir nærmere forklart i steget om enhetstester.
Merk at
``<variabelnavn med mellomrom>``
er brukt for å kunne ha et variabelnavn som inneholder mellomrom. På denne måten kan man ha et funksjonsnavn som beskriver testen og samtidig er lesbar for mennesker.
For å kjøre testen i enhetstestprosjektet kan du bruke følgende kommando
dotnet test test/unit/NRK.Dotnetskolen.UnitTests.fsproj
Determining projects to restore...
All projects are up-to-date for restore.
Unit -> C:\Dev\n[email protected]\dotnetskolen\test\unit\bin\Debug\net5.0\NRK.Dotnetskolen.UnitTests.dll
Test run for C:\Dev\n[email protected]\dotnetskolen\test\unit\bin\Debug\net5.0\NRK.Dotnetskolen.UnitTests.dll (.NETCoreApp,Version=v5.0)
Microsoft (R) Test Execution Command Line Tool Version 16.9.1
Copyright (c) Microsoft Corporation. All rights reserved.
Starting test execution, please wait...
A total of 1 test files matched the specified pattern.
Passed! - Failed: 0, Passed: 1, Skipped: 0, Total: 1, Duration: 2 ms - Unit.dll (net5.0)
På lik linje med dotnet run
, kan du alternativt gå inn i mappen til enhetstestprosjektet, og kjøre dotnet test
derfra:
cd test/unit
dotnet test
Determining projects to restore...
All projects are up-to-date for restore.
Unit -> C:\Dev\n[email protected]\dotnetskolen\test\unit\bin\Debug\net5.0\NRK.Dotnetskolen.UnitTests.dll
Test run for C:\Dev\n[email protected]\dotnetskolen\test\unit\bin\Debug\net5.0\NRK.Dotnetskolen.UnitTests.dll (.NETCoreApp,Version=v5.0)
Microsoft (R) Test Execution Command Line Tool Version 16.9.1
Copyright (c) Microsoft Corporation. All rights reserved.
Starting test execution, please wait...
A total of 1 test files matched the specified pattern.
Passed! - Failed: 0, Passed: 1, Skipped: 0, Total: 1, Duration: 2 ms - Unit.dll (net5.0)
For å opprette integrasjonstestprosjektet, kan du kjøre samme kommando som da du opprettet enhetstestprosjektet, men bytt ut Unit
med Integration
i navnet på testprosjektet, som vist under:
dotnet new xunit -lang F# -o test/integration -n NRK.Dotnetskolen.IntegrationTests
The template "xUnit Test Project" was created successfully.
Processing post-creation actions...
Running 'dotnet restore' on test\integration\NRK.Dotnetskolen.IntegrationTests.fsproj...
Determining projects to restore...
Restored C:\Dev\n[email protected]\dotnetskolen\test\integration\NRK.Dotnetskolen.IntegrationTests.fsproj (in 580 ms).
Restore succeeded.
Du skal nå ha følgende mappestruktur
src
└── api
└── NRK.Dotnetskolen.Api.fsproj
└── Program.fs
test
└── unit
└── NRK.Dotnetskolen.UnitTests.fsproj
└── Program.fs
└── Tests.fs
└── integration
└── NRK.Dotnetskolen.IntegrationTests.fsproj
└── Program.fs
└── Tests.fs
Foreløpig er prosjekt- og test-filene til integrasjonstestprosjektet helt like de fra enhetstestprosjektet (bortsett fra prosjektnavnet). Forskjellen på enhets- og integrasjonstestene blir tydeligere når vi skal skrive testene i hhv. steg 6 og steg 10.
For å kjøre testene i integrasjonstestprosjektet kan du bruke følgende kommando
dotnet test test/integration/NRK.Dotnetskolen.IntegrationTests.fsproj
Determining projects to restore...
All projects are up-to-date for restore.
Integration -> C:\Dev\n[email protected]\dotnetskolen\test\integration\bin\Debug\net5.0\NRK.Dotnetskolen.IntegrationTests.dll
Test run for C:\Dev\n[email protected]\dotnetskolen\test\integration\bin\Debug\net5.0\NRK.Dotnetskolen.IntegrationTests.dll (.NETCoreApp,Version=v5.0)
Microsoft (R) Test Execution Command Line Tool Version 16.9.1
Copyright (c) Microsoft Corporation. All rights reserved.
Starting test execution, please wait...
A total of 1 test files matched the specified pattern.
Passed! - Failed: 0, Passed: 1, Skipped: 0, Total: 1, Duration: 2 ms - Integration.dll (net5.0)
Steg 3 av 10 - 🔝 Gå til toppen ⬆ Forrige steg ⬇ Neste steg
Slik oppsettet er nå, har vi tre prosjekter som er uavhengige av hverandre. Annet enn at de ligger i samme mappe, er det ingenting som kobler dem sammen. For å kunne gjøre operasjoner som å legge til felles pakker, og kjøre alle testene for systemet vårt, kan vi knytte prosjektene sammen i en og samme løsning (solution). Å ha alle prosjektene i en og samme løsning gir også fordelen av at man kan åpne alle prosjektene samlet i en IDE.
For å opprette en solution med dotnet
kan du kjøre følgende kommando:
dotnet new sln -n Dotnetskolen
The template "Solution File" was created successfully.
Du skal nå ha fått filen Dotnetskolen.sln
slik som vist under
src
└── api
└── NRK.Dotnetskolen.Api.fsproj
└── Program.fs
test
└── unit
└── NRK.Dotnetskolen.UnitTests.fsproj
└── Program.fs
└── Tests.fs
└── integration
└── NRK.Dotnetskolen.IntegrationTests.fsproj
└── Program.fs
└── Tests.fs
└── Dotnetskolen.sln
Hvis vi ser på innholdet i Dotnetskolen.sln
ser vi at det ikke er noen referanser til prosjektene våre enda
Microsoft Visual Studio Solution File, Format Version 12.00
# Visual Studio Version 16
VisualStudioVersion = 16.0.30114.105
MinimumVisualStudioVersion = 10.0.40219.1
Global
GlobalSection(SolutionConfigurationPlatforms) = preSolution
Debug|Any CPU = Debug|Any CPU
Release|Any CPU = Release|Any CPU
EndGlobalSection
GlobalSection(SolutionProperties) = preSolution
HideSolutionNode = FALSE
EndGlobalSection
EndGlobal
For å legge til referanser til prosjektene du har opprettet kan du kjøre følgende kommandoer
dotnet sln add src/api/NRK.Dotnetskolen.Api.fsproj
Project `src\api\NRK.Dotnetskolen.Api.fsproj` added to the solution.
dotnet sln add test/unit/NRK.Dotnetskolen.UnitTests.fsproj
Project `test\unit\NRK.Dotnetskolen.UnitTests.fsproj` added to the solution.
dotnet sln add test/integration/NRK.Dotnetskolen.IntegrationTests.fsproj
Project `test\integration\NRK.Dotnetskolen.IntegrationTests.fsproj` added to the solution.
Nå ser vi at Dotnetskolen.sln
inneholder referanser til prosjektene våre
Microsoft Visual Studio Solution File, Format Version 12.00
# Visual Studio Version 16
VisualStudioVersion = 16.0.30114.105
MinimumVisualStudioVersion = 10.0.40219.1
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "src", "src", "{E9AA01D7-D310-46F7-B383-06FE72E1AD22}"
EndProject
Project("{F2A71F9B-5D33-465A-A702-920D77279786}") = "NRK.Dotnetskolen.Api", "src\api\NRK.Dotnetskolen.Api.fsproj", "{08E3BB6B-BCDF-46C6-BB13-85C3402D0BF7}"
EndProject
Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "test", "test", "{66EF0E86-AB2C-4969-A41E-84D88D553785}"
EndProject
Project("{F2A71F9B-5D33-465A-A702-920D77279786}") = "NRK.Dotnetskolen.UnitTests", "test\unit\NRK.Dotnetskolen.UnitTests.fsproj", "{CB49FB6A-9415-4F77-A438-479A056C96D6}"
EndProject
Project("{F2A71F9B-5D33-465A-A702-920D77279786}") = "NRK.Dotnetskolen.IntegrationTests", "test\integration\NRK.Dotnetskolen.IntegrationTests.fsproj", "{4EFF064D-031F-4D0B-A6A9-560DDA1347C1}"
EndProject
Global
GlobalSection(SolutionConfigurationPlatforms) = preSolution
Debug|Any CPU = Debug|Any CPU
Release|Any CPU = Release|Any CPU
EndGlobalSection
GlobalSection(SolutionProperties) = preSolution
HideSolutionNode = FALSE
EndGlobalSection
GlobalSection(ProjectConfigurationPlatforms) = postSolution
{08E3BB6B-BCDF-46C6-BB13-85C3402D0BF7}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{08E3BB6B-BCDF-46C6-BB13-85C3402D0BF7}.Debug|Any CPU.Build.0 = Debug|Any CPU
{08E3BB6B-BCDF-46C6-BB13-85C3402D0BF7}.Release|Any CPU.ActiveCfg = Release|Any CPU
{08E3BB6B-BCDF-46C6-BB13-85C3402D0BF7}.Release|Any CPU.Build.0 = Release|Any CPU
{CB49FB6A-9415-4F77-A438-479A056C96D6}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{CB49FB6A-9415-4F77-A438-479A056C96D6}.Debug|Any CPU.Build.0 = Debug|Any CPU
{CB49FB6A-9415-4F77-A438-479A056C96D6}.Release|Any CPU.ActiveCfg = Release|Any CPU
{CB49FB6A-9415-4F77-A438-479A056C96D6}.Release|Any CPU.Build.0 = Release|Any CPU
{4EFF064D-031F-4D0B-A6A9-560DDA1347C1}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
{4EFF064D-031F-4D0B-A6A9-560DDA1347C1}.Debug|Any CPU.Build.0 = Debug|Any CPU
{4EFF064D-031F-4D0B-A6A9-560DDA1347C1}.Release|Any CPU.ActiveCfg = Release|Any CPU
{4EFF064D-031F-4D0B-A6A9-560DDA1347C1}.Release|Any CPU.Build.0 = Release|Any CPU
EndGlobalSection
GlobalSection(NestedProjects) = preSolution
{08E3BB6B-BCDF-46C6-BB13-85C3402D0BF7} = {E9AA01D7-D310-46F7-B383-06FE72E1AD22}
{CB49FB6A-9415-4F77-A438-479A056C96D6} = {66EF0E86-AB2C-4969-A41E-84D88D553785}
{4EFF064D-031F-4D0B-A6A9-560DDA1347C1} = {66EF0E86-AB2C-4969-A41E-84D88D553785}
EndGlobalSection
EndGlobal
Bildet under viser hvordan "Solution explorer" i Visual Studio viser løsningen.
Steg 4 av 10 - 🔝 Gå til toppen ⬆ Forrige steg ⬇ Neste steg
Siden vi har behov for å installere NuGet-pakker senere i kurset, skal vi sette opp pakkehåndteringsverktøyet Paket for løsningen.
Basebiblioteket i .NET inneholder mye grunnleggende funksjonalitet, men det inneholder ikke alt. For å slippe å skrive kode for mye brukt funksjonalitet på nytt hver gang man trenger den, er det en fordel om utviklere over hele verden kan dele kode med hverandre. De facto måte å dele kode i .NET på er via "NuGet". NuGet er både et offentlig repo for kode utviklet av tredjeparter (tilgjengelig på https://www.nuget.org/), og et verktøy for å laste opp og ned "NuGet-pakker" fra dette repoet.
NuGet som verktøy for å håndtere pakker i et prosjekt har imidlertid noen utfordringer:
- Transitive avhengigheter - Dersom et prosjekt har en avhengighet til
SomeNuGetPackage
, ogSomeNuGetPackage
har en avhengighet tilSomeOtherNuGetPackage
, erSomeOtherNuGetPackage
en transitiv avhengighet i prosjektet. NuGet skiller ikke transitive avhengigheter fra direkte avhengigheter ipackages.config
. Dermed har man ikke kontroll på hvilke avhengigheter ipackages.config
som er direkte, og hvilke som er transitive. - En annen utfordring med NuGet er at dersom to pakker refererer ulike versjoner av en annen pakke, vil NuGet velge den nyeste versjonen av de to pakkereferansene.
- I tillegg må hvert prosjekt i en solution definere hvilke avhengigheter det har, og hvilken versjon. Dermed kan prosjekter i samme solution ha ulike versjoner av samme pakke. Dette kan skape problemer.
Verktøyet "Paket" forsøker å løse utfordringene nevnt over, og er mye brukt i NRK TV og NRK Radio. Derfor blir Paket brukt i dette kurset.
Merk at selv om man bruker Paket som verktøy for å håndtere tredjepartsavhengigheter i en .NET-løsning, benytter man fortsatt NuGet sitt offentlige repo for å laste opp og ned avhengighetene.
Paket finnes som en utvidelse (også kalt "tool") til .NET CLI. Utvidelser i .NET CLI kan enten installeres som globale (tilgjengelig for alle .NET-løsninger på maskinen), eller lokale (kun for prosjektet utvidelsen blir installert i). I dette kurset installerer vi Paket lokalt for vår løsning. Fordelen med dette er at versjonen av Paket vi installerer kun gjelder for denne løsningen. Det gjør at andre løsninger på samme maskin kan ha andre versjoner av Paket. Dersom løsningen ligger på Git, vil i tillegg andre som kloner repoet kunne kjøre dotnet tool restore
, og få installert alle verktøyene de trenger.
Lokale utvidelser av .NET CLI defineres i en egen fil dotnet-tools.json
som ligger i en mappe .config
. Ettersom denne filen ikke finnes enda, oppretter vi den ved å kjøre følgende kommando
dotnet new tool-manifest
The template "Dotnet local tool manifest file" was created successfully.
Du skal nå ha fått dotnet-tools.json
-filen i .config
-mappen slik som vist under.
└── .config
└── dotnet-tools.json
src
└── ...
test
└── ...
└── Dotnetskolen.sln
dotnet-tools.json
inneholder imidlertid ingen utvidelser til .NET CLI enda
{
"version": 1,
"isRoot": true,
"tools": {}
}
For å legge til Paket i listen over utvidelser løsningen skal ha kan du kjøre følgende kommando
dotnet tool install paket
You can invoke the tool from this directory using the following commands: 'dotnet tool run paket' or 'dotnet paket'.
Tool 'paket' (version '5.257.0') was successfully installed. Entry is added to the manifest file C:\Dev\n[email protected]\dotnetskolen\.config\dotnet-tools.json.
Nå ser vi at Paket er lagt til i listen over tools i dotnet-tools.json
{
"version": 1,
"isRoot": true,
"tools": {
"paket": {
"version": "6.2.1",
"commands": [
"paket"
]
}
}
}
For å installere Paket kan du kjøre følgende kommando
dotnet tool restore
Tool 'paket' (version '6.2.1') was restored. Available commands: paket
Restore was successful.
Paket bruker følgende filer for å holde styr på pakkene i en løsning:
paket.dependencies
- en flat liste over alle avhengigheter som inngår på tvers av alle prosjektene i løsningen.<sti til prosjekt>/paket.references
- en flat liste over alle avhengigheter det gitte prosjektet har.paket.lock
- inneholder en oversikt over alle avhengigheter, både direkte og transitive, og hvilken versjon av dem som er brukt i løsningen.
Se forøvrig https://fsprojects.github.io/Paket/faq.html#What-files-should-I-commit for hvilke filer du skal inkludere i Git.
Da vi opprettet testprosjektene i steg 2, ble det lagt til referanser til NuGet-pakker som testprosjektene er avhengige av. Malene i .NET SDK benytter NuGet som pakkehåndteringssystem, og dermed ble disse prosjektreferansene lagt til i .fsproj
-filene til testprosjektene. Dette ser vi under i test/unit/NRK.Dotnetskolen.UnitTests.fsproj
. Det samme gjelder prosjektfilen til integrasjonstestprosjektet.
<Project Sdk="Microsoft.NET.Sdk">
<PropertyGroup>
<TargetFramework>net6.0</TargetFramework>
<IsPackable>false</IsPackable>
<GenerateProgramFile>false</GenerateProgramFile>
</PropertyGroup>
<ItemGroup>
<Compile Include="Tests.fs" />
<Compile Include="Program.fs" />
</ItemGroup>
<ItemGroup>
<PackageReference Include="Microsoft.NET.Test.Sdk" Version="16.11.0" />
<PackageReference Include="xunit" Version="2.4.1" />
<PackageReference Include="xunit.runner.visualstudio" Version="2.4.3">
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
<PrivateAssets>all</PrivateAssets>
</PackageReference>
<PackageReference Include="coverlet.collector" Version="3.1.0">
<IncludeAssets>runtime; build; native; contentfiles; analyzers; buildtransitive</IncludeAssets>
<PrivateAssets>all</PrivateAssets>
</PackageReference>
</ItemGroup>
</Project>
Siden vi ønsker å benytte Paket til å håndtere pakkene i løsningen vår, må vi migrere disse pakkene fra NuGet til Paket. Dette kan vi gjøre ved å bruke den innebygde kommandoen convert-from-nuget
i Paket, slik:
dotnet paket convert-from-nuget
Nå skal alle pakkene som er brukt på tvers av alle prosjektene i løsningen være lagt til i en ny fil paket.dependencies
:
source https://api.nuget.org/v3/index.json
nuget FSharp.Core
nuget coverlet.collector 3.1.0
nuget Microsoft.NET.Test.Sdk 16.11.0
nuget xunit 2.4.1
nuget xunit.runner.visualstudio 2.4.3
I tillegg skal pakkereferansene i et prosjekt være flyttet til en ny fil paket.references
i roten av prosjektmappen. Under ser vi hvordan paket.references
-filen ser ut for enhetstestprosjektet. Det samme gjelder integrasjonstestprosjektet.
Microsoft.NET.Test.Sdk
xunit
xunit.runner.visualstudio
coverlet.collector
FSharp.Core
Du kan lese mer om hvordan convert-from-nuget
-kommandoen fungerer her: https://fsprojects.github.io/Paket/convert-from-nuget-tutorial.html
Nå er du klar til å legge til avhengigheter i prosjektet ditt!
Merk at vi kunne ha latt være å opprette testprosjektene med malen
xunit
, og heller satt opp testprosjektene fra bunnen av ved å brukeconsole
-malen. Da hadde vi unngått å måtte migrere NuGet-pakkene til Paket. Kurset er imidlertid lagt opp på denne måten for å illustrere bruken av ulike maler i .NET SDK.
Takk til @laat som tipset om
convert-from-nuget
-kommandoen i Paket!
Steg 5 av 10 - 🔝 Gå til toppen ⬆ Forrige steg ⬇ Neste steg
Vi skal lage et API for å hente ut en forenklet elektronisk programguide (EPG) for ulike kanaler i NRK TV. Tanken er at dette API-et kunne levert datagrunnlaget til en programguide - f.eks. den som vises her: https://info.nrk.no/presse/tvguide/
Modellen vi bruker for EPG i dette kurset er forenklet sammenliknet med den som benyttes i PS API, og er kun brukt som eksempel.
En EPG kan sees på som en liste med sendinger, og for vårt eksempel i dette kurset inneholder en sending følgende felter:
- Tittel - Tittelen til programmet. Må være mellom 5 og 100 tegn (inklusiv), og kan kun bestå av store og små bokstaver, tall, og enkelte spesialtegn:
, . : - !
- Kanal - Kanalen sendingen går på. I vårt tilfelle begrenses mulige kanaler til NRK1 og NRK2, og må skrives med store bokstaver.
- Startdato- og tidspunkt - dato og tidspunkt for når sendingen starter.
- Sluttdato- og tidspunkt - dato og tidspunkt for når sendingen slutter. Må være etter startdato- og tidspunkt.
Nå som vi har spesifisert domenet vårt, kan vi modellere det i F#. Start med å opprett en ny fil Domain.fs
under src/api
:
└── .config
└── ...
src
└── api
└── Domain.fs
└── NRK.Dotnetskolen.Api.fsproj
└── Program.fs
test
└── ...
└── Dotnetskolen.sln
└── paket.dependencies
Lim inn innholdet under i Domain.fs
:
namespace NRK.Dotnetskolen
module Domain =
open System
type Sending = {
Tittel: string
Kanal: string
Starttidspunkt: DateTimeOffset
Sluttidspunkt: DateTimeOffset
}
type Epg = Sending list
Over definerer vi en F#-modul Domain
i namespacet NRK.Dotnetskolen
. I Domain
-modulen definerer vi domenemodellen vår, som består av to typer:
Sending
- modellerer et enkelt innslag i EPG-en, og inneholder feltene som ble definert i forrige seksjon- Tittel
- Kanal
- Starttidspunkt
- Sluttidspunkt
Epg
- en liste med sendinger
Vi åpnet også modulen System
for å få tilgang til typen DateTimeOffset
.
Legg merke til innrykket på linjene etter
module Domain =
. Dette inntrykket er påkrevd av F# for at koden skal kompilere riktig.
Inkluder Domain.fs
i api-prosjektet ved å legge til <Compile Include="Domain.fs" />
i src/api/NRK.Dotnetskolen.Api.fsproj
slik som vist under:
<Project Sdk="Microsoft.NET.Sdk">
<PropertyGroup>
<OutputType>Exe</OutputType>
<TargetFramework>net5.0</TargetFramework>
</PropertyGroup>
<ItemGroup>
<Compile Include="Domain.fs" />
<Compile Include="Program.fs" />
</ItemGroup>
</Project>
Merk at rekkefølgen filer blir inkludert i F#-prosjektfiler på har betydning. Dersom
modul A
er definert iModulA.fs
ogmodul B
er definert iModulB.fs
, ogmodul A
skal kunne åpnemodul B
måModulB.fs
ligge førModulA.fs
i prosjektfilen.Moduler i F# blir kompilert til det samme i CIL som statiske klasser i C#.
Nå som vi har definert domenemodellen vår, skal vi se hvordan vi kan ta den i bruk. Åpne Program.fs
i web-API-prosjektet og erstatt innholdet med følgende kode:
open System
open NRK.Dotnetskolen.Domain
let epg = [
{
Tittel = "Dagsrevyen"
Kanal = "NRK1"
Starttidspunkt = DateTimeOffset.Parse("2021-04-16T19:00:00+02:00")
Sluttidspunkt = DateTimeOffset.Parse("2021-04-16T19:30:00+02:00")
}
]
printfn "%A" epg
Her oppretter vi en variabel epg
som er en liste med sendinger, slik vi definerte i Domain.fs
.
Kjør API-prosjektet igjen med følgende kommando, og se at epg
-verdien blir skrevet til terminalen.
dotnet run --project src/api/NRK.Dotnetskolen.Api.fsproj
[{ Tittel = "Dagsrevyen"
Kanal = "NRK1"
Starttidspunkt = 16.04.2021 19:00:00 +02:00
Sluttidspunkt = 16.04.2021 19:30:00 +02:00 }]
Merk at noen har rapportert om problemer med feilmeldinger i Rider etter å ha lagt til linjen
open NRK.Dotnetskolen.Domain
. Dersom du opplever det samme kan du høyreklikke på "Solution"-noden i Rider, og klikke på "Unload" etterfulgt av "Reload". Dette skal forhåpentligvis rette opp i problemet.
Steg 6 av 10 - 🔝 Gå til toppen ⬆ Forrige steg ⬇ Neste steg
Domenemodellen som ble innført i forrige steg inneholder både strukturen til EPG-en, og valideringsreglene knyttet til dem. Så langt har vi kun modellert strukturen til domenemodellen i F# (at EPG består av en liste med sendinger, og hvilke felter hver sending inneholder). I dette steget skal vi implementere valideringsreglene i F#, og verifisere at vi har implementert dem riktig ved hjelp av enhetstester.
Vi ønsker å verifisere følgende regler fra domenet vårt:
- Tittel
- Må bestå av 5-100 tegn (inklusiv)
- Kan kun bestå av store og små bokstaver, tall, og følgende spesialtegn:
, . : - !
- Kanal
NRK1
ellerNRK2
.- Kun store bokstaver er lov.
- Sendetidspunkt
- Sluttidspunkt skal være etter starttidspunkt
La oss begynne med å verifisere at vi implementerer valideringsreglene for tittel riktig.
Ettersom tittel har lengdebegrensninger er det viktig å teste grensetilfellene til lengden. I tillegg er det viktig å teste at kun gyldige tegn er lov. Erstatt den eksisterende testen i Tests.fs
i enhetstestprosjektet med testene under.
module Tests
open Xunit
[<Theory>]
[<InlineData("abc12")>]
[<InlineData(".,-:!")>]
[<InlineData("ABCDEFGHIJABCDEFGHIJABCDEFGHIJABCDEFGHIJABCDEFGHIJABCDEFGHIJABCDEFGHIJABCDEFGHIJABCDEFGHIJABCDEFGHIJ")>]
let ``isTittelValid valid tittel returns true`` (tittel: string) =
let isTittelValid = isTittelValid tittel
Assert.True isTittelValid
[<Theory>]
[<InlineData("abcd")>]
[<InlineData("@$%&/")>]
[<InlineData("abcdefghijabcdefghijabcdefghijabcdefghijabcdefghijabcdefghijabcdefghijabcdefghijabcdefghijabcdefghija")>]
let ``isTittelValid invalid tittel returns false`` (tittel: string) =
let isTittelValid = isTittelValid tittel
Assert.False isTittelValid
Her har vi definert to enhetstester som begge tester funksjonen isTittelValid
. Den første testen verifiserer at isTittelValid
returnerer true
når tittelen er gyldig, mens den andre verifiserer det motsatte tilfellet. I xUnit annoterer man testfunksjoner med enten [<Fact>]
eller [<Theory>]
. Testfunksjoner annotert med [<Fact>]
vil kjøre én gang uten noen inputparametere, mens i testfunksjoner annotert med [<Theory>]
kan man ta inn parametere, og annotere testfunksjonen med [<InlineData>]
for å sende inn gitte inputparametere. Da vil testfunksjonen bli kjørt én gang per annotering med [<InlineData>]
.
Hvis du forsøker å kjøre testene, vil du se at testprosjektet ikke kompilerer fordi vi verken har referanse til API-prosjektet (hvor domenet vårt er definert) eller har definert funksjonen isTittelValid
enda.
dotnet test test/unit/NRK.Dotnetskolen.UnitTests.fsproj
Determining projects to restore...
All projects are up-to-date for restore.
C:\Dev\n[email protected]\dotnetskolen\test\unit\Tests.fs(13,24): error FS0039: The value or constructor 'isTittelValid' is not defined. [C:\Dev\n[email protected]\dotnetskolen\test\unit\NRK.Dotnetskolen.UnitTests.fsproj]
C:\Dev\n[email protected]\dotnetskolen\test\unit\Tests.fs(26,24): error FS0039: The value or constructor 'isTittelValid' is not defined. [C:\Dev\n[email protected]\dotnetskolen\test\unit\NRK.Dotnetskolen.UnitTests.fsproj]
For å validere en tittel bruker vi et regulært uttrykk som reflekterer reglene i domenet vårt. Åpne filen Domain.fs
i API-prosjektet, og legg til følgende open
-statement under open system
:
open System.Text.RegularExpressions
Lim deretter inn følgende kode på slutten av filen:
let isTittelValid (tittel: string) : bool =
let tittelRegex = Regex(@"^[\p{L}0-9\.,-:!]{5,100}$")
tittelRegex.IsMatch(tittel)
Det regulære uttrykket lister opp hvilke tegn som er gyldige i en gruppe (tegnene mellom mellom [
og ]
):
\p{L}
- syntaks for å spesifisere enhver bokstav i Unicode0-9
- tall\.,-:!
- spesialtegnene vi tillater
I tillegg spesifiserer {5,100}
at vi tillater 5-100 av tegnene i gruppen over.
For at enhetstestprosjektet skal få tilgang til funksjonen vi nettopp definerte i Domain.fs
må vi legge til en prosjektreferanse til API-prosjektet i enhetstestprosjektet. Det kan vi gjøre vha. .NET CLI med følgende kommando:
dotnet add ./test/unit/NRK.Dotnetskolen.UnitTests.fsproj reference ./src/api/NRK.Dotnetskolen.Api.fsproj
Reference `..\..\src\api\NRK.Dotnetskolen.Api.fsproj` added to the project.
Du kan se effekten av kommandoen over ved å åpne test/unit/NRK.Dotnetskolen.UnitTests.fsproj
:
<?xml version="1.0" encoding="utf-8"?>
<Project Sdk="Microsoft.NET.Sdk">
<PropertyGroup>
<TargetFramework>net6.0</TargetFramework>
<IsPackable>false</IsPackable>
<GenerateProgramFile>false</GenerateProgramFile>
</PropertyGroup>
<ItemGroup>
<Compile Include="Tests.fs" />
<Compile Include="Program.fs" />
</ItemGroup>
<ItemGroup>
<ProjectReference Include="..\..\src\api\NRK.Dotnetskolen.Api.fsproj" />
</ItemGroup>
<Import Project="..\..\.paket\Paket.Restore.targets" />
</Project>
I tillegg til å legge til en referanse til API-prosjektet i enhetstestprosjektet, må vi åpne NRK.Dotnetskolen.Domain
-modulen i Tests.fs
. Det kan du gjøre ved å legge til open NRK.Dotnetskolen.Domain
under open Xunit
i Tests.fs
:
module Tests
open Xunit
open NRK.Dotnetskolen.Domain
Nå skal testene kjøre vellykket:
dotnet test test/unit/NRK.Dotnetskolen.UnitTests.fsproj
Determining projects to restore...
All projects are up-to-date for restore.
NRK.Dotnetskolen.Api -> C:\Dev\n[email protected]\dotnetskolen\src\api\bin\Debug\net5.0\NRK.Dotnetskolen.Api.dll
NRK.Dotnetskolen.UnitTests -> C:\Dev\n[email protected]\dotnetskolen\test\unit\bin\Debug\net5.0\NRK.Dotnetskolen.UnitTests.dll
Test run for C:\Dev\n[email protected]\dotnetskolen\test\unit\bin\Debug\net5.0\NRK.Dotnetskolen.UnitTests.dll (.NETCoreApp,Version=v5.0)
Microsoft (R) Test Execution Command Line Tool Version 16.9.1
Copyright (c) Microsoft Corporation. All rights reserved.
Starting test execution, please wait...
A total of 1 test files matched the specified pattern.
Passed! - Failed: 0, Passed: 6, Skipped: 0, Total: 6, Duration: 6 ms - NRK.Dotnetskolen.UnitTests.dll (net5.0)
Legg merke til at testrapporten viser at seks tester ble kjørt. Foreløpig har vi kun definert to tester. Dette illustrerer at
xUnit
kjører tester en gang per annotasjon med[<InlineData>]
.
Reglene for kanal er ganske enkle ettersom det kun er to gyldige kanaler, og disse kun kan skrives med store bokstaver.
For å teste valideringsreglen for kanal trenger vi én positiv test per gyldige kanal, en negativ test for en kanal med små bokstaver, og en negativ test for en ugyldig kanal. Utvid Tests.fs
i med følgende tester for kanal:
[<Theory>]
[<InlineData("NRK1")>]
[<InlineData("NRK2")>]
let ``isKanalValid valid kanal returns true`` (kanal: string) =
let isKanalValid = isKanalValid kanal
Assert.True isKanalValid
[<Theory>]
[<InlineData("nrk1")>]
[<InlineData("NRK3")>]
let ``isKanalValid invalid kanal returns false`` (kanal: string) =
let isKanalValid = isKanalValid kanal
Assert.False isKanalValid
Før vi kjører testene igjen, definerer vi skallet for isKanalValid
i Domain.fs
:
let isKanalValid (kanal: string) : bool =
// Implementasjon her
☑️ Implementér isKanalValid
slik at enhetstestene passerer.
dotnet test ./test/unit/NRK.Dotnetskolen.UnitTests.fsproj
Determining projects to restore...
All projects are up-to-date for restore.
NRK.Dotnetskolen.Api -> C:\Dev\n[email protected]\dotnetskolen\src\api\bin\Debug\net5.0\NRK.Dotnetskolen.Api.dll
NRK.Dotnetskolen.UnitTests -> C:\Dev\n[email protected]\dotnetskolen\test\unit\bin\Debug\net5.0\NRK.Dotnetskolen.UnitTests.dll
Test run for C:\Dev\n[email protected]\dotnetskolen\test\unit\bin\Debug\net5.0\NRK.Dotnetskolen.UnitTests.dll (.NETCoreApp,Version=v5.0)
Microsoft (R) Test Execution Command Line Tool Version 16.9.1
Copyright (c) Microsoft Corporation. All rights reserved.
Starting test execution, please wait...
A total of 1 test files matched the specified pattern.
Passed! - Failed: 0, Passed: 10, Skipped: 0, Total: 10, Duration: 8 ms - NRK.Dotnetskolen.UnitTests.dll (net5.0)
Det siste vi skal validere i domenet vårt er at sluttidspunkt er etter starttidspunkt.
Under følger én enhetstest for validering av sendetidspunkter i Tests.fs
:
[<Fact>]
let ``areStartAndSluttidspunktValid start before end returns true`` () =
let starttidspunkt = DateTimeOffset.Now
let sluttidspunkt = starttidspunkt.AddMinutes 30.
let areStartAndSluttidspunktValid = areStartAndSluttidspunktValid starttidspunkt sluttidspunkt
Assert.True areStartAndSluttidspunktValid
Merk at du også må legge til følgende open
-statement i Tests.fs
for at DateTimeOffset.Now
fra kodesnutten over skal fungere:
open System
☑️ Legg til flere enhetstester du mener er nødvendig for å verifisere at validering av start- og sluttidspunkt er korrekt.
Merk at her bruker vi
[<Fact>]
-attributtet istedenfor[<Theory>]
.[<InlineData>]
-attributtet som man bruker med[<Theory>]
-attributtet krever verdier som er konstante ved kompilering. Ettersom vi benytterDateTimeOffset
-objekter (som ikke er konstante ved kompilering) som input tilareStartAndSluttidspunktValid
, bruker vi derfor[<Fact>]
-attributtet.
Funksjonen for å validere sendetidspunktene må undersøke om sluttidspunktet er større enn starttidspunktet. Lim inn skallet til areStartAndSluttidspunktValid
i Domain.fs
:
let areStartAndSluttidspunktValid (starttidspunkt: DateTimeOffset) (sluttidspunkt: DateTimeOffset) =
// Implementasjon her
☑️ Implementér areStartAndSluttidspunktValid
og få enhetstestene til å passere.
dotnet test ./test/unit/NRK.Dotnetskolen.UnitTests.fsproj
Determining projects to restore...
All projects are up-to-date for restore.
NRK.Dotnetskolen.Api -> C:\Dev\n[email protected]\dotnetskolen\src\api\bin\Debug\net5.0\NRK.Dotnetskolen.Api.dll
NRK.Dotnetskolen.UnitTests -> C:\Dev\n[email protected]\dotnetskolen\test\unit\bin\Debug\net5.0\NRK.Dotnetskolen.UnitTests.dll
Test run for C:\Dev\n[email protected]\dotnetskolen\test\unit\bin\Debug\net5.0\NRK.Dotnetskolen.UnitTests.dll (.NETCoreApp,Version=v5.0)
Microsoft (R) Test Execution Command Line Tool Version 16.9.1
Copyright (c) Microsoft Corporation. All rights reserved.
Starting test execution, please wait...
A total of 1 test files matched the specified pattern.
Passed! - Failed: 0, Passed: 13, Skipped: 0, Total: 13, Duration: 10 ms - NRK.Dotnetskolen.UnitTests.dll (net5.0)
Nå som vi har funksjoner for å validere de ulike feltene i en sending, kan vi lage en funksjon som validerer en hel sending.
Siden vi har skrevet enhetstester for valideringsfunksjonene til de ulike delene av en sending, kan enhetstestene for validering av hele sendingen være ganske enkle.
☑️ Skriv én positiv test for en gyldig sending, og én negativ test for en ugyldig sending i Tests.fs
som antar at det finnes en funksjon isSendingValid
i Domain.fs
Legg til følgende skall for isSendingValid
i Domain.fs
:
let isSendingValid (sending: Sending) : bool =
// Implementasjon her
☑️ Implementér isSendingValid
, og få enhetstestene til å passere:
dotnet test ./test/unit/NRK.Dotnetskolen.UnitTests.fsproj
Determining projects to restore...
All projects are up-to-date for restore.
NRK.Dotnetskolen.Api -> C:\Dev\n[email protected]\dotnetskolen\src\api\bin\Debug\net5.0\NRK.Dotnetskolen.Api.dll
NRK.Dotnetskolen.UnitTests -> C:\Dev\n[email protected]\dotnetskolen\test\unit\bin\Debug\net5.0\NRK.Dotnetskolen.UnitTests.dll
Test run for C:\Dev\n[email protected]\dotnetskolen\test\unit\bin\Debug\net5.0\NRK.Dotnetskolen.UnitTests.dll (.NETCoreApp,Version=v5.0)
Microsoft (R) Test Execution Command Line Tool Version 16.9.1
Copyright (c) Microsoft Corporation. All rights reserved.
Starting test execution, please wait...
A total of 1 test files matched the specified pattern.
Passed! - Failed: 0, Passed: 15, Skipped: 0, Total: 15, Duration: 12 ms - NRK.Dotnetskolen.UnitTests.dll (net5.0)
Merk at domenemodellen, slik den er implementert i steg 5 og steg 6, har en svakhet i at man kan opprette en
Sending
-verdi som er ugyldig. Vi har implementertisSendingValid
, men det er ingenting som hindrer oss i å opprette enSending
-verdi uten å bruke den. I ekstraoppgaven i steg 11 blir en alternativ tilnærming som bruker prinsipper fra domenedrevet design presentert. De resterende stegene i dette kurset frem til og med steg 10 kommer til å basere seg på domenemodellen slik den er definert her i steg 5 og steg 6 for å ikke innføre for mange prinsipper på en gang, og holde fokus på det kurset er ment for. Dersom du ønsker må du gjerne gå videre til steg 11 nå for å se hvordan det er gjort der. Husk at steg 11 er skrevet med forutsetning av at man har gjennomført kurset til og med steg 10 først.
Steg 7 av 10 - 🔝 Gå til toppen ⬆ Forrige steg ⬇ Neste steg
For å dokumentere hva API-et vårt tilbyr av operasjoner og responser skal vi lage en API-kontrakt. I NRK TV og NRK Radio definerer vi API-kontrakter ved bruk av OpenAPI (https://www.openapis.org/).
For å begrense omfanget av API-et vårt skal vi ha kun én operasjon i det:
- Hent EPG på en gitt dato
Responsen til denne operasjonen vil bestå av to lister med sendinger, én for hver kanal i domenet vårt, hvor hver sending har:
- Tittel - tekststreng som følger reglene definert i domenemodellen vår.
- Startdato- og tidspunkt - tekststreng som følger datoformatet i RFC 3339.
- Sluttdato- og tidspunkt - tekststreng som følger datoformatet i RFC 3339. Er garantert å være større enn startdato- og tidspunkt.
Før vi definerer selve kontrakten til API-et i en OpenAPI-spesifikasjon, skal vi definere et JSON Schema for innholdet i responsen til operasjonen i API-et vårt. Dette er vist under.
{
"$schema": "https://json-schema.org/draft/2020-12/schema",
"type": "object",
"properties": {
"nrk1": {
"type": "array",
"items": {
"$ref": "#/components/schemas/Sending"
}
},
"nrk2": {
"type": "array",
"items": {
"$ref": "#/components/schemas/Sending"
}
}
},
"required": [
"nrk1",
"nrk2"
],
"components": {
"schemas": {
"Tittel": {
"type": "string",
"pattern": "^[\\p{L}0-9\\.,-:!]{5,100}$",
"example": "Dagsrevyen",
"description": "Programtittel"
},
"Sending": {
"type": "object",
"properties": {
"tittel": {
"$ref": "#/components/schemas/Tittel"
},
"starttidspunkt": {
"type": "string",
"format": "date-time",
"description": "Startdato- og tidspunkt for sendingen."
},
"sluttidspunkt": {
"type": "string",
"format": "date-time",
"description": "Sluttdato- og tidspunkt for sendingen. Er alltid større enn sendingens startdato- og tidspunkt."
}
},
"required": [
"tittel",
"starttidspunkt",
"sluttidspunkt"
]
}
}
}
}
Her ser vi at responsen består av et objekt med to felter: nrk1
og nrk2
, som begge er en liste med sendingene på de respektive kanalene. Hver sending inneholder en tittel, samt start- og sluttidspunkt. Hver av feltene er tekststrenger som følger valideringsreglene vi har definert i domenet vårt. Tittel
har pattern
lik det regulære uttrykket vi benyttet i isTittelValid
i Domain.fs
. Starttidspunkt
og Sluttidspunkt
har format: "date-time"
, som følger datoformatet i RFC 3339.
Foreløpig skal vi ikke gjøre noe mer med JSON schemaet enn å ha det som dokumentasjon på API-et vårt. Lag en ny mappe docs
i rotmappen din med en ny fil epg.schema.json
hvor du limer inn JSON schemaet over. Du skal nå ha følgende mappehierarki:
└── .config
└── ...
└── docs
└── epg.schema.json
└── src
└── ...
└── test
└── ...
└── Dotnetskolen.sln
└── paket.dependencies
Nå som vi har formatet på innholdet i responsen vår, kan vi definere Open API-spesifikasjonen for API-et vårt. La oss starte med å opprett en ny fil openapi.json
i docs
-mappen. Du skal nå ha følgende mappehierarki:
└── .config
└── ...
└── docs
└── epg.schema.json
└── openapi.json
└── src
└── ...
test
└── ...
└── Dotnetskolen.sln
└── paket.dependencies
La oss begynne med å definere litt metadata for kontrakten vår.
Lim inn følgende JSON i openapi.json
:
{
"openapi": "3.0.0",
"info": {
"title": "Dotnetskolen EPG-API",
"description": "API for å hente ut EPG for kanalene NRK1 og NRK2 i NRKTV",
"version": "0.0.1"
}
}
Her oppgir vi hvilken versjon av OpenAPI vi benytter, og litt metadata om API-et vårt. Fortsett med å legg til definisjon av hvilke URL-er som er eksponert i API-et vårt:
{
"openapi": "3.0.0",
"info": {
"title": "Dotnetskolen EPG-API",
"description": "API for å hente ut EPG for kanalene NRK1 og NRK2 i NRKTV",
"version": "0.0.1"
},
"paths": {
"/epg/{dato}": {
"get": {
}
}
}
}
Her har vi spesifisert at API-et vårt eksponerer URL-en /epg/{dato}
for HTTP GET
-forespørsler. La oss fortsette med å spesifisere parameteret dato
:
{
"openapi": "3.0.0",
"info": {
"title": "Dotnetskolen EPG-API",
"description": "API for å hente ut EPG for kanalene NRK1 og NRK2 i NRKTV",
"version": "0.0.1"
},
"paths": {
"/epg/{dato}": {
"get": {
"parameters": [
{
"description": "Dato slik den er definert i [RFC 3339](https://tools.ietf.org/html/rfc3339#section-5.6). Eksempel: 2021-11-15.",
"in": "path",
"name": "dato",
"required": true,
"schema": {
"type": "string",
"format": "date"
},
"example": "2021-11-15"
}
]
}
}
}
}
Her har vi spesifisert dato
-parameteret vårt, og sagt at:
- Det er påkrevd
- At det er en tekststreng som oppfyller formatet
date
i OpenAPI 2021-11-15
er et eksempel på en gyldig dato
Nå kan vi legge til hvilke responser endepunktet har: 200 OK
med EPG eller 400 Bad Request
ved ugyldig dato.
{
"openapi": "3.0.0",
"info": {
"title": "Dotnetskolen EPG-API",
"description": "API for å hente ut EPG for kanalene NRK1 og NRK2 i NRKTV",
"version": "0.0.1"
},
"paths": {
"/epg/{dato}": {
"get": {
"parameters": [
{
"description": "Dato slik den er definert i [RFC 3339](https://tools.ietf.org/html/rfc3339#section-5.6). Eksempel: 2021-11-15.",
"in": "path",
"name": "dato",
"required": true,
"schema": {
"type": "string",
"format": "date"
},
"example": "2021-11-15"
}
],
"responses": {
"200": {
"content": {
"application/json": {
"schema": {
"$ref": "./epg.schema.json"
}
}
},
"description": "OK"
},
"400": {
"content": {
"text/plain": {
"schema": {
"type": "string",
"example": "\"Ugyldig dato\""
}
}
},
"description": "Bad Request"
}
}
}
}
}
}
Til slutt legger vi til en ID for operasjonen, og en tekstlig beskrivelse av den.
{
"openapi": "3.0.0",
"info": {
"title": "Dotnetskolen EPG-API",
"description": "API for å hente ut EPG for kanalene NRK1 og NRK2 i NRKTV",
"version": "0.0.1"
},
"paths": {
"/epg/{dato}": {
"get": {
"parameters": [
{
"description": "Dato slik den er definert i [RFC 3339](https://tools.ietf.org/html/rfc3339#section-5.6). Eksempel: 2021-11-15.",
"in": "path",
"name": "dato",
"required": true,
"schema": {
"type": "string",
"format": "date"
},
"example": "2021-11-15"
}
],
"responses": {
"200": {
"content": {
"application/json": {
"schema": {
"$ref": "./epg.schema.json"
}
}
},
"description": "OK"
},
"400": {
"content": {
"text/plain": {
"schema": {
"type": "string",
"example": "\"Ugyldig dato\""
}
}
},
"description": "Bad Request"
}
},
"operationId": "hentEpgPåDato",
"description": "Henter EPG for NRK1 og NRK 2 på den oppgitte datoen. Returnerer 400 dersom dato er ugyldig. Listen med sendinger for en kanal er tom dersom det ikke finnes noen sendinger på den gitte dagen."
}
}
}
}
Kontrakten over er validert ved hjelp av https://editor.swagger.io/
Merk at i OpenAPI-kontrakten over benytter vi versjon
3.0.0
av OpenAPI. I denne versjonen er det ikke full støtte for JSON Schema. Man kan derfor ikke bruke alle features i JSON Schema i OpenAPI-kontrakten. Kontrakten vår bruker imidlertid kun features i JSON Schema som er støttet.OpenAPI 3.1.0
ble lansert 16. februar 2021, som har full støtte for alle features i JSON Schema. Det vil imidlertid ta noe tid før det er støtte for denne i tooling somReDoc
(brukt i steg 12)WebGUI
og linting. Takk til @laat som poengterte det.
I steg 12 ser vi på hvordan man kan sette opp en grafisk fremstilling av OpenAPI-dokumentasjonen som en egen HTML-side i API-et,. Merk at det forutsetter at du har utført steg 1-10 først. Dersom du ønsker å se en grafisk fremstilling nå kan du lime inn koden under på https://editor.swagger.io/.
Bare trykk "OK" dersom du blir spurt om å gjøre om fra JSON til YAML.
{
"openapi": "3.0.0",
"info": {
"title": "Dotnetskolen EPG-API",
"description": "API for å hente ut EPG for kanalene NRK1 og NRK2 i NRKTV",
"version": "0.0.1"
},
"paths": {
"/epg/{dato}": {
"get": {
"parameters": [
{
"description": "Dato slik den er definert i [RFC 3339](https://tools.ietf.org/html/rfc3339#section-5.6). Eksempel: 2021-11-15.",
"in": "path",
"name": "dato",
"required": true,
"schema": {
"type": "string",
"format": "date"
},
"example": "2021-11-15"
}
],
"responses": {
"200": {
"content": {
"application/json": {
"schema": {
"type": "object",
"properties": {
"nrk1": {
"type": "array",
"items": {
"$ref": "#/components/schemas/Sending"
}
},
"nrk2": {
"type": "array",
"items": {
"$ref": "#/components/schemas/Sending"
}
}
},
"required": [
"nrk1",
"nrk2"
]
}
}
},
"description": "OK"
},
"400": {
"content": {
"text/plain": {
"schema": {
"type": "string",
"example": "\"Ugyldig dato\""
}
}
},
"description": "Bad Request"
}
},
"operationId": "hentEpgPåDato",
"description": "Henter EPG for NRK1 og NRK 2 på den oppgitte datoen. Returnerer 400 dersom dato er ugyldig. Listen med sendinger for en kanal er tom dersom det ikke finnes noen sendinger på den gitte dagen."
}
}
},
"components": {
"schemas": {
"Tittel": {
"type": "string",
"pattern": "^[\\p{L}0-9\\.,-:!]{5,100}$",
"example": "Dagsrevyen",
"description": "Programtittel"
},
"Sending": {
"type": "object",
"properties": {
"tittel": {
"$ref": "#/components/schemas/Tittel"
},
"starttidspunkt": {
"type": "string",
"format": "date-time",
"description": "Startdato- og tidspunkt for sendingen."
},
"sluttidspunkt": {
"type": "string",
"format": "date-time",
"description": "Sluttdato- og tidspunkt for sendingen. Er alltid større enn sendingens startdato- og tidspunkt."
}
},
"required": [
"tittel",
"starttidspunkt",
"sluttidspunkt"
]
}
}
}
}
Merk at https://editor.swagger.io/ ikke støtter at JSON Schema og Open-API-kontrakt er definert i to forskjellige filer. Derfor er kontrakten over en sammenslåing av
epg.schema.json
ogopenapi.json
.
Steg 8 av 10 - 🔝 Gå til toppen ⬆ Forrige steg ⬇ Neste steg
I steg-5 definerte vi domenemodellen vår som en F#-type. Domenemodellen representerer EPG-en slik vi konseptuelt tenker på den, både når det gjelder struktur og regler for gyldige tilstander. API-kontrakter er ikke nødvendigvis en-til-en med domenemodeller.
- For det første kan strukturen til typene i API-et være annerledes enn i domenemodellen. Dette ser vi i vårt tilfelle hvor domenemodellen har alle sendinger, på tvers av kanaler, i én liste, mens API-kontrakten har én liste med sendinger per kanal.
- I tillegg er vi begrenset til å representere data med tekst i API-et ettersom HTTP er en tekstbasert protokoll. For eksempel benytter vi en
DateTimeOffset
til å representere start- og sluttidspunkt i domenemodellen vår, mens vi benytterstring
i OpenAPI-kontrakten vår.
For at vi skal kunne oversette domenemodellen til OpenAPI-kontrakten skal vi innføre en egen F#-type som reflekterer typene i OpenAPI-kontrakten vår. Generelt blir typer som representerer dataene våre slik vi kommuniserer med andre systemer på kalt "data transfer objects", eller "DTO".
Start med å opprett en fil Dto.fs
i mappen src/api
:
└── .config
└── ...
└── docs
└── ...
src
└── api
└── Domain.fs
└── Dto.fs
└── NRK.Dotnetskolen.Api.fsproj
└── Program.fs
test
└── ...
└── Dotnetskolen.sln
└── paket.dependencies
Lim inn innholdet under i Dto.fs
:
namespace NRK.Dotnetskolen
module Dto =
type SendingDto = {
Tittel: string
Starttidspunkt: string
Sluttidspunkt: string
}
type EpgDto = {
Nrk1: SendingDto list
Nrk2: SendingDto list
}
På samme måte som da vi opprettet domenemodellen, må vi legge til Dto.fs
i prosjektfilen til API-prosjektet:
<Project Sdk="Microsoft.NET.Sdk">
<PropertyGroup>
<OutputType>Exe</OutputType>
<TargetFramework>net6.0</TargetFramework>
</PropertyGroup>
<ItemGroup>
<Compile Include="Domain.fs" />
<Compile Include="Dto.fs" />
<Compile Include="Program.fs" />
</ItemGroup>
</Project>
Steg 9 av 10 - 🔝 Gå til toppen ⬆ Forrige steg ⬇ Neste steg
I dette steget skal vi sette opp et skall for web-API-et, og verifisere at vi når API-et ved å skrive en integrasjonstest. Før vi begynner å kode skal vi se på et par relevante konsepter i .NET.
Fra og med .NET Core opererer .NET med ulike SDK-prosjekttyper avhengig av hva slags type applikasjon man ønsker å utvikle. Via de ulike prosjekttype får man tilgang til forskjellig funksjonalitet knyttet til kompilering og publisering av prosjektene. Da vi opprettet API- og testprosjektene fikk vi prosjekter med den grunnleggende prosjekttypen .NET SDK
. Siden vi i dette steget er avhengig av funksjonalitet som finnes i .NET Web SDK
skal vi endre prosjekttypene til API- og integrasjonstestprosjektene.
Åpne filen src/api/NRK.Dotnetskolen.Api.fsproj
, og endre Sdk
-attributtet på Project
-elementet fra Microsoft.NET.Sdk
til Microsoft.NET.Sdk.Web
:
<Project Sdk="Microsoft.NET.Sdk.Web">
<PropertyGroup>
<OutputType>Exe</OutputType>
<TargetFramework>net6.0</TargetFramework>
</PropertyGroup>
<ItemGroup>
<Compile Include="Domain.fs" />
<Compile Include="Dto.fs" />
<Compile Include="Program.fs" />
</ItemGroup>
</Project>
Gjenta steget over for test/integration/NRK.Dotnetskolen.IntegrationTests.fsproj
for å endre SDK-prosjekttypen til integrasjonstestprosjektet:
<?xml version="1.0" encoding="utf-8"?>
<Project Sdk="Microsoft.NET.Sdk.Web">
<PropertyGroup>
<TargetFramework>net6.0</TargetFramework>
<IsPackable>false</IsPackable>
<GenerateProgramFile>false</GenerateProgramFile>
</PropertyGroup>
<ItemGroup>
<Compile Include="Tests.fs" />
<Compile Include="Program.fs" />
</ItemGroup>
<Import Project="..\..\.paket\Paket.Restore.targets" />
</Project>
Du kan lese mer om de ulike prosjekttypene i .NET her: https://docs.microsoft.com/en-us/dotnet/core/project-sdk/overview
Før vi setter opp skallet til web-API-et, skal vi se på noen grunnleggende konsepter som er brukt i .NET for å lage applikasjoner.
Når vi utvikler og kjører en applikasjon har vi behov for tilgang til felles ressurser som konfigurasjon, avhengigheter og logging. I tillegg ønsker vi å ha kontroll på hvordan prosessen til applikasjonen vår starter og slutter. Microsoft tilbyr et objekt, IHost
, som holder styr på disse tingene for oss. Typisk bygger man opp og starter et IHost
-objekt i Program.fs
. Det skal vi gjøre nå ved å kalle en innebygd funksjon i Microsoft sitt bibliotek Host.CreateDefaultBuilder
.
Åpne Program.fs
i web-API-prosjektet og erstatt innholdet med følgende:
open Microsoft.Extensions.Hosting
Host.CreateDefaultBuilder().Build().Run()
Her åpner vi Microsoft.Extensions.Hosting
for å få tilgang til CreateDefaultBuilder
. CreateDefaultBuilder
kommer fra biblioteket til Microsoft, og sørger for å lese konfigurasjon, sette opp grunnleggende logging, og setter filstien til ressursfilene til applikasjonen (også kalt "content root").
Til slutt bygger vi hosten vår, og starter den slik Host.CreateDefaultBuilder().Build().Run()
.
Du kan kjøre hosten med følgende kommando:
dotnet run --project ./src/api/NRK.Dotnetskolen.Api.fsproj
info: Microsoft.Hosting.Lifetime[0]
Application started. Press Ctrl+C to shut down.
info: Microsoft.Hosting.Lifetime[0]
Hosting environment: Production
info: Microsoft.Hosting.Lifetime[0]
Content root path: C:\Dev\n[email protected]\dotnetskolen\src\api
Foreløpig gjør ikke hosten vår noen ting. Den bare starter, og kjører helt til vi avslutter den ved å trykke Ctrl+C
i terminalen. I outputen over ser vi imidlertid tre logginnslag av typen info
som er blitt skrevet av hosten. Dette illustrerer at CreateDefaultBuilder
har satt opp logging til konsoll. Logginnslagene forteller at applikasjonen har startet, at miljøet er Production
, og hva filstien til "content root" er.
Trykk Ctrl+C
for å stoppe hosten:
// Trykker `Ctrl+C`
info: Microsoft.Hosting.Lifetime[0]
Application is shutting down...
Production
er default miljø i .NET med mindre annet er spesifisert. Du kan lese mer om miljøer i .NET her: https://docs.microsoft.com/en-us/aspnet/core/fundamentals/environments?view=aspnetcore-5.0Du kan lese mer om
Host
-konseptet og hva det innebærer her: https://docs.microsoft.com/en-us/aspnet/core/fundamentals/host/generic-host?view=aspnetcore-6.0
Microsoft har laget et rammeverk for web-applikasjoner i .NET, ASP.NET (ASP står for "active server pages"). Web-applikasjoner i ASP.NET er konfigurerbare og modulære, og gjennom å konfigurere modulene i den har man kontroll på hvordan HTTP-forespørsler blir prosessert helt fra de kommer inn til serveren, og til HTTP-responsen blir sendt tilbake til klienten. Modulene i denne sammenhengen kalles mellomvare (eller "middleware" på engelsk), og de henger sammen i en lenket liste hvor HTTP-forespørselen blir prosessert suksessivt av mellomvarene i listen. Denne lenkede listen blir omtalt som "middleware pipeline".
Du kan se en illustrasjon av hvordan mellomvarer henger sammen i ASP.NET her: https://learn.microsoft.com/en-us/aspnet/core/fundamentals/middleware/?view=aspnetcore-6.0#create-a-middleware-pipeline-with-webapplication
Alle mellomvarer har i utgangspunktet anledning til å prosessere HTTP-forespørselen både før og etter den neste mellomvaren i listen prosesserer den, og kan på den måten være med å påvirke responsen som blir sendt tilbake til klienten. Enhver mellomvare har ansvar for å kalle den neste mellomvaren. På denne måten kan en mellomvare stoppe videre prosessering av forespørselen også. Et eksempel på en slik mellomvare er autentisering, hvor man ikke sender forespørselen videre i pipelinen dersom den ikke er tilstrekkelig autentisert. Pga. denne kortslutningen ligger autentisering tidlig i listen over mellomvarer.
Hosten vi opprettet i forrige avsnitt er et utgangspunkt for hvilken som helst applikasjon. Det kan bli f.eks. en bakgrunnstjeneste eller en web-applikasjon. Siden vi skal lage et web-API skal vi gå videre med å tilpasse hosten til å bli en web-server. Microsoft har laget en spesiell funksjon akkurat til dette formålet: WebApplication.CreateBuilder
. Denne likner på Host.CreateDefaultBuilder
som vi brukte i tidligere i avsnittet om host, bare at hosten den lager er en web-server som har mulighet til å konfigurere en "middleware pipeline". For å lage en web-applikasjon istedenfor en generisk applikasjon, åpne Microsoft.AspNetCore.Builder
, og bytt ut linjen Host.CreateDefaultBuilder().Build().Run()
med WebApplication.CreateBuilder().Build().Run()
slik at Program.fs
i API-prosjektet nå ser slik ut:
open Microsoft.AspNetCore.Builder
WebApplication.CreateBuilder().Build().Run()
WebApplication.CreateBuilder
sørger bl.a. for å sette opp Kestrel som web-server for applikasjonen vår. I tillegg returnerer den et objekt av typen WebApplicationBuilder
som vi kan bruke til å konfigurere web-applikasjonen etter våre behov. Vi kaller umiddelbart på WebApplicationBuilder
sin funksjon Build
for å bygge web-applikasjonen vår. Build
returnerer et objekt av typen WebApplication
, og vi kaller til slutt Run
-metoden på WebApplication
-objektet for å starte web-applikasjonen.
Hvis du nå kjører hosten igjen, vil du se to nye logginnslag:
dotnet run --project ./src/api/NRK.Dotnetskolen.Api.fsproj
info: Microsoft.Hosting.Lifetime[0]
Now listening on: http://localhost:5000
info: Microsoft.Hosting.Lifetime[0]
Now listening on: https://localhost:5001
...
Fra logginnslagene over ser vi at hosten vår nå lytter på HTTP-forespørsler på port 5000
og 5001
for hhv. HTTP og HTTPS. I og med at vi ikke har lagt til noen middlewares i pipelinen vår enda, svarer API-et med 404 Not Found
på alle forespørsler. Det kan du verifisere ved å åpne http://localhost:5000/ i en nettleser.
Du kan lese mer om middleware i .NET-web-applikasjoner her: https://docs.microsoft.com/en-us/aspnet/core/fundamentals/middleware/?view=aspnetcore-6.0
Nå som vi har blitt kjent med noen grunnleggende konsepter i .NET-applikasjoner, kan vi starte å sette sammen vårt eget web-API. For å gjøre det trenger vi en middleware pipeline som kan behandle HTTP-forespørslene som kommer inn til API-et vårt.
I .NET 6 innførte Microsoft "minimal APIs" som er en rekke metoder som gjør det enklere å komme i gang med å definere oppførslen til en host. For web-applikasjoner har Microsoft laget "minimal APIs" som gjør det enkelt å legge til funksjoner i "middleware pipelinen" til en web-applikasjon som håndterer innkommende HTTP-forespørsler for en gitt sti. Dette kan vi bruke for å lage et "ping"-endepunkt.
Åpne Program.fs
i API-prosjektet, og bytt ut innholdet i filen med koden under:
open System
open Microsoft.AspNetCore.Builder
let app = WebApplication.CreateBuilder().Build()
app.MapGet("/ping", Func<string>(fun () -> "pong")) |> ignore
app.Run()
Her har vi tatt vare på WebApplication
-objektet, som WebApplication.CreateBuilder().Build()
returnerer, i en egen variabel app
. Dette har vi gjort for å få tilgang til "minimal API"-metodene Microsoft har definert for WebApplication
. Videre har vi brukt én av dem, nemlig MapGet
, som tar inn to argumenter:
- En tekststreng som spesifiserer hvilken sti i URL-en som leder til denne funksjonen. I dette tilfellet
ping
. - En funksjon uten parametere som returnerer en tekststreng. I dette tilfellet
pong
.
Merk at som andre parameter til
MapGet
har vi oppgittFunc<string>(fun () -> "pong")
som strengt tatt ikke er en funksjon.Func
er .NET sin måte å opprette etDelegate
på. Delegates er .NET sin måte å pakke inn funksjonskall som objekter på. Siden "Minimal APIs" er skrevet for å fungere for hvilket som helst programmeringsspråk i .NET, har Microsoft vært nødt til å velge en modell som passer både for både det objektorienterte programmeringsparadigmet så vel som det funksjonelle programmeringsparadigmet. Dermed tarMapGet
strengt tatt inn etDelegate
-objekt som andre parameter, og måten man oppretter etDelegate
-objekt i F# på er ved å kalleFunc
sin konstruktør. I konstruktøren tilFunc
sender vi inn den anonyme F#-funksjonenfun () -> "pong"
.<string>
delen avFunc<string>
definerer hva slags type returverdien til den anonyme funksjonen har. Ettersom den anonyme funksjonen ikke tar inn noen parametere er det ikke spesifisert noe mer iFunc<string>
for det. Dersom den anonyme funksjonen hadde tatt inn et parameter av typenint
, hadde kallet tilFunc
sett slik ut:Func<int, string>
. Du kan lese mer om delegates i F# her: https://docs.microsoft.com/en-us/dotnet/fsharp/language-reference/delegatesDu kan lese mer om "minimal APIs" her: https://docs.microsoft.com/en-us/aspnet/core/fundamentals/minimal-apis?view=aspnetcore-6.0
Start API-et med følgende kommando:
dotnet run --project ./src/api/NRK.Dotnetskolen.Api.fsproj
info: Microsoft.Hosting.Lifetime[0]
Now listening on: https://localhost:5001
info: Microsoft.Hosting.Lifetime[0]
Now listening on: http://localhost:5000
info: Microsoft.Hosting.Lifetime[0]
Application started. Press Ctrl+C to shut down.
info: Microsoft.Hosting.Lifetime[0]
Hosting environment: Development
info: Microsoft.Hosting.Lifetime[0]
Content root path: C:\Dev\n[email protected]\dotnetskolen\src\api
Dette starter web-API-et på http://localhost:5000
. Verifiser at API-et fungerer ved å gå til http://localhost:5000/ping i nettleseren din og se at svaret er pong
.
Merk at dersom du forsøker å åpne applikasjonen på https://localhost:5001 kan du få beskjed om at nettleseren din ikke stoler på sertifikatet. For å komme rundt dette må man sette opp "self signed"-sertifikat på maskinen. Microsoft har skrevet en artikkel om hvordan å gjøre det her, men det å sette opp "self signed"-sertifikat er ikke en del av dette kurset.
Før vi fortsetter med å implementere web-API-et skal vi sette opp en integrasjonstest som verifiserer at API-et er oppe og kjører, og at det svarer på HTTP-forespørsler. Det skal vi gjøre ved å:
- Kjøre web-API-et vårt på en webserver som kjører i minnet under testen, en såkalt
TestServer
. - Sende forespørsler mot denne testserveren
- Verifisere at testserveren svarer med de verdiene vi forventer
Siden vi gir hele web-API-et vårt som input til testserveren er responsene vi får tilsvarende de web-API-et svarer med i et deployet miljø, og dermed kan vi være trygge på at API-et oppfyller kontrakten vi har definert også når det deployes.
Webserveren vi skal kjøre i integrasjonstestene er dokumentert her: https://docs.microsoft.com/en-us/dotnet/api/microsoft.aspnetcore.testhost.testserver?view=aspnetcore-6.0
Inspirasjonen til å skrive integrasjonstestene på måten beskrevet over er fra et kurs som @erikly har arrangert.
En liknende metode er også beskrevet i denne artikkelen skrevet av Microsoft: https://docs.microsoft.com/en-us/aspnet/core/test/integration-tests?view=aspnetcore-6.0. Artikkelen belager seg imidlertid på konsepter fra objektorientert programmering, og siden dette kurset fokuserer på F# og funksjonell programmering er det valgt å skrive integrasjonstestene med en mer funksjonell tilnærming.
For å kunne kjøre integrasjonstestene våre er vi avhengig av et par NuGet-pakker og en prosjektreferanse til web-API-et. De følgende avsnittene forklarer hvordan du legger dem til.
For å få tilgang til testserverem vi skal kjøre under integrasjonstestene er vi avhengig av NuGet-pakken Microsoft.AspNetCore.Mvc.Testing
.
Kjør følgende kommando fra rotenmappen din for å installere pakken:
dotnet paket add Microsoft.AspNetCore.Mvc.Testing --project ./test/integration/NRK.Dotnetskolen.IntegrationTests.fsproj
For å kunne referere til API-et vårt fra testprosjektet må vi legge til en referanse til API-prosjektet fra integrasjonstestprosjektet.
Gjør dette ved å kjør følgende kommando fra rotmappen din:
dotnet add ./test/integration/NRK.Dotnetskolen.IntegrationTests.fsproj reference ./src/api/NRK.Dotnetskolen.Api.fsproj
For å kunne lage en testserver som representerer API-et vårt når vi kjører testene må vi konfiguerere API-et vårt til å bruke en testserver, men kun når vi faktisk kjører testene, og ikke når API-et kjører ellers. For å få til dette må vi kalle en funksjon på WebApplicationBuilder
-objektet (som vi oppretter i main
-funksjonen i Program.fs
i API-prosjektet) når vi setter opp testserveren i testene.
Husk at Program.fs
i API-prosjektet nå ser slik ut:
open System
open Microsoft.AspNetCore.Builder
let app = WebApplication.CreateBuilder().Build()
app.MapGet("/ping", Func<string>(fun () -> "pong")) |> ignore
app.Run()
For å få tak i WebApplicationBuilder
-objektet som WebApplication.CreateBuilder
returnerer fra integrasjonstesten, trekker vi ut oppretting av WebApplicationBuilder
-objektet til en egen funksjon createWebApplicationBuilder
slik:
open System
open Microsoft.AspNetCore.Builder
let createWebApplicationBuilder () =
WebApplication.CreateBuilder()
let app = createWebApplicationBuilder().Build()
app.MapGet("/ping", Func<string>(fun () -> "pong")) |> ignore
app.Run()
Ved å bruke funksjonen createWebApplicationBuilder
fra integrasjonstestprosjektet kan vi konfiguerere WebApplicationBuilder
-objektet til å bruke testserveren når testene kjører.
I tillegg til å konfigurere WebApplicationBuilder
-objektet til å bruke en testserver trenger vi å få tak i app
-objektet fra main
-funksjonen i API-prosjektet for å opprette en HTTP-klient som sender HTTP-forespørsler til testserveren. For å få til dette trekker vi ut koden som oppretter og konfigurerer WebApplication
-objektet i API-et slik:
open System
open Microsoft.AspNetCore.Builder
let createWebApplicationBuilder () =
WebApplication.CreateBuilder()
let createWebApplication (builder: WebApplicationBuilder) =
let app = builder.Build()
app.MapGet("/ping", Func<string>(fun () -> "pong")) |> ignore
app
let builder = createWebApplicationBuilder()
let app = createWebApplication builder
app.Run()
Ved å bruke funksjonen createWebApplication
fra integrasjonstestprosjektet kan vi hente ut WebApplication
-objektet som representerer hele web-API-et vårt, og sende HTTP-forespørsler mot det fra integrasjonstestene våre.
For å kunne referere til de to nye funksjonene vi lagde i API-prosjektet, createWebApplicationBuilder
og createWebApplication
, fra integrasjonstestprosjektet må vi legge dem i en egen modul, slik:
namespace NRK.Dotnetskolen.Api
module Program =
open System
open Microsoft.AspNetCore.Builder
let createWebApplicationBuilder () =
WebApplication.CreateBuilder()
let createWebApplication (builder: WebApplicationBuilder) =
let app = builder.Build()
app.MapGet("/ping", Func<string>(fun () -> "pong")) |> ignore
app
let builder = createWebApplicationBuilder()
let app = createWebApplication builder
app.Run()
Merk at vi her også la til linjen
namespace NRK.Dotnetskolen.Api
øverst. Dette setter modulenProgram
i kontekst avNRK.Dotnetskolen.Api
, og gjør at når vi skal referere til funksjonenecreateWebApplicationBuilder
ogcreateWebApplication
må vi åpneNRK.Dotnetskolen.Api.Program
.
Nå er vi klare til å kunne sette opp integrasjonstestene. Åpne Tests.fs
i integrasjonstestprosjektet, og erstatt innholdet i filen med koden under:
module Tests
open System.Net.Http
open System.Threading.Tasks
open Xunit
open Microsoft.AspNetCore.TestHost
open NRK.Dotnetskolen.Api.Program
let runWithTestClient (test: HttpClient -> Task<unit>) =
task {
let builder = createWebApplicationBuilder()
builder.WebHost.UseTestServer() |> ignore
use app = createWebApplication builder
do! app.StartAsync()
let testClient = app.GetTestClient()
do! test testClient
}
[<Fact>]
let ``Get "ping" returns "pong"`` () =
runWithTestClient (fun httpClient ->
task {
let! response = httpClient.GetStringAsync("ping")
Assert.Equal(response, "pong")
}
)
La oss se litt nærmere på hva denne koden gjør.
Først definerer vi en modul som heter Tests
:
module Tests
Deretter åpner vi de namespacene vi er avhengige av:
open System.Net.Http
open System.Threading.Tasks
open Xunit
open Microsoft.AspNetCore.TestHost
open NRK.Dotnetskolen.Api.Program
System.Net.Http
for å ha tilgang tilHttpClient
-typenSystem.Threading.Tasks
for å ha tilgang tilTask
-typenXunit
for å ha tilgang til[<Fact>]
som attributt til test-funksjonene våreMicrosoft.AspNetCore.TestHost
for å ha tilgang til funksjoneneUseTestServer
ogGetTestClient
som lar hhv. lar oss konfigurereWebApplicationBuilder
til å bruke testserveren, samt hente ut enHttpClient
som sender forespørsler til testserveren.NRK.Dotnetskolen.Api.Program
for å ha tilgang til funksjonenecreateWebApplicationBuilder
ogcreateWebApplication
for å kunne hente ut hhv.WebApplicationBuilder
-objektet ogWebApplication
-objektet til API-et vårt.
Deretter definerer vi en funksjon runWithTestClient
. Hensikten med denne funksjonen er å samle koden som konfigurerer testserveren og henter ut HttpClient
-objektet som kan sende HTTP-forespørsler til denne.
let runWithTestClient (test: HttpClient -> Task<unit>) =
task {
let builder = createWebApplicationBuilder()
builder.WebHost.UseTestServer() |> ignore
use app = createWebApplication builder
do! app.StartAsync()
let testClient = app.GetTestClient()
do! test testClient
}
runWithTestClient
kaller createWebApplicationBuilder
fra API-prosjektet, og konfigurerer WebHost
-objektet til å bruke testserveren.
Deretter kaller runWithTestClient
createWebApplication
med WebApplicationBuilder
som argument for å få WebApplication
-objektet som representerer API-et vårt, og starter web-API-et.
Videre henter runWithTestClient
ut et HttpClient
-objekt fra WebApplication
-objektet. Det er dette HttpClient
-objektet som kan sende HTTP-forespørsler til testserveren.
Til slutt kaller runWithTestClient
test
-funksjonen og sender med testClient
som parameter.
Merk at
runWithTestClient
lager ettask
"computation expression" (task {...}
). Med slike blokker kan vi sette i gang .NET tasks, som lar oss kjøre kode asynkront. F# har to typer "computation expressions" for å kjøre asynkron kode på:async
ogtask
.async
kom først, og er hittil det mest vanlige å bruke, menstask
kom i F# 6 inkludert i .NET 6. Du kan lese mer om "computation expressions",async
ogtask
her:
- https://docs.microsoft.com/en-us/dotnet/fsharp/language-reference/computation-expressions
- https://docs.microsoft.com/en-us/dotnet/fsharp/language-reference/async-expressions
- https://docs.microsoft.com/en-us/dotnet/fsharp/language-reference/task-expressions
Merk at vi bruker
use
-kodeordet når vi oppretter test-HTTP-klienten. Dette sørger for at kompilatoren rydder opp ressursene som objektet bruker når testen er ferdig.
Til slutt definerer vi en test Get "ping" returns "pong"
som kaller runWithTestClient
med en anonym funksjon. Den anonyme funksjonen tar inn HttpClient
-objektet som sender HTTP-forespørsler til testserveren vår. Deretter kaller den httpClient.GetStringAsync("/ping")
for å sende en HTTP GET til testserveren med ping
som sti i URL-en. Til slutt verifiserer den at responsen fra testserveren var pong
.
[<Fact>]
let ``Get "ping" returns "pong"`` () =
runWithTestClient (fun httpClient ->
task {
let! response = httpClient.GetStringAsync("/ping")
Assert.Equal(response, "pong")
}
)
Merk at her bruker vi
let!
istedenforlet
førhttpClient.GetStringAsync(/ping")
. Ved å brukelet!
venter vi på at den asynkrone handlingen på høyresiden av=
(httpClient.GetStringAsync("/ping")
) returnerer før vi går videre.
Kjør integrasjonstesten med følgende kommando:
dotnet test ./test/integration/NRK.Dotnetskolen.IntegrationTests.fsproj
Passed! - Failed: 0, Passed: 1, Skipped: 0, Total: 1, Duration: < 1 ms - NRK.Dotnetskolen.IntegrationTests.dll (net6.0)
Steg 10 av 10 - 🔝 Gå til toppen ⬆ Forrige steg
I forrige steg opprettet vi et skall for web-API-et ved å legge til et ping
-endepunkt med en tilhørende integrasjonstest. I dette steget skal vi utvide web-API-et med endepunkt for å hente EPG. I tillegg skal vi skrive integrasjonstester for å verifisere at implementasjonen av web-API-et er i henhold til Open API-dokumentasjonen vår. Vi bruker en testdrevet tilnærming ved at vi skriver en integrasjonstest som feiler, og deretter gjør vi endringer i API-et slik at testen passerer. Slik fortsetter vi til vi har implementert ferdig API-et vårt.
I den første integrasjonstesten skal vi sende en forespørsel til API-et vårt som henter ut EPG-en for dagen i dag, og validere at vi får 200 OK tilbake. Start med å legg til følgende "open"-statement før open System.Net.Http
i Tests.fs
-filen i integrasjonstestprosjektet.
open System
Legg deretter til følgende test etter ping
-testen i Tests.fs
-filen:
[<Fact>]
let ``Get EPG today returns 200 OK`` () =
runWithTestClient (fun httpClient ->
task {
let todayAsString = DateTimeOffset.Now.ToString "yyyy-MM-dd"
let url = $"/epg/{todayAsString}"
let! response = httpClient.GetAsync(url)
response.EnsureSuccessStatusCode() |> ignore
}
)
På tilsvarende måte som ping
-testen vår, bruker vi runWithTestClient
-funksjonen til å få en HTTP-klient som sender HTTP-forespørsler til testserveren vår. Deretter benytter vi HTTP-klienten til å sende en GET-forespørsel til /epg/<dagens dato>
. Vi forventer å få 200 OK i respons, og verifiserer dette ved å kalle response.EnsureSuccessStatusCode()
.
Kjør integrasjonstesten med følgende kommando:
dotnet test ./test/integration/NRK.Dotnetskolen.IntegrationTests.fsproj
[xUnit.net 00:00:00.73] Tests.Get EPG today returns 200 OK [FAIL]
Failed Tests.Get EPG today returns 200 OK [102 ms]
Error Message:
System.Net.Http.HttpRequestException : Response status code does not indicate success: 404 (Not Found).
...
Failed! - Failed: 1, Passed: 1, Skipped: 0, Total: 2, Duration: 10 ms - NRK.Dotnetskolen.IntegrationTests.dll (net6.0)
Som vi ser over feiler testen foreløpig ettersom web-API-et returnerer 404 (Not Found)
. La oss endre API-et slik at integrasjonstesten passerer.
Dersom vi ser på API-kontrakten vi definerte i steg 7 inneholder den én operasjon /epg/{dato}
som returnerer 200 OK
med den aktuelle EPG-en dersom alt er OK, og 400 Bad Request
dersom den ikke klarer å parse datoen:
...
"paths": {
"/epg/{dato}": {
"get": {
...
"responses": {
"200": {
...
"description": "OK"
},
"400": {
...
"description": "Bad Request"
}
}
...
"description": "Henter EPG for NRK1 og NRK 2 på den oppgitte datoen. Returnerer 400 dersom dato er ugyldig. Listen med sendinger for en kanal er tom dersom det ikke finnes noen sendinger på den gitte dagen."
}
}
}
}
Det er to ting som definerer operasjonen i API-et vårt:
- URL-en
/epg/{dato}
- At den er tilgjengelig gjennom HTTP
GET
-verbet
Dette kan vi bruke når vi skal definere operasjonen i WebApplication
-objektet vårt. Utvid createWebApplication
i Program.fs
i API-prosjektet med linjen app.MapGet("/epg/{date}", Func<string, string>(fun (date) -> date)) |> ignore
slik:
let createWebApplication (builder: WebApplicationBuilder) =
let app = builder.Build()
app.MapGet("/ping", Func<string>(fun () -> "pong")) |> ignore
app.MapGet("/epg/{date}", Func<string, string>(fun date -> date)) |> ignore
app
Her spesifiserer vi at vi ønsker å kjøre den anonyme funksjonen fun date -> date)
for HTTP GET
-forespørsler til URL-en epg/{date}
, hvor {date}
matcher tekststrengen oppgitt i URL-en etter /epg/
.
Legg merke til bruken av delegates her også gjennom
Func<string, string>(fun date -> date)
. Her ser vi at delegaten vår tar inn et parameter av typenstring
, og returnerer en verdi av typenstring
.
Start API-et igjen og se hva som skjer dersom du går til http://localhost:5000/epg/2021-01-01 i nettleseren.
dotnet run --project ./src/api/NRK.Dotnetskolen.Api.fsproj
Nå skal også integrasjonstesten som verifiserer om API-et vårt svarer på /epg/{dato}
passere. Det kan vi se ved å kjøre følgende kommando:
dotnet test ./test/integration/NRK.Dotnetskolen.IntegrationTests.fsproj
Passed! - Failed: 0, Passed: 2, Skipped: 0, Total: 2, Duration: 9 ms - NRK.Dotnetskolen.IntegrationTests.dll (net6.0)
I den neste testen skal vi verifisere at API-et validerer datoen som oppgis i URL-en. Utvid Tests.fs
i integrasjonstestprosjektet med følgende open
-statement og testfunksjon:
open System.Net
[<Fact>]
let ``Get EPG invalid date returns bad request`` () =
runWithTestClient (fun httpClient ->
task {
let invalidDateAsString = "2021-13-32"
let url = $"/epg/{invalidDateAsString}"
let! response = httpClient.GetAsync(url)
Assert.Equal(HttpStatusCode.BadRequest, response.StatusCode)
}
)
Her sender vi inn en ugyldig dato, og forventer å få 400 Bad Request som respons.
Kjør integrasjonstestene igjen med følgende kommando:
dotnet test ./test/integration/NRK.Dotnetskolen.IntegrationTests.fsproj
[xUnit.net 00:00:00.81] Tests.Get EPG invalid date returns bad request [FAIL]
Failed Tests.Get EPG invalid date returns bad request [10 ms]
Error Message:
Assert.Equal() Failure
Expected: BadRequest
Actual: OK
...
Failed! - Failed: 1, Passed: 2, Skipped: 0, Total: 3, Duration: 37 ms - NRK.Dotnetskolen.IntegrationTests.dll (net6.0)
Den nye testen vi la til feiler fordi API-et ikke validerer den oppgitte datoen. La oss endre implementasjonen av web-API-et slik at testen passerer.
Den anonyme funksjonen som håndterer HTTP GET-forespørsler til /epg/{dato}
gir ikke så mye verdi slik den står nå. La oss gå videre med å implementere operasjonen slik den er definert i API-kontrakten vår. Overordnet ønsker vi at funksjonen skal gjøre følgende:
- Validere datoen som er oppgitt i URL-en, og returnere
400 Bad Request
dersom den er ugyldig - Hente sendinger for den oppgitte datoen
- Returnere EPG på JSON-format som oppfyller API-kontrakten vår
La oss starte med å trekke ut den anonyme funksjonen til en egen funksjon epgHandler
som vi legger i en ny modul HttpHandlers
. Opprett en ny fil HttpHandlers.fs
som du legger i mappen src/api
slik:
...
src
└── api
└── NRK.Dotnetskolen.Api.fsproj
└── Domain.fs
└── Dto.fs
└── HttpHandlers.fs
└── Program.fs
...
Husk å legg til HttpHandlers.fs
i prosjektfilen til API-prosjektet:
<Project Sdk="Microsoft.NET.Sdk.Web">
<PropertyGroup>
<OutputType>Exe</OutputType>
<TargetFramework>net6.0</TargetFramework>
</PropertyGroup>
<ItemGroup>
<Compile Include="Domain.fs" />
<Compile Include="Dto.fs" />
<Compile Include="HttpHandlers.fs" />
<Compile Include="Program.fs" />
</ItemGroup>
<Import Project="..\..\.paket\Paket.Restore.targets" />
</Project>
Legg til følgende kode i HttpHandlers.fs
:
namespace NRK.Dotnetskolen.Api
module HttpHandlers =
let epgHandler (dateAsString: string) =
dateAsString
Her oppretter vi en modul HttpHandlers
i namespacet NRK.Dotnetskolen.Api
. I modulen har vi en funksjon epgHandler
, som tar inn en tekststreng, og foreløpig returnerer funksjonen den samme tekststrengen. Returverdien av epgHandler
er foreløpig lik som den anonyme funksjonen vi hadde i Program.fs
, men nå har vi anledning til å utvide den uten at koden i Program.fs
blir uoversiktlig.
Åpne modulen HttpHandlers
i Program.fs
og kall funksjonen epgHandler
istedenfor den anonyme funksjonen vi hadde:
open NRK.Dotnetskolen.Api.HttpHandlers
let createWebApplication (builder: WebApplicationBuilder) =
let app = builder.Build()
app.MapGet("/ping", Func<string>(fun () -> "pong")) |> ignore
app.MapGet("/epg/{date}", Func<string, string>(fun date -> epgHandler date)) |> ignore
app
La oss fortsette med å validere datoen vi får inn i epgHandler
-funksjonen. Lim inn følgende open
-statements, og parseAsDateTime
-funksjon før epgHandler
-funksjonen i HttpHandlers.fs
:
open System
open System.Globalization
open System.Threading.Tasks
let parseAsDateTime (dateAsString : string) : DateTimeOffset option =
try
let date = DateTimeOffset.ParseExact(dateAsString, "yyyy-MM-dd", CultureInfo.InvariantCulture, DateTimeStyles.None)
Some date
with
| _ -> None
parseAsDateTime
-funksjonen forsøker å parse tekststrengen den får som parameter til en dato på formatet yyyy-MM-dd
og returnerer en DateTimeOffset option
verdi som indikerer om det gikk bra eller ikke. parseAsDateTime
benytter DateTimeOffset.ParseExact
-funksjonen fra basebiblioteket til Microsoft. DateTimeOffset.ParseExact
kaster en Exception
dersom den oppgitte string
-verdien ikke matcher det oppgitte formatet. Derfor har vi en try/with
-blokk rundt kallet til funksjonen, og returnerer None
(ingen verdi) dersom DateTimeOffset.ParseExact
kaster Exception
, og Some date
dersom funksjonskallet lykkes.
Nå kan vi bruke parseAsDateTime
-funksjonen i epgHandler
til å returnere 400 Bad Request
dersom datoen er ugyldig. Legg til følgende open
-statement, og endre implementasjonen av epgHandler
slik:
open Microsoft.AspNetCore.Http
let epgHandler (dateAsString: string) =
match (parseAsDateTime dateAsString) with
| Some date -> Results.Ok(date)
| None -> Results.BadRequest("Invalid date")
Her bruker vi et match
-statement i F# som sammenlikner resultatet av å kalle parseAsDateTime dateAsString
med Some date
(i tilfellet datoen ble vellykket parset som en dato på formatet vi har spesifisert i parseAsDateTime
) eller None
i motsatt fall. Dersom datoen ble vellykket parset som en dato returnerer vi Results.Ok(date)
som setter statuskoden til 200 OK
og returnerer datoen. I motsatt fall returnerer vi Results.BadRequest("Invalid date")
som setter statuskoden til 400 Bad Request
, og returnerer teksten Invalid date
.
Siden vi nå har endret returtypen til epgHandler
fra string
til IResult
(samleinterface for blant annet Ok
og BadRequest
), må vi også endre typen til delegaten i MapGet("/epg/{date}"
. Åpne Microsoft.AspNetCore.Http
, og endre typen til delegaten slik:
open Microsoft.AspNetCore.Http
app.MapGet("/epg/{date}", Func<string, IResult>(fun date -> epgHandler date)) |> ignore
Start API-et igjen og se hva som skjer dersom du går til http://localhost:5000/epg/2021-01-01 i nettleseren.
dotnet run --project ./src/api/NRK.Dotnetskolen.Api.fsproj
Det vi nå får tilbake er ASP.NET sin serialisering av det parsede datoobjektet.
Kjør integrasjonstestene på nytt, og se at testen som verifiserer at API-et vårt responderer med 400 Bad Request
med en ugyldig dato passerer nå:
dotnet test ./test/integration/NRK.Dotnetskolen.IntegrationTests.fsproj
Passed! - Failed: 0, Passed: 3, Skipped: 0, Total: 3, Duration: 16 ms - NRK.Dotnetskolen.IntegrationTests.dll (net6.0)
I den siste testen skal vi verifisere at responsen API-et gir følger formatet vi har spesifisert i OpenAPI-kontrakten vår.
For å kunne validere at responsen fra web-API-et er i henhold til OpenAPI-kontrakten, skal vi benytte NuGet-pakken JsonSchema.Net
. Installer denne pakken ved å kjøre følgende kommando fra rotmappen din:
dotnet paket add JsonSchema.Net --project ./test/integration/NRK.Dotnetskolen.IntegrationTests.fsproj
For å kunne verifisere at responsen fra API-et vårt følger den definerte kontrakten, må vi inkludere JsonSchema-et for responsen vår i integrasjonstestprosjektet. Det kan vi gjøre ved å legge til følgende i slutten av samme ItemGroup
som Program.fs
og Tests.fs
i prosjektfilen til integrasjonstestprosjektet:
<Content Include="../../docs/epg.schema.json">
<CopyToOutputDirectory>Always</CopyToOutputDirectory>
</Content>
Legg deretter til følgende "open"-statement i Tests.fs
:
open Json.Schema
open System.Text.Json
Legg til slutt til følgende test i Test.fs
-klassen:
[<Fact>]
let ``Get EPG today return valid response`` () =
runWithTestClient (fun httpClient ->
task {
let todayAsString = DateTimeOffset.Now.ToString "yyyy-MM-dd"
let url = $"/epg/{todayAsString}"
let jsonSchema = JsonSchema.FromFile "./epg.schema.json"
let! response = httpClient.GetAsync(url)
response.EnsureSuccessStatusCode() |> ignore
let! bodyAsString = response.Content.ReadAsStringAsync()
let bodyAsJsonDocument = JsonDocument.Parse(bodyAsString).RootElement
let isJsonValid = jsonSchema.Validate(bodyAsJsonDocument, ValidationOptions(RequireFormatValidation = true)).IsValid
Assert.True(isJsonValid)
}
)
Denne testen bygger på de foregående testene vi har skrevet, og validerer i tillegg at responsen følger JsonSchema-et som vi definerte i OpenAPI-kontrakten:
let jsonSchema = JsonSchema.FromFile "./epg.schema.json"
oppretter en .NET-representasjon av JSON Schemaet vi definerte i kapittel 7let! bodyAsString = response.Content.ReadAsStringAsync()
henter ut innholdet i responsen som enstring
let bodyAsJsonDocument = JsonDocument.Parse(bodyAsString).RootElement
oppretter en .NET-representasjon av JSON-dokumentet som API-et returnerer, og henter en referanse til rotelementet i JSON-dokumentetlet isJsonValid = jsonSchema.Validate(bodyAsJsonDocument, ValidationOptions(RequireFormatValidation = true)).IsValid
benytter JSON Schemaet vårt til å validere om JSON-objektet som web-API-et returnerte tilfredstiller API-kontrakten
Kjør integrasjonstestene igjen med følgende kommando.
dotnet test ./test/integration/NRK.Dotnetskolen.IntegrationTests.fsproj
[xUnit.net 00:00:01.13] Tests.Get EPG today return valid response [FAIL]
Failed Tests.Get EPG today return valid response [98 ms]
Error Message:
Assert.True() Failure
Expected: True
Actual: False
...
Failed! - Failed: 1, Passed: 3, Skipped: 0, Total: 4, Duration: 408 ms - NRK.Dotnetskolen.IntegrationTests.dll (net5.0)
Testen feiler. La oss implementere ferdig API-et.
Før vi koder videre skal vi ta en snartur innom et mye brukt prinsipp i programvareutvikling: "Inversion of control" (IoC). Inversion of control går kort fortalt ut på at man lar kontrollen over implementasjonen av avhengighetene man har i koden sin ligge på utsiden av der man har behov for avhengigheten. På denne måten kan man endre hva som implementerer avhengigheten man har, og man kan enklere enhetsteste koden sin fordi man kan sende inn fiktive implementasjoner av avhengighetene.
Et eksempel på dette er dersom man har en funksjon isLoginValid
for å validere brukernavn og passord som kommer inn fra et innloggingsskjema, har man behov for å hente entiteten som korresponderer til det oppgitte brukernavnet fra brukerdatabasen. Ved å ta inn en egen funksjon getUser
i ValidateLogin
har man gitt kontrollen over hvordan getUser
skal implementeres til utsiden av ValidateLogin
-funksjonen.
let isLoginValid (getUser: string -> UserEntity) (username: string) (password: string) : bool ->
En måte å oppnå IoC på er å bruke "dependency injection" (DI). Da sender man inn de nødvendige avhengighetene til de ulike delene av koden sin fra utsiden. Dersom en funksjon A
har avhengiheter funksjonene B
og C
, og B
og C
har hhv. avhengiheter til funksjonene D
og E
, må man ha implementasjoner for B
, C
, D
og E
for å kunne kalle funksjon A
. Disse avhengighetene danner et avhengighetstre, og dersom man skal kalle en funksjon på toppen av treet er man nødt til å ha implementasjoner av alle de interne nodene og alle løvnodene i avhengighetstreet. For hver toppnivåfunksjon (slik som A
) man har i applikasjonen sin, vil man ha et avhengighetstre.
Den delen av applikasjonen som har ansvar for å tilfredsstille alle avhengighetene til alle toppnivåfunksjoner i applikasjonen kalles "composition root".
Du kan lese mer om "dependency injection" her: https://docs.microsoft.com/en-us/aspnet/core/fundamentals/dependency-injection?view=aspnetcore-6.0
Neste steg i å implementere API-et nå er å hente EPG for den validerte datoen. Siden det å hente sendinger for en gitt dato kan implementeres på flere måter (kalle web-tjeneste, spørre database, hente fra fil), benytter vi IoC-prinsippet, og sier at dette er en funksjon vi må få inn til epgHandler
. Vi definerer denne funksjonen som getEpgForDate: DateTimeOffset -> Epg
hvor Epg
er typen fra domenemodellen vår. Utvid epgHandler
i HttpHandlers.fs
med denne avhengigheten slik som vist under:
open NRK.Dotnetskolen.Domain
...
let epgHandler (getEpgForDate: DateTimeOffset -> Epg) (dateAsString: string) =
match (parseAsDateTime dateAsString) with
| Some date -> Results.Ok(date)
| None -> Results.BadRequest("Invalid date")
Nå kan vi kalle getEpgForDate
med den validerte datoen for å få alle sendingene for den gitte datoen slik som vist under:
let epgHandler (getEpgForDate: DateTimeOffset -> Epg) (dateAsString: string) =
match (parseAsDateTime dateAsString) with
| Some date ->
let epg = getEpgForDate date
Results.Ok(epg)
| None -> Results.BadRequest("Invalid date")
Det eneste som gjenstår i epgHandler
nå er å mappe fra domenemodellen til kontraktstypen vår, og returnere kontraktstypen som JSON.
Vi begynner med å mappe fra domenemodellen til kontraktstypen vår. Utvid Dto.fs
med en funksjon fromDomain
som tar inn et Epg
-objekt og returnerer et EpgDto
-objekt:
let fromDomain (domain: Domain.Epg) : EpgDto =
// Implementasjon her
☑️ Implementér fromDomain
-funksjonen.
💡Tips!
- For å konvertere en
DateTimeOffset
tilstring
på riktig format, kan man brukeToString("o")
på enDateTimeOffset
-verdi slik:let dateTimeOffsetAsString = myDateTimeOffset.ToString("o")
- Husk at
EpgDto
-typen har to felter: ett forNrk1
og ett forNrk2
, og at sendingene iEpg
-typen må filtreres før de settes i de to feltene. FunksjonenList.filter
kan brukes til å filtrere elementer i en liste.- Dersom man har en liste med sendinger for en gitt kanal, kan man bruke
List.map
til å mappe enSending
-verdi til enSendingDto
-verdi.
Nå som vi har implementert fromDomain
-funksjonen kan vi bruke den i epgHandler
. Legg til følgende open
-statement, og bruk fromDomain
i epgHandler
i HttpHandlers.fs
slik:
open NRK.Dotnetskolen.Dto
...
let epgHandler (getEpgForDate: DateTimeOffset -> Epg) (dateAsString: string) =
match (parseAsDateTime dateAsString) with
| Some date ->
let epg = getEpgForDate date
let dto = fromDomain epg
Results.Ok(dto)
| None -> Results.BadRequest("Invalid date")
Skrevet med |>
-operatoren i F# ser epgHandler
-funksjonen slik ut:
let epgHandler (getEpgForDate: DateTimeOffset -> Epg) (dateAsString: string) =
match (parseAsDateTime dateAsString) with
| Some date ->
let response =
date
|> getEpgForDate
|> fromDomain
Results.Ok(response)
| None -> Results.BadRequest("Invalid date")
I steget hente EPG definerte vi at funksjonen epgHandler
hadde en avhengighet til en funksjon getEpgForDate: DateTimeOffset -> Epg
. Husk fra kapitlet om "dependency injection" at vi må sørge for at slike avhengigheter er tilfredsstilt når vi kaller funksjonen.
epgHandler
-funksjonen blir kalt av MapGet
i createWebApplication
-funksjonen i Program.fs
i API-prosjektet. Dermed er det her vi må sende inn implementasjonen av getEpgForDate
-funksjonen.
La oss begynne med å definere funksjonen getEpgForDate
i en ny fil Services.fs
:
...
src
└── api
└── NRK.Dotnetskolen.Api.fsproj
└── Domain.fs
└── Dto.fs
└── HttpHandlers.fs
└── Program.fs
└── Services.fs
...
Husk å legg til Services.fs
i prosjektfilen til API-prosjektet:
<Project Sdk="Microsoft.NET.Sdk.Web">
<PropertyGroup>
<OutputType>Exe</OutputType>
<TargetFramework>net6.0</TargetFramework>
</PropertyGroup>
<ItemGroup>
<Compile Include="Domain.fs" />
<Compile Include="Dto.fs" />
<Compile Include="Services.fs" />
<Compile Include="HttpHandlers.fs" />
<Compile Include="Program.fs" />
</ItemGroup>
<Import Project="..\..\.paket\Paket.Restore.targets" />
</Project>
Legg til følgende kode i Services.fs
:
namespace NRK.Dotnetskolen.Api
module Services =
open System
open NRK.Dotnetskolen.Domain
let getEpgForDate (date : DateTimeOffset) : Epg =
[]
Foreløpig returnerer vi bare en tom liste slik at vi kan se hvordan vi kan benytte getEpgForDate
i epgHandler
.
Legg til følgende open
-statement i Program.fs
i API-prosjektet:
...
open NRK.Dotnetskolen.Api.Services
...
Send deretter inn getEpgForDate
fra NRK.Dotnetskolen.Api.Services
til epgHandler
i createWebApplication
-funksjonen i Program.fs
i API-prosjektet slik:
app.MapGet("/epg/{date}", Func<string, IResult>(fun date -> epgHandler getEpgForDate date)) |> ignore
Kjør web-API-et med følgende kommando, og gå til http://localhost:5000/epg/2021-04-23 for å se hva API-et returnerer.
dotnet run --project src/api/NRK.Dotnetskolen.Api.fsproj
La oss gå videre med å implementere getEpgForDate
i Services.fs
.
Oppgaven til getEpgForDate
er å filtrere sendinger på den oppgitte datoen, men hvor skal den få sendingene fra? På tilsvarende måte som vi gjorde i epgHandler
-funksjonen i HttpHandlers
, kan vi her si at vi ønsker å delegere ansvaret til å faktisk hente sendinger til noen andre. Dette kan vi gjøre ved å ta inn en funksjon getAlleSendinger: () -> Epg
i getEpgForDate
:
let getEpgForDate (getAlleSendinger : unit -> Epg) (date : DateTimeOffset) : Epg =
let alleSendinger = getAlleSendinger ()
☑️ Fullfør implementasjonen for getEpgForDate
og sørg for at Epg-verdien som returneres kun har sendinger som starter på den oppgitte datoen date
.
💡Tips!
List.filter
kan være til hjelp for å filtrere sendingene fragetAlleSendinger
Nå kan vi bestemme hvor vi skal hente sendinger fra. Skal vi hente dem fra en web-tjeneste, database, fil? getAlleSendinger
-funksjonen skjuler denne implementasjonsdetaljen fra resten av koden vår. For eksemplet vårt i dette kurset er det tilstrekkelig å definere sendinger i en egen fil DataAccess.fs
og implementere getAlleSendinger
der.
Opprett DataAccess.fs
i src/api
:
...
src
└── api
└── NRK.Dotnetskolen.Api.fsproj
└── DataAccess.fs
└── Domain.fs
└── Dto.fs
└── HttpHandlers.fs
└── Program.fs
└── Services.fs
...
Husk å legg til DataAccess.fs
i prosjektfilen til API-prosjektet:
<Project Sdk="Microsoft.NET.Sdk.Web">
<PropertyGroup>
<OutputType>Exe</OutputType>
<TargetFramework>net6.0</TargetFramework>
</PropertyGroup>
<ItemGroup>
<Compile Include="Domain.fs" />
<Compile Include="DataAccess.fs" />
<Compile Include="Dto.fs" />
<Compile Include="Services.fs" />
<Compile Include="HttpHandlers.fs" />
<Compile Include="Program.fs" />
</ItemGroup>
<Import Project="..\..\.paket\Paket.Restore.targets" />
</Project>
Vi later som at vi henter sendingene våre fra en database, og implementerer derfor egne typer som representerer hvordan sendingene og EPG-en er lagret i databasen:
namespace NRK.Dotnetskolen.Api
module DataAccess =
open System
type SendingEntity = {
Tittel: string
Kanal: string
Starttidspunkt: string
Sluttidspunkt: string
}
type EpgEntity = SendingEntity list
Deretter kan vi definere noen sendinger i en egen liste vi kaller database
:
let database =
[
{
Tittel = "Testprogram"
Kanal = "NRK1"
Starttidspunkt = "2021-04-12T13:00:00Z"
Sluttidspunkt = "2021-04-12T13:30:00Z"
}
{
Tittel = "Testprogram"
Kanal = "NRK2"
Starttidspunkt = "2021-04-12T14:00:00Z"
Sluttidspunkt = "2021-04-12T15:00:00Z"
}
{
Tittel = "Testprogram"
Kanal = "NRK3"
Starttidspunkt = "2021-04-12T14:00:00Z"
Sluttidspunkt = "2021-04-12T16:30:00Z"
}
]
Nå kan vi implementere getAlleSendinger
-funksjonen ved å legge til følgende open
-statement, og funksjonen getAlleSendinger
på slutten av DataAccess.fs
:
open NRK.Dotnetskolen.Domain
let getAlleSendinger () : Epg =
// Implementasjon her
Legg merke til at getAlleSendinger
-funksjonen skal returnere en verdi av typen Epg
fra Domain
-modulen.
☑️ Implementér getAlleSendinger
-funksjonen.
Tips: det kan være lurt å skrive en eller flere funksjoner som mapper en
SendingEntity
-verdi tilSending
-verdi ogEpgEntity
-verdi tilEpg
-verdi. Husk i den forbindelse å validére omEpg
-verdien er gyldig i ettertid. Vi kan ikke garantere datakvaliteten til databasen.
Ettersom vi innførte getAlleSendinger
som en avhengighet til getEpgForDate
, må vi endre createWebApplication
slik at getEpgForDate
får inn denne avhengigheten.
Legg til følgende open
-statement, og utvid kallet til app.MapGet("/epg/{date}"
i createWebApplication
i Program.fs
i web-API-prosjektet slik:
open NRK.Dotnetskolen.Api.DataAccess
let createWebApplication (builder: WebApplicationBuilder) =
let app = builder.Build()
app.MapGet("/ping", Func<string>(fun () -> "pong")) |> ignore
app.MapGet("/epg/{date}", Func<string, IResult>(fun date -> epgHandler (getEpgForDate getAlleSendinger) date)) |> ignore
app
Merk at over har vi kalt getEpgForDate
med getAlleSendinger
, og fått en ny funksjon i retur som tar inn en DateTimeOffset
og returnerer en Epg
-verdi. Det å sende inn et subsett av parameterene til en funksjon, og få en funksjon i retur som tar inn de resterende parameterene kalles "partial application". Du kan lese mer om "partial application" av funksjoner i F# her: https://docs.microsoft.com/en-us/dotnet/fsharp/language-reference/functions/#partial-application-of-arguments
Kjør API-et med følgende kommando, gå til http://localhost:5000/epg/2021-04-12, og se hva du får i retur.
dotnet run --project src/api/NRK.Dotnetskolen.Api.fsproj
Et problem med integrasjonstestene våre slik de er nå er at vi ikke har kontroll på avhengighetene til applikasjonen under kjøringen av integrasjonstestene. Mer konkret brukte vi den faktiske dataaksessen til web-API-et da vi kjørte testene. I et faktisk system ville ikke dataene være hardkodet i web-API-et, men heller lagret i den database eller liknende. For å slippe å være avhengig av en database ved kjøring av integrasjonstestene, kan vi endre hosten vi bruker i integrasjonstestene til å benytte et datalager vi spesifiserer i testene istedenfor å bruke det datalageret web-API-et er konfigurert til å bruke.
La oss implementere vår egen getAlleSendinger
-funksjon i integrasjonstestprosjektet, og få API-et vårt til å bruke den istedenfor.
Opprett filen Mock.fs
i mappen /test/integration
:
...
test
└── unit
└── ...
└── integration
└── Mock.fs
└── NRK.Dotnetskolen.IntegrationTests.fsproj
└── Program.fs
└── Tests.fs
└── Dotnetskolen.sln
Husk å legg til Mock.fs
i prosjektfilen til integrasjonstestprosjektet:
<?xml version="1.0" encoding="utf-8"?>
<Project Sdk="Microsoft.NET.Sdk.Web">
<PropertyGroup>
<TargetFramework>net6.0</TargetFramework>
<IsPackable>false</IsPackable>
<GenerateProgramFile>false</GenerateProgramFile>
</PropertyGroup>
<ItemGroup>
<Compile Include="Mock.fs" />
<Compile Include="Tests.fs" />
<Compile Include="Program.fs" />
<Content Include="../../docs/epg.schema.json">
<CopyToOutputDirectory>Always</CopyToOutputDirectory>
</Content>
</ItemGroup>
<ItemGroup>
<ProjectReference Include="..\..\src\api\NRK.Dotnetskolen.Api.fsproj" />
</ItemGroup>
<Import Project="..\..\.paket\Paket.Restore.targets" />
</Project>
Lim inn følgende kode i Mock.fs
hvor vi hardkoder noen sendinger som alltid har dagens dato:
namespace NRK.Dotnetskolen.IntegrationTests
module Mock =
open System
open NRK.Dotnetskolen.Domain
let getAlleSendinger () : Epg =
let now = DateTimeOffset.Now
[
// Sendinger tilbake i tid
{
Tittel = "Testprogram"
Kanal = "NRK1"
Starttidspunkt = now.AddDays(-10.)
Sluttidspunkt = now.AddDays(-10.).AddMinutes(30.)
}
{
Tittel = "Testprogram"
Kanal = "NRK2"
Starttidspunkt = now.AddDays(-10.)
Sluttidspunkt = now.AddDays(-10.).AddMinutes(30.)
}
// Sendinger i dag
{
Tittel = "Testprogram"
Kanal = "NRK1"
Starttidspunkt = now
Sluttidspunkt = now.AddMinutes(30.)
}
{
Tittel = "Testprogram"
Kanal = "NRK2"
Starttidspunkt = now
Sluttidspunkt = now.AddMinutes(30.)
}
// Sendinger frem i tid
{
Tittel = "Testprogram"
Kanal = "NRK1"
Starttidspunkt = now.AddDays(10.)
Sluttidspunkt = now.AddDays(10.).AddMinutes(30.)
}
{
Tittel = "Testprogram"
Kanal = "NRK2"
Starttidspunkt = now.AddDays(10.)
Sluttidspunkt = now.AddDays(10.).AddMinutes(30.)
}
]
Nå har vi en egen implementasjon av getAlleSendinger
som vi ønsker å bruke kun når integrasjonstestene kjører. Hvordan får vi til det? La oss se nøyere på hvordan Program.fs
i API-prosjektet ser ut:
namespace NRK.Dotnetskolen.Api
module Program =
open System
open Microsoft.AspNetCore.Http
open Microsoft.AspNetCore.Builder
open NRK.Dotnetskolen.Api.Services
open NRK.Dotnetskolen.Api.DataAccess
open NRK.Dotnetskolen.Api.HttpHandlers
let createWebApplicationBuilder () =
WebApplication.CreateBuilder()
let createWebApplication (builder: WebApplicationBuilder) =
let app = builder.Build()
app.MapGet("/ping", Func<string>(fun () -> "pong")) |> ignore
app.MapGet("/epg/{date}", Func<string, IResult>(fun date -> epgHandler (getEpgForDate getAlleSendinger) date)) |> ignore
app
let builder = createWebApplicationBuilder()
let app = createWebApplication builder
app.Run()
Her ser vi at epgHandler
tar inn getEpgForDate
"partially applied" med getAlleSendinger
som første parameter. getEpgForDate
og getAlleSendinger
her er tatt fra hhv. Services
- og DataAccess
-modulene i API-prosjektet, men vi ønsker å sende med egne implementasjoner av disse i integrasjonstestene slik at vi har kontroll på avhengighetene til API-et under kjøring av integrasjonstestene. Husk at runWithTestClient
-funksjonen i Tests.fs
i integrasjonstestprosjektet kaller createWebApplication
-funksjonen fra Program.fs
i API-prosjektet. Dersom vi hadde utvidet createWebApplication
-funksjonen til å ta inn getEpgForDate
som et eget parameter kunne vi sendt én implementasjon av funksjonen fra API-et, og en annen implementasjon fra integrasjonstestene. La oss gjøre det.
Legg til følgende open
-statement, og utvid createWebApplication
-funksjonen i Program.fs
i API-prosjektet med et parameter til getEpgForDate
, og send dette inn til epgHandler
slik:
open NRK.Dotnetskolen.Domain
let createWebApplication (builder: WebApplicationBuilder) (getEpgForDate: DateTimeOffset -> Epg) =
let app = builder.Build()
app.MapGet("/ping", Func<string>(fun () -> "pong")) |> ignore
app.MapGet("/epg/{date}", Func<string, IResult>(fun date -> epgHandler getEpgForDate date)) |> ignore
app
Send deretter getEpgForDate
fra Services
-modulen "partially applied" med getAlleSendinger
fra DataAccess
-modulen inn som andre parameter til createWebApplication
, slik:
let app = createWebApplication builder (getEpgForDate getAlleSendinger)
Program.fs
i API-prosjektet skal nå se slik ut:
namespace NRK.Dotnetskolen.Api
module Program =
open System
open Microsoft.AspNetCore.Http
open Microsoft.AspNetCore.Builder
open NRK.Dotnetskolen.Domain
open NRK.Dotnetskolen.Api.Services
open NRK.Dotnetskolen.Api.DataAccess
open NRK.Dotnetskolen.Api.HttpHandlers
let createWebApplicationBuilder () =
WebApplication.CreateBuilder()
let createWebApplication (builder: WebApplicationBuilder) (getEpgForDate: DateTimeOffset -> Epg) =
let app = builder.Build()
app.MapGet("/ping", Func<string>(fun () -> "pong")) |> ignore
app.MapGet("/epg/{date}", Func<string, IResult>(fun date -> epgHandler getEpgForDate date)) |> ignore
app
let builder = createWebApplicationBuilder()
let app = createWebApplication builder (getEpgForDate getAlleSendinger)
app.Run()
Nå som kan styre implementasjonen av getEpgForDate
fra utsiden av createWebApplication
-funksjonen kan vi lage en egen getEpgForDate
i integrasjonstestprosjektet som bruker mock-implementasjonen av getAlleSendinger
. Start med å åpne Services
-modulen fra API-prosjektet, og Mock
-modulen fra integrasjonstestprosjektet i Tests.fs
i integrasjonstestprosjektet, slik:
open NRK.Dotnetskolen.Api.Services
open NRK.Dotnetskolen.IntegrationTests.Mock
Endre deretter kallet til createWebApplication
fra runWithTestClient
i Tests.fs
i integrasjonstestprosjektet til å sende med en "partially applied" versjon av getEpgForDate
fra Services
med getAlleSendinger
fra Mock
-modulen slik:
use app = createWebApplication builder (getEpgForDate getAlleSendinger)
Hele runWithTestClient
-funksjonen skal nå se slik ut:
let runWithTestClient (test: HttpClient -> Task<unit>) =
task {
let builder = createWebApplicationBuilder()
builder.WebHost.UseTestServer() |> ignore
use app = createWebApplication builder (getEpgForDate getAlleSendinger)
do! app.StartAsync()
let testClient = app.GetTestClient()
do! test testClient
}
Dersom du kjører integrasjonstestene igjen, skal de fortsatt passere:
dotnet test test/integration/NRK.Dotnetskolen.IntegrationTests.fsproj
Passed! - Failed: 0, Passed: 4, Skipped: 0, Total: 4, Duration: 124 ms - NRK.Dotnetskolen.IntegrationTests.dll (net6.0)
Gratulerer! 🎉
Du har nå implementert et web-API i F#, med enhets- og integrasjonstester, API-dokumentasjon i OpenAPI, og gjort alt ved hjelp av .NET CLI.
Implementasjonen av domenemodellen slik vi gjorde det i steg 5 og steg 6 har en svakhet: det er ingen garanti for at verdier vi oppretter for Sending
og Epg
er gyldige. Det er kun epgEntityToDomain
-funksjonen i DataAccess.fs
som kaller isSendingValid
når sendinger hentes. Det er ingen garanti for at alle opprettelser av Sending
- og Epg
-verdier kommer gjennom epgEntityToDomain
. I dette steget skal vi se hvordan vi kan endre domenemodellen vår slik at man ikke kan opprette Sending
- og Epg
-verdier uten at de er gyldige.
I steg 5 modellerte vi tittel og kanal som string
, og start- og sluttidspunktene som DateTimeOffset
. Utover at feltene har disse typene er det ingenting i Sending
-typen vår som sier hvilke regler som gjelder for dem. Det kan vi imidlertid gjøre noe med.
La oss ta tittel som eksempel. Dersom vi oppretter en egen type for tittel Tittel
, og setter konstruktøren som private
er det ingen som kan opprette en Tittel
-verdi direkte. For å gjøre det mulig å opprette Tittel
-verdier kan vi lage en modul med samme navn som typen vår, Tittel
, med en create
-funksjon i. create
-funksjonen tar inn tittel som en string
, validerer om den er gyldig, og returnerer en Tittel option
avhengig av om tittelen er gyldig eller ikke. Dersom tittelen er gyldig returnerer create
-funksjonen Some (Tittel tittel)
, hvor tittel
er string
-verdien man sender inn til create
, Tittel
er konstruktøren til Tittel
-typen, og Some
er den ene konstruktøren til option
-verdier. Dersom tittelen imidlertid er ugyldig returnerer create
-funksjonen None
. På tilsvarende måte som man er avhengig av create
-funksjonen for å opprette Tittel
-verdier, er vi også avhengig av å ha en funksjon for å hente ut den indre verdien til en tittel, selve string
-verdien. Til det oppretter vi en value
-funksjon. La oss se hvordan det ser ut i kode.
Legg til koden under i Domain.fs
, mellom open
-statementene og type Sending
.
type Tittel = private Tittel of string
let isTittelValid (tittel: string) : bool =
let tittelRegex = Regex(@"^[\p{L}0-9\.,-:!]{5,100}$")
tittelRegex.IsMatch(tittel)
module Tittel =
let create (tittel: String) : Tittel option =
if isTittelValid tittel then
Tittel tittel
|> Some
else
None
let value (Tittel tittel) = tittel
Her ser vi at vi har definert tittel som en egen type Tittel
, som er en "single case union"-type med privat konstruktør. Deretter har vi isTittelValid
-funksjonen slik vi definerte den i steg 6. Til slutt har vi Tittel
-modulen med create
- og value
-funksjonene.
Merk at
isTittelValid
-funksjonen over er den samme som tidligere, bare at den har byttet plass. Du kan fjerneisTittelValid
-funksjonen som tidligere var definert iDomain.fs
.
Nå som vi har laget en egen type for tittel i en sending, kan vi ta den i bruk i Sending
-typen vår i Domain.fs
:
type Sending = {
Tittel: Tittel
Kanal: string
Starttidspunkt: DateTimeOffset
Sluttidspunkt: DateTimeOffset
}
Her ser vi at istedenfor å bruke string
for tittel, bruker vi den nye typen vi har opprettet, Tittel
.
Dersom vi forsøker å kompilere API-prosjektet vårt nå, vil det feile fordi vi har endret typen til feltet Tittel
i Sending
-typen vår. La oss fikse kompileringsfeilene.
dotnet build ./src/api/NRK.Dotnetskolen.Api.fsproj
Build FAILED.
...
3 Error(s)
...
Det første som feiler er isSendingValid
-funksjonen i Domain.fs
:
let isSendingValid (sending: Sending) : bool =
(isTittelValid sending.Tittel) &&
(isKanalValid sending.Kanal) &&
(areStartAndSluttidspunktValid sending.Starttidspunkt sending.Sluttidspunkt)
Her kaller vi isTittelvalid
med sending.Tittel
. Ettersom isTittelValid
tar inn et argument av typen string
, og sending.Tittel
nå har typen Tittel
feiler typesjekken. På grunn av at vi har gjort konstruktøren til Tittel
privat, er den eneste måten å opprette en Tittel
-verdi på ved å bruke create
-funksjonen i Tittel
-modulen. Siden create
-funksjonen kun returnerer en Tittel
-verdi dersom den oppgitte tittelen er gyldig, vet vi at Tittel
-feltet i en Sending
-verdi må være gyldig. Dermed kan vi fjerne sjekken på om tittel er gyldig i isSendingValid
, slik:
let isSendingValid (sending: Sending) : bool =
(isKanalValid sending.Kanal) &&
(areStartAndSluttidspunktValid sending.Starttidspunkt sending.Sluttidspunkt)
Det neste som feiler er opprettelsen av en Sending
-verdi i DataAccess.fs
. Under er implementasjonen av funksjonen som mapper SendingEntity
til Sending
hentet fra løsningsforslaget for kapittel 10.
let sendingEntityToDomain (sendingEntity: SendingEntity) : Sending =
{
Sending.Tittel = sendingEntity.Tittel
Kanal = sendingEntity.Kanal
Starttidspunkt = DateTimeOffset.Parse(sendingEntity.Starttidspunkt)
Sluttidspunkt = DateTimeOffset.Parse(sendingEntity.Sluttidspunkt)
}
Her forsøker vi å sette Sending.Tittel
direkte til Tittel
-feltet fra SendingEntity
-verdien. Siden Tittel
-feltet i SendingEntity
-typen er string
, og Sending.Tittel
er av typen Tittel
feiler typesjekken. For å fikse dette må vi kalle Tittel.create
-funksjonen med SendingEntity
sin Tittel
som input, slik:
let sendingEntityToDomain (sendingEntity: SendingEntity) : Sending =
{
Sending.Tittel = (Tittel.create sendingEntity.Tittel).Value
Kanal = sendingEntity.Kanal
Starttidspunkt = DateTimeOffset.Parse(sendingEntity.Starttidspunkt)
Sluttidspunkt = DateTimeOffset.Parse(sendingEntity.Sluttidspunkt)
}
Ettersom Tittel.create
returnerer en Tittel option
, må vi kalle .Value
-funksjonen på returverdien av Tittel.create
for å få ut Tittel
-verdien. Merk at dersom den oppgitte tittelen er ugyldig, vil kallet til .Value
kaste en System.NullReferenceException
.
Det neste som feiler er uthentingen av Tittel
-verdien fra Sending
i fromDomain
-funksjonen i Dto.fs
. Under er funksjonen fromDomain
vist slik den er implementert i løsningsforslaget til steg 10.
let fromDomain (domain : Domain.Epg) : EpgDto =
let mapSendingerForKanal (kanal : string) =
domain
|> List.filter (fun s -> s.Kanal = kanal)
|> List.map (fun s -> {
Tittel = s.Tittel
Starttidspunkt = s.Starttidspunkt.ToString("o")
Sluttidspunkt = s.Sluttidspunkt.ToString("o")
})
{
Nrk1 = mapSendingerForKanal "NRK1"
Nrk2 = mapSendingerForKanal "NRK2"
}
Her forsøker vi å sette Tittel
-verdien til SendingDto
-typen til en Tittel
-verdi, men siden SendingDto.Tittel
er en string
feiler typesjekken. For å hente ut den indre string
-verdien til en Tittel
-verdi kan vi kalle Tittel.value
med Tittel
-verdien som input, slik:
let fromDomain (domain : Domain.Epg) : EpgDto =
let mapSendingerForKanal (kanal : string) =
domain
|> List.filter (fun s -> s.Kanal = kanal)
|> List.map (fun s -> {
Tittel = Domain.Tittel.value s.Tittel
Starttidspunkt = s.Starttidspunkt.ToString("o")
Sluttidspunkt = s.Sluttidspunkt.ToString("o")
})
{
Nrk1 = mapSendingerForKanal "NRK1"
Nrk2 = mapSendingerForKanal "NRK2"
}
Dersom du forsøker å bygge API-prosjektet igjen nå, skal det lykkes:
dotnet build ./src/api/NRK.Dotnetskolen.Api.fsproj
Build succeeded.
...
Det gjenstår imidlertid å fikse kompileringsfeil i testprosjektene våre. Dersom du forsøker å bygge løsningen, vil du se at kompilering av testprosjektene feiler:
dotnet build
Build FAILED.
...
8 Error(s)
...
Det første vi må rette opp er opprettelsen av Sending
-verdier i Tests.fs
i enhetstestprosjektet. Her må vi gjøre det samme som vi gjorde i DataAccess.fs
, og kalle Tittel.create
-funksjonen for å opprette Tittel
-verdier i Sending
-typen.
☑️ Fiks kompileringsfeilene i Tests.fs
i enhetstestprosjektet på samme måte som vi gjorde for DataAccess.fs
.
Opprettelsen av Sending
-verdier i Mock.fs
i integrasjonstestprosjektet feiler av samme grunn som over.
☑️ Fiks kompileringsfeilene på samme måte.
Nå som vi har sett hvordan vi kan implementere en egen type for Tittel
-feltet i Sending
-typen, kan vi gå videre til å følge samme mønster for kanal også.
☑️ Følg samme mønster for kanal som vi gjorde for tittel. Husk følgende punkter:
- Opprett egen type for
Kanal
iDomain.fs
med privat konstruktør - Opprett egen modul for
Kanal
iDomain.fs
medcreate
- ogvalue
-funksjoner - Flytt
isKanalValid
-funksjonen mellom typen og modulen forKanal
- Benytt den nye
Kanal
-typen iSending
-typen iDomain.fs
- Fiks kompileringsfeil i API-prosjektet:
- Fjern sjekk for kanal i
isSendingValid
-funksjonen iDomain.fs
- Opprettelse av
Kanal
-verdier isendingEntityToDomain
-funksjonen iDataAccess.fs
- Uthenting av
Kanal
-verdier ifromDomain
-funksjonen iDto.fs
- Fjern sjekk for kanal i
- Fiks kompileringsfeil i testprosjektene:
- Opprettelse av
Sending
-verdier iTests.fs
i enhetstestprosjektet - Opprettelse av
Sending
-verdier iMock.fs
i integrasjonstestprosjektet
- Opprettelse av
Vi kan følge de samme prinsippene som for tittel og kanal for start- og sluttidspunkt også, men ettersom man ikke kan si om start- og sluttidspunktene er gyldige med mindre man har begge to, må vi lage en type som har begge feltene:
type Sendetidspunkt = private {
Starttidspunkt: DateTimeOffset
Sluttidspunkt: DateTimeOffset
}
let areStartAndSluttidspunktValid (starttidspunkt: DateTimeOffset) (sluttidspunkt: DateTimeOffset) =
starttidspunkt < sluttidspunkt
module Sendetidspunkt =
let create (starttidspunkt: DateTimeOffset) (sluttidspunkt: DateTimeOffset) : Sendetidspunkt option =
if areStartAndSluttidspunktValid starttidspunkt sluttidspunkt then
{
Starttidspunkt = starttidspunkt
Sluttidspunkt = sluttidspunkt
}
|> Some
else
None
let starttidspunkt (sendeTidspunkt: Sendetidspunkt) = sendeTidspunkt.Starttidspunkt
let sluttidspunkt (sendeTidspunkt: Sendetidspunkt) = sendeTidspunkt.Sluttidspunkt
Her har vi definert en samletype Sendetidspunkt
, som inneholder både start- og sluttidspunkt. Legg merke til at create
-funksjonen tar inn begge disse, og bruker areStartAndSluttidspunktValid
-funksjonen til å undersøke om de er gyldige opp mot hverandre, før en Sendetidspunkt
-verdi opprettes. Merk at vi ikke har laget en value
-funksjon her, men istedenfor laget en starttidspunkt
- og en sluttidspunkt
-funksjon, som begge tar inn en Sendetidspunkt
-verdi, og returnerer den respektive verdien fra Sendetidspunkt
-verdien.
Nå som vi har laget en egen type for start- og sluttidspunkt i en sending, kan vi ta dem i bruk i Sending
-typen vår:
type Sending = {
Tittel: Tittel
Kanal: Kanal
Sendetidspunkt: Sendetidspunkt
}
Her ser vi at vi bruker Sendetidspunkt
istedenfor DateTimeOffset
for start- og sluttidspunkt. Legg merke til at Sending
ikke har privat konstruktør. Det er ikke nødvendig ettersom alle feltene i Sending
-typen må opprettes gjennom deres create
-funksjoner. Dermed vil en Sending
-verdi alltid være gyldig. Som en beleilighet for de som skal ta i bruk Sending
-typen kan vi likevel lage en create
-funksjon i en egen Sending
-modul, slik at man enklere kan lage en Sending
-verdi uten å kalle create
-funksjonene i modulen som korresponderer til typen til hvert felt.
module Sending =
let create (tittel: string) (kanal: string) (starttidspunkt: DateTimeOffset) (sluttidspunkt: DateTimeOffset) : Sending option =
let tittel = Tittel.create tittel
let kanal = Kanal.create kanal
let sendeTidspunkt = Sendetidspunkt.create starttidspunkt sluttidspunkt
if tittel.IsNone || kanal.IsNone || sendeTidspunkt.IsNone then
None
else
Some {
Tittel = tittel.Value
Kanal = kanal.Value
Sendetidspunkt = sendeTidspunkt.Value
}
Over ser vi Sending
-modulen med create
-funksjonen som tar inn verdier for alle feltene i en Sending
-verdi. create
-funksjonen til Sending
kaller create
-funksjonen til hver av typene som den består av, og returnerer en Sending
-verdi kun dersom alle verdiene ble vellykket opprettet.
For å oppsummere ser Domain.fs
nå slik ut:
namespace NRK.Dotnetskolen
module Domain =
open System
open System.Text.RegularExpressions
type Tittel = private Tittel of string
let isTittelValid (tittel: string) : bool =
let tittelRegex = Regex(@"^[\p{L}0-9\.,-:!]{5,100}$")
tittelRegex.IsMatch(tittel)
module Tittel =
let create (tittel: String) : Tittel option =
if isTittelValid tittel then
Tittel tittel
|> Some
else
None
let value (Tittel tittel) = tittel
type Kanal = private Kanal of string
let isKanalValid (kanal: string) : bool =
List.contains kanal ["NRK1"; "NRK2"]
module Kanal =
let create (kanal: string) : Kanal option =
if isKanalValid kanal then
Kanal kanal
|> Some
else
None
let value (Kanal kanal) = kanal
type Sendetidspunkt = private {
Starttidspunkt: DateTimeOffset
Sluttidspunkt: DateTimeOffset
}
let areStartAndSluttidspunktValid (starttidspunkt: DateTimeOffset) (sluttidspunkt: DateTimeOffset) =
starttidspunkt < sluttidspunkt
module Sendetidspunkt =
let create (starttidspunkt: DateTimeOffset) (sluttidspunkt: DateTimeOffset) : Sendetidspunkt option =
if areStartAndSluttidspunktValid starttidspunkt sluttidspunkt then
{
Starttidspunkt = starttidspunkt
Sluttidspunkt = sluttidspunkt
}
|> Some
else
None
let starttidspunkt (sendeTidspunkt: Sendetidspunkt) = sendeTidspunkt.Starttidspunkt
let sluttidspunkt (sendeTidspunkt: Sendetidspunkt) = sendeTidspunkt.Sluttidspunkt
type Sending = {
Tittel: Tittel
Kanal: Kanal
Sendetidspunkt: Sendetidspunkt
}
type Epg = Sending list
module Sending =
let create (tittel: string) (kanal: string) (starttidspunkt: DateTimeOffset) (sluttidspunkt: DateTimeOffset) : Sending option =
let tittel = Tittel.create tittel
let kanal = Kanal.create kanal
let sendeTidspunkt = Sendetidspunkt.create starttidspunkt sluttidspunkt
if tittel.IsNone || kanal.IsNone || sendeTidspunkt.IsNone then
None
else
Some {
Tittel = tittel.Value
Kanal = kanal.Value
Sendetidspunkt = sendeTidspunkt.Value
}
Legg merke til at isSendingValid
-funksjonen er fjernet, ettersom Sending.create
-funksjonen har overtatt dens ansvar.
Dersom du forsøker å bygge løsningen nå, vil du se at det feiler:
dotnet build
Build FAILED.
...
La oss starte med sendingEntityToDomain
-funksjonen i DataAccess.fs
:
let sendingEntityToDomain (sendingEntity: SendingEntity) : Sending =
{
Sending.Tittel = (Tittel.create s.Tittel).Value
Kanal = sendingEntity.Kanal
Starttidspunkt = DateTimeOffset.Parse(sendingEntity.Starttidspunkt)
Sluttidspunkt = DateTimeOffset.Parse(sendingEntity.Sluttidspunkt)
}
Her forsøker vi å sette Starttidspunkt
og Sluttidspunkt
direkte, men disse er nå flyttet inn i feltet Sendetidspunkt
. Vi kunne ha brukt Sendetidspunkt.create
-funksjonen til å løse det på tilsvarende vis som for Tittel
og Kanal
, men ettersom vi har innført Sending.create
-funksjonen som kaller create
-funksjonen for alle de nye typene for oss, kan vi heller bruke den, slik:
let sendingEntityToDomain (sendingEntity: SendingEntity) : Sending option =
Sending.create sendingEntity.Tittel sendingEntity.Kanal (DateTimeOffset.Parse(sendingEntity.Starttidspunkt)) (DateTimeOffset.Parse(sendingEntity.Sluttidspunkt))
let epgEntityToDomain (epgEntity: EpgEntity) : Epg =
epgEntity
|> List.map sendingEntityToDomain
|> List.filter (fun s -> s.IsSome)
|> List.map (fun s -> s.Value)
Over kaller vi sendingEntityToDomain
for hver sending i EpgEntity
som vi får inn til epgEntityToDomain
. sendingEntityToDomain
kaller igjen på Sending.create
. Husk at Sending.create
-funksjonen returnerer en Sending option
, så sendingEntityToDomain
vil returnere None
for ugyldige SendingEntity
-verdier. For å filtrere bort disse kan vi kalle List.filter (fun e -> e.IsSome)
etterfulgt av List.map (fun s -> s.Value)
for å hente ut selve Sending
-verdien fra Sending option
. Alternativt kan man kalle List.choose id
slik:
let epgEntityToDomain (epgEntity: EpgEntity) : Epg =
epgEntity
|> List.map sendingEntityToDomain
|> List.choose id
List.choose
tar inn en funksjonf
, og returnerer en liste med de interne verdiene til innslagene i listen hvorf
returnererSome
.ìd
er en innebygd funksjon i F# som returnerer det den får inn. Ved å kombinereList.choose
medid
-funksjonen oppnår vi det samme som vi gjorde medList.filter (fun s -> s.IsSome)
ogList.map (fun s -> s.Value)
etter hverandre.
Legg også merke til at i koden over fjernet vi List.filter (fun d -> isSendingValid d)
, og på den måten flyttet ansvaret for å validere en Sending
-verdi fra sendingEntityToDomain
-funksjonen i DataAccess.fs
til Sending.create
-funksjonen i Domain.fs
.
fromDomain
-funksjonen i Dto.fs
feiler også ettersom den ikke får hentet ut verdiene til Starttidspunkt
og Sluttidspunkt
i en Sending
-verdi slik den forventer.
let fromDomain (domain : Domain.Epg) : EpgDto =
let mapSendingerForKanal (kanal : string) =
domain
|> List.filter (fun s -> (Domain.Kanal.value s.Kanal) = kanal)
|> List.map (fun s -> {
Tittel = Domain.Tittel.value s.Tittel
Starttidspunkt = s.Starttidspunkt.ToString("o")
Sluttidspunkt = s.Sluttidspunkt.ToString("o")
})
{
Nrk1 = mapSendingerForKanal "NRK1"
Nrk2 = mapSendingerForKanal "NRK2"
}
Start- og sluttidspunkt er nå lagret i en samletype Sendetidspunkt
, så uthentingen av start- og sluttidspunkt vil ikke fungere. Vi kan imidlertid bruke funksjonene vi definerte tidligere i dette steget til å hente ut de indre verdiene til Sendetidspunkt
slik:
open Domain
...
let fromDomain (domain: Domain.Epg): EpgDto =
let mapSendingerForKanal (kanal: string) =
domain
|> List.filter (fun s -> Kanal.value s.Kanal = kanal)
|> List.map (fun s ->
{ Tittel = Tittel.value s.Tittel
Starttidspunkt = (Sendetidspunkt.starttidspunkt s.Sendetidspunkt).ToString("o")
Sluttidspunkt = (Sendetidspunkt.sluttidspunkt s.Sendetidspunkt).ToString("o") })
{ Nrk1 = mapSendingerForKanal "NRK1"
Nrk2 = mapSendingerForKanal "NRK2" }
Vi henter start- og sluttidspunkt ved å kalle hhv. Sendetidspunkt.starttidspunkt
og Sendetidspunkt.sluttidspunkt
med s.Sendetidspunkt
som input.
I getEpgForDate
-funksjonen i Services.fs
filtrerer vi sendinger basert på dato:
let getEpgForDate (getAlleSendinger : unit -> Epg) (date : DateTimeOffset) : Epg =
getAlleSendinger ()
|> List.filter (fun s -> s.Starttidspunkt.Date.Date = date.Date)
Ettersom vi har innført en ny måte å hente ut starttidspunkt fra en sending på, må vi oppdatere getEpgForDate
til å reflektere dette:
let getEpgForDate (getAlleSendinger : unit -> Epg) (date : DateTimeOffset) : Epg =
getAlleSendinger ()
|> List.filter (fun s -> (Sendetidspunkt.starttidspunkt s.Sendetidspunkt).Date.Date = date.Date)
Istedenfor å hente starttidspunktet direkte, kaller vi Sendetidspunkt.starttidspunkt
med s.Sendetidspunkt
som input.
I enhetstestprosjektet har vi tester for funksjonen isSendingValid
som vi hadde i Domain.fs
. Ettersom Sending.create
-funksjonen har tatt over ansvaret til isSendingValid
må vi skrive om testene til å bruke Sending.create
-funksjonen istedenfor:
[<Fact>]
let ``Sending.create valid sending returns Some`` () =
let now = DateTimeOffset.Now
let sending = Sending.create "Dagsrevyen" "NRK1" now (now.AddMinutes 30.)
match sending with
| Some t ->
Assert.Equal("Dagsrevyen", Tittel.value t.Tittel)
Assert.Equal("NRK1", Kanal.value t.Kanal)
Assert.Equal(now, Sendetidspunkt.starttidspunkt t.Sendetidspunkt)
Assert.Equal(now.AddMinutes 30., Sendetidspunkt.sluttidspunkt t.Sendetidspunkt)
| None -> Assert.True false
[<Fact>]
let ``Sending.create invalid sending returns None`` () =
let now = DateTimeOffset.Now
let sending = Sending.create "@$%&/" "nrk3" now (now.AddMinutes 30.)
Assert.True sending.IsNone
I Mock
-modulen i integrasjonstestprosjektet opprettet vi Sending
-verdier for å ha kontroll på dataaksessen under integrasjonstestene. Nå som vi har en egen funksjon for å opprette Sending
-verdier, Sending.create
, kan vi bruke den istedenfor å opprette Sending
-verdier direkte, slik:
let getAlleSendinger () : Epg =
let now = DateTimeOffset.Now
let past = now.AddDays(-10.)
let future = now.AddDays(10.)
[
// Sendinger tilbake i tid
(Sending.create "Testprogram" "NRK1" past (past.AddMinutes(30.))).Value
(Sending.create "Testprogram" "NRK2" past (past.AddMinutes(30.))).Value
// Sendinger i dag
(Sending.create "Testprogram" "NRK1" now (now.AddMinutes(30.))).Value
(Sending.create "Testprogram" "NRK2" now (now.AddMinutes(30.))).Value
// Sendinger frem i tid
(Sending.create "Testprogram" "NRK1" future (future.AddMinutes(30.))).Value
(Sending.create "Testprogram" "NRK2" future (future.AddMinutes(30.))).Value
]
I steg 7 innførte vi OpenAPI-kontrakt for API-et vårt, og la den i mappen /docs
. Foreløpig er dokumentasjonen kun tilgjengelig for de som har tilgang til repoet til koden. For at de som skal integrere med API-et skal kunne se kontrakten, er det fint om den er publisert et sted. I dette steget skal vi se hvordan vi kan tilgjengeliggjøre OpenAPI-kontrakten som en egen nettside i API-et ved hjelp av ReDoc. Med ReDoc kan vi kopiere en HTML-side fra dokumentasjonen deres og lime inn en referanse til OpenAPI-dokumentasjonen vår, så får vi en fin grafisk fremstilling av API-et vårt, som vist under:
Kort oppsummert er dette stegene vi skal gjøre for å lage en egen ReDoc-side i API-et vårt:
- Flytte
docs/epg.schema.json
ogdocs/openapi.json
tilsrc/api/wwwroot/documentation
- Opprette HTML-fil
openapi.html
isrc/api/wwwroot
med innhold fra dokumentasjonen til ReDoc, og endre referansen til OpenAPI-dokumentet iopenapi.html
- Konfigurere web-API-et til å serve statiske filer
I steg 7 la vi dokumentasjonen til API-et vårt i docs
-mappen. Ettersom vi nå skal eksponere den på internett gjennom API-et vårt, må vi legge den et sted som er tilgjengelig for webserveren. Opprett derfor en ny mappe wwwroot
med en ny mappe documentation
i src/api
slik:
...
└── docs
└── epg.schema.json
└── openapi.json
└── src
└── api
└── wwwroot
└── documentation
...
Flytt deretter filene epg.schema.json
og openapi.json
fra docs
til src/api/wwwroot/documentation
:
...
└── docs
└── src
└── api
└── wwwroot
└── documentation
└── epg.schema.json
└── openapi.json
...
Til slutt kan du slette mappen docs
:
...
└── src
└── api
└── wwwroot
└── documentation
└── epg.schema.json
└── openapi.json
...
I steg 9 la vi til en referanse til epg.schema.json
i prosjektfilen til integrasjonstestprosjektet. Siden vi har flyttet denne filen, må vi oppdatere referansen. Åpne filen test/integration/NRK.Dotnetskolen.IntegrationTests.fsproj
, og endre referansen til JSON Schemaet:
...
<Content Include="../../src/api/wwwroot/documentation/epg.schema.json">
<CopyToOutputDirectory>Always</CopyToOutputDirectory>
</Content>
...
Verifiser at integrasjonstestene kjører med følgende kommando:
dotnet test ./test/integration/NRK.Dotnetskolen.IntegrationTests.fsproj
Opprett filen openapi.html
i mappen src/api/wwwroot
, slik:
...
└── src
└── api
└── wwwroot
└── documentation
└── epg.schema.json
└── openapi.json
└── openapi.html
...
Åpne openapi.html
, og lim inn innholdet vist i dokumentasjonen til ReDoc, slik:
<!DOCTYPE html>
<html>
<head>
<title>Redoc</title>
<!-- needed for adaptive design -->
<meta charset="utf-8" />
<meta name="viewport" content="width=device-width, initial-scale=1">
<link href="https://fonts.googleapis.com/css?family=Montserrat:300,400,700|Roboto:300,400,700" rel="stylesheet">
<!--
Redoc doesn't change outer page styles
-->
<style>
body {
margin: 0;
padding: 0;
}
</style>
</head>
<body>
<redoc spec-url='http://petstore.swagger.io/v2/swagger.json'></redoc>
<script src="https://cdn.redoc.ly/redoc/latest/bundles/redoc.standalone.js"> </script>
</body>
</html>
Legg merke til at linjen som starter med <redoc spec-url=
(nesten helt nederst i filen) refererer til en eksempel-dokumentasjon http://petstore.swagger.io/v2/swagger.json
. Denne skal vi nå endre til vår egen dokumentasjon. Endre spec-url
i denne linja til /documentation/openapi.json
, slik:
...
<redoc spec-url='/documentation/openapi.json'></redoc>
...
I utgangspunktet kan ikke web-applikasjonen slik vi har konfigurert den nå servere statiske filer (slik som openapi.html
som nettopp opprettet). For å kunne serve statiske filer må vi legge til en egen "middleware" i "middleware pipelinen" til web-API-et vårt som gjør akkurat dette. For å legge til denne "middlewaren" kaller vi app.UseStaticFiles()
på WebApplication
-objektet som vi oppretter i createWebApplication
-funksjonen i Program.fs
i API-prosjektet vårt, slik:
let createWebApplication (builder: WebApplicationBuilder) (getEpgForDate: DateTimeOffset -> Epg) =
let app = builder.Build()
app.UseStaticFiles() |> ignore
app.MapGet("/ping", Func<string>(fun () -> "pong")) |> ignore
app.MapGet("/epg/{date}", Func<string, IResult>(fun date -> epgHandler getEpgForDate date)) |> ignore
app
Her kaller vi UseStaticFiles
-funksjonen, som sørger for at statiske filer blir servet av webserveren. Som default konfigureres serveren til å se etter statiske filer i wwwroot
-mappen. Legg merke til at vi kaller UseStaticFiles
før MapGet
-funksjonene. Siden middlewares i .NET prosesserer innkommende forespørsler i den rekkefølgen de blir lagt til i "middleware pipelinen", legger vi til serving av statiske filer før håndtering av andre HTTP-forespørsler, slik at dersom det finnes en statisk fil identifisert av path-en i HTTP-forespørselen returnerer vi den istedenfor å gå videre med å evaluere endepunktene vi har satt opp.
Dersom du nå starter web-API-et med dotnet run --project src/api/NRK.Dotnetskolen.Api.fsproj
, og åpner http://localhost:5000/openapi.html skal du se noe liknende som skjermbildet under: