With Devskiller.com you can assess your candidates' programming skills as a part of your recruitment process. We have found that programming tasks are the best way to do this and have built our tests accordingly. The way our test works is your candidate is asked to modify the source code of an existing project.
During the test, your candidates have the option of using our browser-based code editor and can build the project inside the browser at any time. If they would prefer to use an IDE they are more comfortable with, they can also download the project code or clone the project’s Git repository and work locally.
You can check out this short video to see the test from the candidate's perspective.
This repo contains a sample project for C#/.NET 6 and below you can find a detailed guide for creating your own programming project.
Please make sure to read our Getting started with programming projects guide first
It is possible to automatically assess a solution posted by the candidate. Automatic assessment is based on Tests results and Code Quality measurements.
All unit tests that are executed during the build will be detected by the Devskiller platform.
There are two kinds of unit tests:
- Candidate tests - unit tests that are visible for the candidate during the test. These should be used to do only basic verification and are designed to help the candidate understand the requirements. Candidate tests WILL NOT be used to calculate the final score.
- Verification tests - unit tests that are hidden from the candidate during the assessment. Files containing verification tests will be added to the project after the candidate finishes the test and will be executed during verification phase. Verification test results WILL be used to calculate the final score. After the candidate finishes the test, our platform builds the project posted by the candidate and executes the verification tests and static code analysis.
To create automatic assessment, you'll need compilable .NET 6 solution along with working unit tests. Any language of .NET platform can be used (C#, F#, VisualBasic), though this article focus on c# only. Currently Devskiller platform supports .NET 6 version.
Dotnet command
will be used to build, restore packages and test your code. You can use any unit-testing framework like NUnit, XUnit or MSTest.
Please refer to msdn for details about using testing frameworks.
Don't forget to add reference to Microsoft.NET.Test.Sdk
in your test projects.
To prepare your solution for automatic assessment you should follow those 3 steps:
This project should reside in separate folder in the repository. The folder structure could look like this:
CalculatorTask
│ .gitignore
│ README.md
│ devskiller.json
│ CalculatorSample.sln
│
└───CalculatorSample
│ CalculatorSample.csproj
│ Calculator.cs
│
└───CalculatorSample.Tests
│ CalculatorSample.Tests.csproj
│ Tests.cs
│
└───CalculatorSample.VerifyTests
CalculatorSample.sln
CalculatorSample.VerifyTests.csproj
VerifyTests.cs
The CalculatorTask\CalculatorSample.VerifyTests folder from example above, contains the *.csproj and code file for verification tests that will be invisible for candidate. Please note there is also additional *.sln file in this folder - we will get back to it later.
Programming task can be configured with the Devskiller project descriptor file. Just create a devskiller.json
file and place it in the root directory of your project. Here is an example project descriptor:
{
"readOnlyFiles" : [ "CalculatorSample.sln" ],
"verification" : {
"testNamePatterns" : [".*VerifyTests.*"],
"pathPatterns" : ["CalculatorSample.VerifyTests/**"],
"overwrite" : {
"CalculatorSample.VerifyTests/CalculatorSample.sln" : "CalculatorSample.sln"
}
}
}
You can find more details about devskiller.json descriptor in our documentation
In example above, by setting readOnlyFiles
field with a solution file, we make sure candidate won't be able to edit it. It's important during phase of verification tests execution, don't forget to add it!
testNamePatterns
- an array of RegEx patterns which should match all the test names of verification tests. Test names should contain:[namespace_name].[Class_name].[method_name]
. In our sample project, all verification tests are inside VerifyTests class, so the following pattern will be sufficient:
"testNamePatterns" : [".*VerifyTests.*"]
pathPatterns
- an array of GLOB patterns which should match all the files containing verification tests. All the files that match defined patterns will be deleted from candidates projects and will be added to the projects during the verification phase.
"pathPatterns" : ["CalculatorSample.VerifyTests/**"]
Because files with verification tests will be deleted from candidates projects, you need to make sure, that during final solution build, Devskiller platform will be aware of them. To make that happen, you must point which solution file should be overwritten - you want solution from CalculatorTask\CalculatorSample.VerifyTests folder to overwrite the root solution file
"CalculatorSample.VerifyTests/CalculatorSample.sln" : "CalculatorSample.sln"
So the last thing is to prepare proper solution files:
You need two solution files. One in root folder, this is the solution file that will be used by the candidate. It should have project structure that the candidate should see, so there should be no verification test project there:
CalculatorTask\CalculatorSample.sln
solution structure:
Solution 'CalculatorSample'
│
└───CalculatorSample
│
└───CalculatorSample.Tests
Second one is the 'temporary' solution file residing in verification tests folder. During final testing it will override the root solution file and thanks to that, test platform will be aware of existence of CalculatorSample.VerifyTests.csproj
in solution
CalculatorSample.VerifyTests\CalculatorSample.sln
solution structure:
Solution 'CalculatorSample'
│
└───CalculatorSample
│
└───CalculatorSample.Tests
│
└───CalculatorSample.VerifyTests
Easiest way to have those two *.sln files is to prepare solution with verification tests, copy it to verification tests folder, than go back to root solution, remove the verification tests project and save it.
- Remember that you aren't bounded to unit tests. You can do some integration tests, in example .NET 6 mvc apps are easy to instantiate within single test scope, try to use that.
- Make sure, each test is self-runnable and independent of external components like: native system libraries, external databases, etc.
- Avoid using external libraries in your source. You will never know on what operating system your code will be executed. If you need some external libraries please reference them as NuGet packages. This will make sure, you're code will behave on Devskiller platform in the same way it behaves on your environment.
- When needed and applicable, consider usage of in-memory database engine emulation when aplicable. It's fast to run and easy to use.
- Don't forget that you can utilize .NET framework references in .NET 6 projects if needed - do it with caution though (maybe only in verification tests? - its up to you), you want to test candidate against .NET 6... don't you? Otherwise consider using MSBuild
- Try to make test names clear and as short as possible. Long names will be harder to read for recruiters and can be confusing. Some test-parameter-injection methods tend to generate complex text output when executing tests. Make sure to check, if the test output looks good.
- Remember to describe clearly the task instructions in
README.md
file. - When leaving gaps in code to be filled by candidate, consider throwing
NotImplementedException
, rather than in ex. returningnull
,false
,0
, etc. in your methods. As those returned values, dependent on logic, could be some edge cases, besides, the unit tests execution will instantly fail even before checking assertions, so the candidadte will be 100% sure where he is expected to make code changes. - You want candidate to start working on the task as soon as possible, not to struggle with configuration. Project delivered for candidate should be compilable and working without any configuration needed. It should only fail tests on execution.