A simple software packages dependencies checker.
Table of contents:
The implemented library allows to analyze software package dependencies.
The main purpose is to check if given dependencies description is valid.
A system with package dependencies is described like this:
2
A,1
B,1
3
A,1,B,1
A,2,B,2
C,1,B,1
- The first line is the number
Nof packages to install. - The next
Nlines are packages to install. These are in the formp,vwherepis a package name andvis a version that needs to be installed. - The next line is the number
Mof dependencies - The following
Mlines are of the formp1,v1,p2,v2indicating that packagep1in versionv1depends onp2in versionv2. - Packages and version are guaranteed to not contain
,characters.
If more than one version of a package is required the installation is invalid.
The sample input above is valid, but the sample input below is invalid as A,1 requires B,1 and we are trying to install B,2:
2
A,1
B,2
3
A,1,B,1
A,2,B,2
C,1,B,1
The dependencies checker is accessible via a Command Line Interface CLI.
The program CLI.exe can be run in command line to verify given software package dependencies file description.
When run without options will output help information:
> .\publish\CLI.exe
No option specified
CLI:
Command line interface for packages dependencies validation.
Usage:
CLI <file_path>
Returns on standard output:
PASS - When packages dependencies configuration is valid.
FAIL - When packages dependencies configuration is invalid.
Errors:
Parsing errors and others are printed to standard error.
Options:
<file_path> Path to the file with dependencies description.
-v, --version Show version information
/?, -?, -h, --help Show help and usage information
To verify a configuration file simply specify a path to it:
> .\publish\CLI.exe .\testdata\input000.txt
PASS
> .\publish\CLI.exe .\testdata\input001.txt
FAIL
The program will output PASS or FAIL messages to indicate if given configuration is valid or no.
In case of invalid syntax in dependencies description file the program will output message with description about the error on standard error:
> .\publish\CLI.exe .\testdata\syntaxError.txt
Error while parsing file: '.\testdata\syntaxerror.txt'
Parse error at line 2: 'Wrong number of delimiters ',': 2', line: 'P1,42,'
| OS | Version | Architectures |
|---|---|---|
| Windows Client | 7 SP1+, 8.1 | x64, x86 |
| Windows 10 Client | Version 1607+ | x64, x86 |
| Windows Server | 2012 R2+ | x64, x86 |
| Nano Server | Version 1803+ | x64, ARM32 |
More at: system requirements details
The .NET Core in version 3.1 is required to compile and run the project.
Instal the SDK and check if it works by typing in the console:
dotnet --version
The result of the command should be value 3.1.201 or higher.
After installing .NET Core 3.1 SDK go into project directory and run the publish.bat script.
Now in the newly created publish directory will be CLI.exe executable which can be used according to Using the CLI section.
│ build.bat # builds whole solution
│ test.bat # runs tests
│ clean.ps1 # removes build files
│ publish.bat # publishes CLI.exe
|
│ dependency-sentence.sln
│ README.md
|
| # Business Logic Layer
| # Contains all the logic about parsing and
| # validating the packages dependencies files.
├───BLL
│ BLL.csproj
│ Package.cs
│ PackageDependency.cs
│ Parser.cs
│ ParseResult.cs
│ Validator.cs
│
| # Unit tests for BLL
├───BLL.Tests
│ BLL.Tests.csproj
│ ParserTests.cs
│ ValidatorTests.cs
│
| # The simple command line "front-end" to the BLL library
└───CLI
CLI.csproj
Program.cs
- Why
.batscripts?
As a simple form of documentation how to perform certain actions in the project.
They are also used by my editor to quickly build and test the projects.
In case of more complex automation tasks I use PowerShell scripts instead of.batones. - Why
stringtype for storing version of a package?
In my experience the software version number are rarely numbers.
Often they contain more then single dot or an suffix like-alphaetc. - Why name of the package is not forced to be lower case or upper case?
In my opinion the namepackageandPACKageare different names therefore my solution is case sensitive for packages names. - Why there are no documentation comments for the methods of
Package.cs,PackageDependency.csandParseResult.cs?
Those classes are simple value objects and contains only well known overloaded methods and operators. - Why
Parser.csandValidator.csare classes if they do not contain any state? Why not static classes or different solution?
For now it is very simple system but when they grow there is always need to add come kind of configuration or additional dependencies to the classes.
Having them as regular classes which have to be instantiated before use we have the ability to use them in dependency injection system or to add them configuration parameters like i.e. parameter to change the delimiters from commas,into something else in the parser. TheValidator.cscould have a configuration specifying which validation rules to use etc.
It is easy to create static class/methods but hard to go back after system grows and multiple different instances are necessary. - Why validator method
ValidateDependenciesreturns simpleboolinstead of aValidationResultobject?
I would do it like so if there was a requirement for an error message explaining why given configuration is invalid. But in this current system simpleboolis enough.
I leftTODOcomment about that in theValidator.csclass. - Why
DetermineAllPackagesToInstall()is a function and not a separate system/class?
I would for sure separate validation from determining packages to install if there was a need to actually install the packages. In that case a different system would need to have that ability so I would separate those functionalities and validator would receive this functionality as a dependency in constructor. - Why allowing for empty lines in files?
Because you left a few in your test files.
Also I think it would be to strict of a requirement to enforce no empty lines in real system. - Why comma delimiter is hardcoded?
I added the possibility to configure the delimiter but not commited it.
The syntax given in requirements is weird because the delimiter between packages is the same as delimiter between package name and version. If the format would be specified as:A,1;B,1I would for sure add the possibility to change to specify the separator in the parser.
But the situation when the separator between packages and package name and version can be the same the completely different algorithm is required and it was much less readable therefore I reverted it to the current simpler version. - Why using
FormatExceptioninstead of custom one?
It fits perfectly the given scenario and it is always better to use well known framework exceptions then to create new ones just to have a different name in stack trace.
I acctualy started with the custom parse error class but removed it (see commit dafe2df). - Why merge with
squash?
So you can easily see the branches structure - commandgit log --all --graph --oneline.
Also that is just my style, ofc I can addapt to Your specification. - Why this project took 2 days?
It did not. I was working on it in the span of 2 days, but only a few hours each day. - Why do you handle command line arguments by hand instead of using library?
The current state of libraries for command line arguments handling in .NET Core is IMO a mess.
Therefore for such a simple example I handled them myself.