Covered in this document:
- Creating a project from the template repository.
- Creating a project from scratch.
- CI: test results output to CI accepted formats JUNIT, TRX, other.
- Code Coverage: where to find your code coverage output.
- Follow the getting started guide.
Before getting started with this step by step:
- if building a new extension from scratch its simpler/faster to start with the above project template rather then following the step by step.
- if adding a UT project to an existing extension you can still just scaffold the content from above then follow this guide for customizing the config to point to your existing extension.
This tutorial will provide you step by step instructions for creating a UnitTest project for your Azure Portal Extension. The resulting folder structure will look like the following:
+-- Extension
+-- Extension.UnitTests
| +-- test/CreateBlade.test.ts
| +-- test/ResourceOverviewBlade.test.ts
| +-- test-main.js
| +-- karma.conf.js
| +-- msportalfx-ut.config.json
| +-- package.json
| +-- tsconfig.json
+-- .npmrc
// The build and code generation will add the following
| +-- _generated
| +-- Ext
| +-- Fx
| +-- Output
Note:
- This document uses relative paths to indicate where you should add each file relative to the root of your test folder e.g ./package.json indicates adding an package.json at the root of your test project folder Extension.UnitTests/package.json
- All code snippets provided are for
Microsoft.Portal.Tools.V2.targets
if you are using it's predecessorMicrosoft.Portal.Tools.targets
see the FAQ at the bottom of this document.
msportalfx-ut is available from the internal AzurePortal Registry. To configure your project to use this registry add the following:
Add a ./.npmrc file
registry=https://msazure.pkgs.visualstudio.com/_packaging/AzurePortal/npm/registry/
always-auth=true
- Add the ./package.json
{
"name": "extension-ut",
"version": "1.0.0",
"description": "",
"main": "index.js",
"scripts": {
"restore": "npm install --silent --no-color --no-optional",
"build": "npm run restore && gulp generateAmdModuleFromResx --no-color --gulpfile=./node_modules/msportalfx-ut/gulpfile.js --silent --cwd ./ && tsc -p tsconfig.json",
"test": "npm run build && karma start",
"test-ci": "npm run build && karma start --single-run --no-colors"
},
"keywords": [
"unittest"
],
"author": "Microsoft",
"license": "MIT",
"dependencies": {},
"devDependencies": {
"@types/chai": "4.1.7",
"@types/mocha": "5.2.5",
"@types/nconf": "0.10.0",
"@types/sinon": "7.0.5",
"chai": "4.2.0",
"gulp": "4.0.0",
"gulp-concat": "2.6.1",
"karma": "^4.0.1",
"karma-chai": "0.1.0",
"karma-chrome-launcher": "2.2.0",
"karma-coverage": "1.1.2",
"karma-edge-launcher": "0.4.2",
"karma-mocha": "1.3.0",
"karma-mocha-reporter": "2.2.5",
"karma-junit-reporter": "1.2.0",
"karma-requirejs": "1.1.0",
"karma-trx-reporter": "0.4.0",
"mocha": "5.2.0",
"msportalfx-ut": "file:../../packages/Microsoft.Portal.TestFramework.UnitTest.$(CURRENT_BUILD_VERSION)/msportalfx-ut-$(NPM_CURRENT_BUILD_VERSION).tgz",
"nconf": "0.10.0",
"requirejs": "2.3.6",
"sinon": "7.2.3",
"typescript": "3.5.3"
}
}
In the package.json you can see we're using mocha and chai but you can choose your own test and assertion framework.
-
Update ./package.json to refer directly to msportalfx-ut i.e rather then using the file:// syntax simply specify
"msportalfx-ut" : "5.302.VersionOfSdkYouAreUsing"
(notice difference in versioning, no 0 as minor version) -
run the following command.
npm install --no-optional
Note:
- If you receive auth errors against the internal NPM feed see the "Connect to feed" instructions here
msportalfx-ut.config.json defines paths to those files needed by the msportalfx-ut node module to generate everything under ./_generated/*
.
add ./msportalfx-ut.config.json with the following:
{
"UTNodeModuleRootPath": "./node_modules/msportalfx-ut",
"GeneratedAssetRootPath": "./_generated",
"ResourcesResxRootDirectory": "../Extension/Client"
}
Customize the paths to those of your extension. The definition of each key in the config is as follows:
UTNodeModuleRootPath
: the root path to where the msportalfx-ut node module was installed.GeneratedAssetRootPath
: the root path to save all assets that will be generated e.g your resx to js AMD module so you can use resource strings directly in your test.ResourcesResxRootDirectory
: extension client root directory that contains all *.resx files. These will be used to generate js AMD string resource modules intoGeneratedAssetRootPath
and the associated require config.
Note:
- if your official build environment uses different paths then your dev environment you can override them either by using command line arguments or environmental variables. The msportalfx-ut gulpfile will search in the following order command line argument > environmental variable > ./msportalfx-ut.config.json file. An example of overriding an item for an official build via a command line argument
gulp --UTNodeModuleResolutionPath ./some/other/location
Add a CreateBlade test to ./test/CreateBlade.test.ts. This demonstrates how to provide the provisioning context to your CreateBlade that portal would normally provide via your gallery package. You can modify this example for your own extension.
import { CreateBlade } from "Resource/Create/CreateBlade";
import * as sinon from "sinon";
import { TemplateBladeHarness } from "msportalfx-ut/Harness";
describe("Create Blade Tests", () => {
let server: sinon.SinonFakeServer;
beforeEach(function () {
server = sinon.fakeServer.create();
server.respondImmediately = true;
});
afterEach(function () {
server.restore();
});
it("Verify initial state of CreateBlade", () => {
return TemplateBladeHarness.initializeBlade(CreateBlade, {
parameters: null,
blade: new CreateBlade(),
provisioningContext: {
initialValues: { locationNames: [], subscriptionIds: [] },
telemetryId: "",
provisioningConfig: {
dashboardPartReference: {
dashboardPartKeyId: "id",
options: {
extensionName: "ExtensionTemplate",
},
partName: "ResourcePart",
parameters: null,
},
},
marketplaceItem: null,
},
beforeOnInitializeCalled: (blade) => {
console.log("Add any before on init tests here");
},
afterOnInitializeCalled: (blade) => {
console.log("Add any after on init tests here");
},
}).then((blade) => {
console.log("Add any init complete tests here");
});
});
});
Add a TemplateBlade test to ./test/ResourceOverviewBlade.test.ts. You can modify this example for your own extension.
import { assert } from "chai"; // type issues with node d.ts and require js d.ts so using chai
import { Parameters, ResourceOverviewBlade } from "Resource/Blades/Overview/ResourceOverviewBlade";
import ClientResources = require("ClientResources");
import * as sinon from "sinon";
import { TemplateBladeHarness } from "msportalfx-ut/Harness";
describe("Resource Overview Blade Tests", () => {
let server: sinon.SinonFakeServer;
beforeEach(function () {
server = sinon.fakeServer.create();
server.respondImmediately = true;
});
afterEach(function () {
server.restore();
});
it("title populated with content from ARM", () => {
// arrange
const resourceId = "/subscriptions/0c82cadf-f711-4825-bcaf-44189e8baa9f/resourceGroups/sdfsdfdfdf/providers/Providers.Test/statefulIbizaEngines/asadfasdff";
server.respondWith((request) => {
if (request.url.startsWith(`${MsPortalFx.getEnvironmentValue("armEndpoint")}/batch`)
&& JSON.parse(request.requestBody).requests[0].url.endsWith(`${resourceId}?api-version=${MsPortalFx.getEnvironmentValue("armApiVersion")}`)) {
request.respond(200, { "Content-Type": "application/json" }, JSON.stringify({
"responses": [
{
"httpStatusCode": 200,
"content": {
"id":`${resourceId}`,
"name": "bar",
"type": "Providers.Test/statefulIbizaEngines",
"location": "East Asia",
"properties": {},
},
},
],
}));
} else {
request.respond(404, null, "not mocked");
}
});
const bladeParameters : Parameters = { id: resourceId };
// options for the blade under test. optional callbacks beforeOnInitializeCalled, afterOnInitializeCalled and afterRevealContentCalled
// can be supplied to execute custom test code
// get blade instance with context initialized and onInitialized called
return TemplateBladeHarness.initializeBlade(ResourceOverviewBlade, {
parameters: bladeParameters,
blade: new ResourceOverviewBlade(),
afterOnInitializeCalled: (blade) => {
console.log("after on init called");
},
beforeOnInitializeCalled: (blade) => {
console.log("before on init called");
},
afterRevealContentCalled: (blade) => {
console.log("after reveal called");
},
}).then((resourceBlade) => {
assert.equal(resourceBlade.title(), "bar");
assert.equal(resourceBlade.subtitle, ClientResources.resourceOverviewBladeSubtitle);
});
});
});
-
To compile your test and for dev time intellisense you will need a ./tsconfig.json
{ "compileOnSave": true, "compilerOptions": { "baseUrl": ".", "experimentalDecorators": true, "module": "amd", "noImplicitAny": true, "noImplicitThis": true, "noUnusedLocals": true, "outDir": "./Output", "rootDir": ".", "sourceMap": false, "strictBindCallApply": true, "target": "es5", "paths": { "msportalfx-ut/*": [ "./node_modules/msportalfx-ut/lib/*" ], "*": [ "../Extension/Output/Content/Scripts/*", "./node_modules/@types/*/index" ] } }, "references": [ { "path": "../Extension" } ], "exclude": [], "include": [ "../Extension/Definitions/*", "../Extension/Output/typings/ClientResources.d.ts", "test/**/*" ]
}
```
Update the paths in the tsconfig.json to for your specific extension paths.
-
Build your extension
-
run the following command to build your tests
npm run build
Now that your tests are building, add the following to run your tests.
requirejs and mocha need to know where all your modules are located for your extension and any frameworks that you are using.
add a ./test-main.js file as the main entrypoint for your app
window.fx.environment.armEndpoint = "https://management.azure.com";
window.fx.environment.armApiVersion = "2014-04-01";
const allTestFiles = []
if (window.__karma__) {
const TEST_REGEXP = /^\/base\/Extension.UnitTests\/Output\/.*(spec|test)\.js$/i;
// Get a list of all the test files to include
Object.keys(window.__karma__.files).forEach(function (file) {
if (TEST_REGEXP.test(file)) {
// Normalize paths to RequireJS module names.
// If you require sub-dependencies of test files to be loaded as-is (requiring file extension)
// then do not normalize the paths
const normalizedTestModule = file.replace(/^\/base\/Extension.UnitTests\/|\.js$/g, "")
allTestFiles.push(normalizedTestModule)
}
});
}
mocha.setup({
ui: "bdd",
timeout: 60000,
ignoreLeaks: false,
globals: []
});
rjs = require.config({
// Karma serves files under /base, which is the basePath from your config file
baseUrl: window.__karma__ ? "/base/Extension.UnitTests" : "",
paths: {
"_generated": "../Extension/Output/Content/Scripts/_generated",
"Resource": "../Extension/Output/Content/Scripts/Resource",
"Shared": "../Extension/Output/Content/Scripts/Shared",
"sinon": "node_modules/sinon/pkg/sinon",
"chai": "node_modules/chai/chai",
},
// dynamically load all test files
deps: allTestFiles,
// kickoff karma or mocha
callback: window.__karma__ ? window.__karma__.start : function () { return mocha.run(); }
});
Update for your specific extension paths.
In this example we will using karmajs as a test runner. It provides a rich plugin ecosystem for watch based compile on save based dev/test cycles, test reporting and code coverage to name a few.
add a file named ./karma.conf.js
// Karma configuration
// Generated on Fri Feb 16 2018 15:06:08 GMT-0800 (Pacific Standard Time)
module.exports = function (config) {
config.set({
// base path that will be used to resolve all patterns (eg. files, exclude)
basePath: "../",
// frameworks to use
// available frameworks: https://npmjs.org/browse/keyword/karma-adapter
frameworks: ["mocha"],
plugins: [
require("karma-mocha"),
require("karma-mocha-reporter"),
require("karma-edge-launcher"),
require("karma-coverage"), // Include if you want coverage
require("karma-chrome-launcher"),
require("karma-junit-reporter"), // Include if you want junit reporting
require("karma-trx-reporter") // Include if you want trx reporting
],
// list of files / patterns to load in the browser
files: [
// chai assertion framework.
{ pattern: "Extension.UnitTests/node_modules/chai/**/*.js", included: false },
// sinonjs used for mocking xhr.
{ pattern: "Extension.UnitTests/node_modules/sinon/**/*.js", included: false },
// aggregate script of portal bundles required for test.
"Extension.UnitTests/node_modules/msportalfx-ut/lib/FxScripts.js",
// karma requirejs adapter required to successfully load requirejs in karma.
"Extension.UnitTests/node_modules/karma-requirejs/lib/adapter.js",
// generated require configs for extension resx files.
{ pattern: "Extension.UnitTests/_generated/Ext/**/*RequireConfig.js", included: true },
// msportalfx-ut test harness and other test scripts you may load within a unit test.
{ pattern: "Extension.UnitTests/node_modules/msportalfx-ut/lib/*.js", included: false },
// portal framework scripts.
{ pattern: "Extension.UnitTests/node_modules/msportalfx-ut/lib/fx/Content/Scripts/**/*.js", included: false },
// reserved directory for generated content for framework.
{ pattern: "Extension.UnitTests/_generated/Fx/**/*.js", included: false },
// generated content for extension.
{ pattern: "Extension.UnitTests/_generated/Ext/**/*.js", included: false },
// make available compiled tests from tsconfig.json outDir
{ pattern: "Extension.UnitTests/Output/**/*.test.js", included: false },
// make available all client extension code that unit tests will use.
{ pattern: "Extension/Output/Content/Scripts/**/*.js", included: false },
// the entrypoint for running unit tests.
"Extension.UnitTests/test-main.js",
],
client: {
mocha: {
reporter: "html",
ui: "bdd"
}
},
// list of files / patterns to exclude
exclude: [
],
// preprocess matching files before serving them to the browser
// available preprocessors: https://npmjs.org/browse/keyword/karma-preprocessor
preprocessors: {
"./Extension/Output/Content/Scripts/**/*.js": "coverage"
},
// available reporters: https://npmjs.org/browse/keyword/karma-reporter
reporters: ["mocha", "trx", "junit", "coverage"],
// the default trx configuration
trxReporter: { outputFile: "./TestResults/test-results.trx", shortTestName: false },
junitReporter: {
outputDir: "./Extension.UnitTests/TestResults", // results will be saved as $outputDir/$browserName.xml
outputFile: "test-results.xml", // if included, results will be saved as $outputDir/$browserName/$outputFile
suite: "Extension.UnitTests", // suite will become the package name attribute in xml testsuite element
useBrowserName: true, // add browser name to report and classes names
nameFormatter: undefined, // function (browser, result) to customize the name attribute in xml testcase element
classNameFormatter: undefined, // function (browser, result) to customize the classname attribute in xml testcase element
properties: {} // key value pair of properties to add to the <properties> section of the report
},
coverageReporter: {
type: "html",
dir: "./Extension.UnitTests/TestResults/coverage/"
},
// web server port
port: 9876,
// enable / disable colors in the output (reporters and logs)
colors: true,
// level of logging
// possible values: config.LOG_DISABLE || config.LOG_ERROR || config.LOG_WARN || config.LOG_INFO || config.LOG_DEBUG
logLevel: config.LOG_INFO,
// enable / disable watching file and executing tests whenever any file changes
autoWatch: true,
// start these browsers
// available browser launchers: https://npmjs.org/browse/keyword/karma-launcher
browsers: ["Chrome_No_Sandbox", "Edge"],
customLaunchers: {
Chrome_No_Sandbox: {
base: 'Chrome',
flags: ['--no-sandbox']
}
},
// Continuous Integration mode
// if true, Karma captures browsers, runs the tests and exits
singleRun: false,
// Concurrency level
// how many browser should be started simultaneous
concurrency: Infinity,
})
}
Update for your specific extension paths.
You can run your tests with npm run test
. This command is specified in packages.json and will start karmajs in your configured target browser(s) in watch mode. This is particularly useful when used in conjunction with compile on save allowing for an efficient inner dev loop i.e dev > compile > automatic test execution, in short any change to your extension src and test project src will be automatically compiled (due to tsconfig.json setup) and then tests will automatically (due to karma.conf.js). The net result is real time feedback on what tests you've broken as you modifying extension code.
Using karmajs for a single test run useful for scenarios such as running in CI
npm run test-ci
launches karmajs in your configured target browsers for a single run.
Note that the karma.conf.js is configured to run your tests in both in Edge and Chrome. You may also pick and choose additional browsers via the launcher plugins documented here.
In dev scenarios the test results from running your tests via karmajs are available in the console output. In addition to the console output reporters have been added to karma.conf.js
to provide formats that are useful to CI scenarios.
By Default the project template/steps above will generate a project configured to produce JUNIT and TRX output. This should be useful for most CI environments. The content will be output under ./TestResults/**/*.xml|trx. To configure the output paths update karma.conf.js. For more plugins for outputing in a format your CI environment supports see karmajs official docs.
Note:
- TRX and JUNIT output are generated when running
npm run test
ornpm run test-ci
via karmajs and its karma.conf.js. - drop the TRX or JUNIT reporter that is not needed for your CI environment.
By Default the project template/steps above will generate a project configured that also produces code coverage using karma-coverage. The content will be output under ./TestResults/Coverage/**/index.html
Note:
- coverage results are generated when running
npm run test
ornpm run test-ci
via karmajs its karma.conf.js.
Clicking through from the summary view to the ResourceOverviewBlade you can see code coverage line by line
Severity Code Description Project File Line Suppression State
Error Could not install package 'PortalFx.NodeJS8 10.0.0.125'. You are trying to install this package into a project that targets '.NETFramework,Version=v4.6.1', but the package does not contain any assembly references or content files that are compatible with that framework. For more information, contact the package author
Solution: Install the Node Tools for Visual Studio
from here
Build environments which are setup using Corext will need to manually add additional lines in order to specify where to pick up the Unit Test Framework NuGet package. This NuGet package will be expanded at a different location (CxCache) than the default, so you need to update your Corext config and the npm packages.json in order to point to the correct location.
If you are a executing the default instructions on your Corext environment, this is the error you will see:
npm ERR! enoent ENOENT: no such file or directory, stat 'C:\...\ExtensionName\packages\Microsoft.Portal.TestFramework.UnitTest.5.0.302.979\msportalfx-ut-5.302.979.tgz'
This error indicates that it cannot find the expanded NuGet package for the Unit Test Framework.
-
Update your Repository's
corext.config
Found under
\<ExtensionRepoName>\.corext\corext.config
Under the
<generator>
section add the following to the bottom:<!-- Unit Test Framework --> <package id="Microsoft.Portal.TestFramework.UnitTest" />
-
Update your Extension's
package.config
Found under
\<ExtensionRepoName>\src\<ExtensionName>\packages.config
Add the folowing to your
<packages>
:<package id="Microsoft.Portal.TestFramework.UnitTest" version="5.0.302.1016" targetFramework="net45" />
Note: The version listed above should match the version of the Portal SDK you are using for your extension and will match the
"Microsoft.Portal.Framework"
package in yourpackages.config
. -
Update your Unit Test
package.json
Found under
<ExtensionRepoName>\src\<ExtensionName>.UnitTests\package.json
If you have the package
msportalfx-ut
in your dependencies, remove it.Update your scripts init command to be the following:
"init": "npm install --no-optional && npm install %PkgMicrosoft_Portal_TestFramework_UnitTest%\\msportalfx-ut-5.302.1016.tgz --no-save",
Note: The version listed above should match the version of the Portal SDK you are using for your extension and will match the
"Microsoft.Portal.Framework"
package in yourpackages.config
. -
Run the following commands to finish the setup
- Run Corext Init - From the root of your repository run:
init
- Run the Npm install - From your Unit Test directory run:
npm run init
- Run Corext Init - From the root of your repository run:
Use the Harness.init function and supply an options object of type InitializationOptions with your own stub(s) that will override the default stub provided by the unit test framework.
example usage:
import * as harness from "msportalfx-ut/Harness";
...
const options : harness.InitializationOptions = {
getUserInfo: sinon.stub().returns(Q({
email: "ibizaems@microsoft.com",
isOrgId: false,
givenName: "givenName",
surname: "surName",
directoryId: "00000000-0000-0000-0000-000000000000",
directoryName: "directoryName",
uniqueDirectoryName: "uniqueDirectoryName",
domainName: "domainName"
})),
...
};
harness.init(options);
...
I am still using Microsoft.Portal.Tools.targets rather than Microsoft.Portal.Tools.V2.targets. What do I need to change
Ideally you should move to v2 targets to benefit from the dev productivity it will bring to your dev/test inner loop. Migration only takes about an hour or so, see https://aka.ms/portalfx/cloudbuild for details. If you can't migrate to v2 update the following:
-
Update your ../Extension/Extension.csproj add
xml
<PropertyGroup> <TypeScriptGeneratesDeclarations>true</TypeScriptGeneratesDeclarations> </PropertyGroup>
-
./package.json update the
script
namedprereq
to be the following"prereq": "npm run init && gulp --gulpfile=./node_modules/msportalfx-ut/gulpfile.js --cwd ./"
-
./msportalfx-ut.config.js
"ExtensionTypingsFiles": "../Extension/**/*.d.ts", "ResourcesResxRootDirectory": "../Extension/Client"
-
./tsconfig.json
{ "compileOnSave": true, "compilerOptions": { ... "paths": { "msportalfx-ut/*": [ "./node_modules/msportalfx-ut/lib/*" ], "*": [ "./_generated/Ext/typings/Client/*", "./node_modules/@types/*/index" ] } ... }, ... "include": [ "./_generated/Ext/typings/Definitions/*", "test/**/*" ] }
-
./karma.conf.js
... files: [ ..., { pattern: "Extension/Output/Content/Scripts**/*.js", included: false }, ... ], ... preprocessors: { "./Extension/Client/**/*.js": "coverage" }
-
./test-main.js
rjs = require.config({ ..., "_generated": "../Extension/Client/_generated", "Resource": "../Extension/Client/Resource", "Shared": "../Extension/Client/Shared" ...
I can't use the internal npm registry https://msazure.pkgs.visualstudio.com/_packaging/AzurePortal/npm/registry/ because ...
Try the following:
- If you receive auth errors against the internal NPM feed see the "Connect to feed" instructions here
- If your not a member of any of the Groups on https://msazure.visualstudio.com/One/Azure%20Portal/_packaging?feed=AzurePortal&_a=settings&view=permissions please join
Azure Portal Partner Contributors – 19668
in https://myaccess .
- you can commit your version of msportalfx-ut.zip into your repo and install it directly using relative file path syntax
"msportalfx-ut": "file:../externals/msportalfx-ut-5.302.someversion.tgz",