NOTE: This section of the document assumes that the reader has reviewed the hosting service document located at top-extensions-hosting-service.md.
NOTE: This section is only relevant to extension developers who are using OneBranch VSRM and Ev2 for deployment, or who plan to migrate to OneBranch VSRM and Ev2 for deployment.
If you are not familiar with OneBranch VSRM and Ev2, it is recommended that you read the documentation provided by their teams. If you have any questions about these systems please reach out to the respective teams.
- Onboard VSRM: https://aka.ms/portalfx/vsrm
- Onboard Express V2: https://aka.ms/ev2
- Migrate WARM to VSRM: https://aka.ms/portalfx/warmmigratetovsrm
- Onboard WARM (deprecated): https://aka.ms/warm
Deploying an extension with hosting requires extension developers to upload the zip file that was generated during the build to a storage account that is read-only to the public.
Since Ev2 does not provide an API to upload the zip file, setting up the deployment infrastructure can become an unmanageable task. The deployment process is simplified by leveraging the Ev2 extension that was developed by the Ibiza team. The Ev2 extension allows the upload of the zip file to a storage account in a way that is compliant.
In the basic scenario Microsoft.Portal.Tools.targets will generate Ev2 templates. This includes the rollout spec, service model schema and parameter files. The procedure is as follows.
-
Configure Build
Microsoft.Portal.Tools.targets provides the following properties to control hosting service output generation. The properties have default values that you can override in your Extension.csproj file to meet the needs of your extension.
HostingServiceCreateDeploymentArtifacts
: defaults is undefined. Set totrue
to generate versioned hosting service zip file.HostingServiceRoutePrefix
: Defaults to$(MSBuildProjectName)
. The prefix name of your extension e.g scheduler that is supplied as part of onboarding to the extension host.HostingServicePackageOutputRootDir
: Defaults to$(OutDir)
. This is the output directory in which the build create a HostingSvc directory with generated zip file.HostingServiceEv2ExportTemplates
: Defaults totrue
. When true will generate ev2 deployment artifacts toHostingServiceEv2ServiceGroupRootSourceDir
.HostingServiceEv2OutputRootDir
: Defaults to$(HostingServicePackageOutputRootDir)
. Root directory where generated Ev2 Rollout specs will placed.HostingServiceEv2ServiceGroupRootReplacementsFilePath
: Defaults to$(MSBuildProjectDirectory)\ServiceGroupRootReplacements.json
.HostingServiceEv2ServiceGroupRootSourceDir
: Defaults to$(MSBuildThisFileDirectory)\Ev2\ServiceGroupRoot
. Use to change the default templates used for Ev2 spec generation.
For example this is the customized configuration for playground extension in CoreXT
<PropertyGroup> <HostingServiceCreateDeploymentArtifacts>$(IsOfficialBuild)</HostingServiceCreateDeploymentArtifacts> <HostingServiceRoutePrefix>[YourExtensionNameInHostingService]<</HostingServiceRoutePrefix> <HostingServiceEv2ExportTemplates>true</HostingServiceEv2ExportTemplates> </PropertyGroup>
[YourExtensionNameInHostingService]
should be replaced by an alphanumeric string that will be used when registering the extension in the hosting service. This string will be used to create the extension host name as well. Once the extension is registered in the hosting service, this value should not be changed. Outside of CoreXT, you may want to setHostingServiceCreateDeploymentArtifacts
to default to true as IsOfficialBuild will not be defined. Note to change the output path of the built zip useHostingServiceCreateDeploymentArtifacts
. To change the outputpath of Generated Ev2Artifacts setHostingServiceEv2OutputRootDir
. -
Add a ServiceGroupRootReplacements.json file to the root of your project (right next to your web.config file).
The
ServiceGroupRootReplacements.json
is a JSON file that defines few properties which are used to generate artifacts that can be used to deploy the extension using Ev2. You can define multiple objects in this file, each of those objects will map to an environment to which you would like to deploy your extension. Below are the properties that should be added to those objects.ServiceGroupRootReplacementsVersion: The schema version of the json file. The current value is 3.
AzureSubscriptionId: The Id of the subscription that contains the storage account.
CertKeyVaultUri: The Key Vault uri that contains the certificate required by Ev2. This will require you to set up Key Vault.
ContactEmail: The contact email of the extension owners.
TargetStorageCredentialsKeyVaultUri: The Key Vault Uri that contains the storage account connection string, account key, or SASToken. This will require you to set up Key Vault.
StorageAccountCredentialsType: The type of credential provided in the
TargetStorageCredentialsKeyVaultUri
property. Valid values are "SASToken", "ConnectionString", or "AccountKey". A SASToken is the preferred type to use because it is a managed secret, meaning that Key Vault will manage the secret rotation.TargetContainerName: The name of the blob container to which to upload the zip files.
PortalExtensionName: The name of the extension as it is registered in the portal, for example,
Microsoft_Azure_Compute
.FriendlyNames: A string array that contains friendly names that are managed.
MonitorDuration: The time to wait between each stage of a deployment, specified in hours and days. Also known as bake time. The minimum allowed value is 30 minutes (PT30M).
SkipSafeDeployment: A boolean value that will generate Ev2 templates that will allow you to deploy your extension without following the Safe Deployment Procedures. This is the type of deployment to use during a break-glass situation.
HotfixDeployment: A boolean value that will generate Ev2 templates that will allow you to deploy your extension as a Hotfix.
HotfixMonitorDuration: The time to wait between each stage of a hotfix deployment, specified in hours and days. Also known as bake time. The minimum allowed value is 30 minutes (PT30M).
LastKnownGoodSupport: A boolean value that will generate Ev2 templates that will enable keeping track of the last two deployments that completed the SDP process. If HotfixDeployment is also enabled, the last two completed Hotfix deployments will also be tracked.
Add
ServiceGroupRootReplacements.json
to the extension csproj, as in the following example.<Content Include="ServiceGroupRootReplacements.json"> <CopyToOutputDirectory>PreserveNewest</CopyToOutputDirectory> </Content>
The following file is an example of a
ServiceGroupRootReplacements.json
file.{ "production": { "ServiceGroupRootReplacementsVersion": 3, "AzureSubscriptionId": "<SubscriptionId>", "CertKeyVaultUri": "https://sometest.vault.azure.net/secrets/PortalHostingServiceDeploymentCertificate", "StorageAccountCredentialsType": "<ConnectionString | AccountKey | SASToken>", "TargetStorageCredentialsKeyVaultUri": "<https://sometest.vault.azure.net/secrets/PortalHostingServiceStorageConnectionString | https://sometest.vault.azure.net/secrets/PortalHostingServiceStorageAccountKey | https://sometest.vault.azure.net/secrets/PortalHostingServiceStorage-SASToken>", "TargetContainerName": "hostingservice", "ContactEmail": "youremail@microsoft.com", "PortalExtensionName": "Microsoft_Azure_Monitoring", "FriendlyNames": [ "friendlyname_1", "friendlyname_2", "friendlyname_3" ] }, "mooncake": { "ServiceGroupRootReplacementsVersion": 3, "AzureSubscriptionId": "<SubscriptionId>", "CertKeyVaultUri": "https://sometest.vault.azure.cn/secrets/PortalHostingServiceDeploymentCertificate", "StorageAccountCredentialsType": "<ConnectionString | AccountKey | SASToken>", "TargetStorageCredentialsKeyVaultUri": "<https://sometest.vault.azure.cn/secrets/PortalHostingServiceStorageConnectionString | https://sometest.vault.azure.cn/secrets/PortalHostingServiceStorageAccountKey | https://sometest.vault.azure.cn/secrets/PortalHostingServiceStorage-SASToken>", "TargetContainerName": "hostingservice", "ContactEmail": "youremail@microsoft.com", "PortalExtensionName": "Microsoft_Azure_Monitoring", "FriendlyNames": [ "friendlyname_1", "friendlyname_2", "friendlyname_3" ] } }
Other environments that are supported:
-
Test i.e. Dogfood
-
Production
-
Fairfax
-
Blackforest
-
Mooncake
-
Note: If using a non standard path for ServiceGroupRootReplacements set build property HostingServiceEv2ServiceGroupRootReplacementsFilePath
to the non standard location. This already defaults toDefaults to $(MSBuildProjectDirectory)\ServiceGroupRootReplacements.json
.
The following procedure describes how to set up the Key Vault that is required by the ServiceGroupRootReplacements.json
file, as specified in #configuring-builds-for-Ev2-based-deployments.
-
Set up Key Vault
During deployment, the zip file from the official build will be copied to the storage account that was provided when onboarding to the hosting service. To do this, Ev2 and the hosting service needs two secrets:
-
The certificate that Ev2 will use to call the hosting service to initate a deployment.
NOTE: You can use any valid certificate as Hosting Service Ev2 extension ignores this certificate but it is still required by Ev2 itself. The extension is validated based on an allowed list of storage accounts and the storage credential you supply by using the
PortalExtensionName
,TargetStorageCredentialsKeyVaultUri
andTargetContainerName
settings. -
The credentials to the target storage account where the extension will be deployed. This can be in the form of a SAS token, storage connection string, or account key.
-
-
Onboard to Key Vault
The official guidance from Ev2 is located at https://aka.ms/portalfx/ev2keyvault. Environments that are supported are the Dogfood test environment and the Production environment. Follow the instructions to:
-
Create a Key Vault.
-
Grant Ev2 read access to your Key Vault via a compound identity.
-
Create a Key Vault managed certificate.
The storage secret can be implemented a few different ways. The recommended approach is to use a SAS token generated from a Key Vault Managed Storage Account. You can find instructions in the next section. Alternate unmanaged approaches are a storage connection string or an account key.
The format of the storage connection string is the default form `DefaultEndpointsProtocol=https;AccountName={0};AccountKey={1};EndpointSuffix={3}`, which is the format provided from `portal.azure.com`.
-
-
Setup a Key Vault Managed Storage Account and SAS token
You can find additional guidance and instructions at https://aka.ms/portalfx/ev2keyvaultsastoken.
PowerShell example $regenerationPeriod = [System.Timespan]::FromDays(365) Add-AzureKeyVaultManagedStorageAccount -VaultName $keyVaultName -AccountName $storageAccountName -AccountResourceId $storageAccount.Id -ActiveKeyName $storageAccountKey -RegenerationPeriod $regenerationPeriod $sctx = New-AzureStorageContext -StorageAccountName $storageAccountName -Protocol Https -StorageAccountKey $storageAccountKey $start = [System.DateTime]::Now.AddDays(-1) $end = [System.DateTime]::Now.AddMonths(1) $at = New-AzureStorageAccountSasToken -Service Blob -ResourceType Service,Container,Object -Permission "rwl" -Protocol HttpsOnly -StartTime $start -ExpiryTime $end -Context $sctx $validityPeriod = [System.Timespan]::FromDays(10) Set-AzureKeyVaultManagedStorageSasDefinition -AccountName $storageAccountName -VaultName $keyVaultName -Name "Ev2DeploymentSas" -TemplateUri $at -SasType 'account' -ValidityPeriod $validityPeriod
You can quickly run a test deployment from a local build, previous to onboarding to WARM, by using the New-AzureServiceRollout
commandlet. Be sure that you are not testing in production, i.e, that the target storage account in the key vault that is being used is not the production storage account, and the -RolloutInfra switch is set to Test
. The following is an example of the PowerShell command.
New-AzureServiceRollout -ServiceGroupRoot E:\dev\vso\AzureUX-PortalFX\out\ServiceGroupRoot -RolloutSpec E:\dev\vso\AzureUX-PortalFX\out\ServiceGroupRoot\RolloutSpec.P1D.json -RolloutInfra Test -Verbose -WaitToComplete
Replace \<RolloutSpec>
with the path to RolloutSpec.P1D.json
in the build.
NOTE: The Ev2 Json templates perform either a 24-hour or a 6-hour rollout to each stage within the hosting service's safe deployment stages. Currently, the gating health check endpoint returns true
in all cases, so really the check only provides a time gated rollout. This means that you need to validate the health of the deployment in each stage, or cancel the deployment using the Stop-AzureServiceRollout
command if something goes wrong. Once a stop is executed, you need to rollback the content to the previous version using the New-AzureServiceRollout
command. This document will be updated once health check rules are defined and enforced. If you would like to implement your own health check endpoint you can customize the Ev2 json specs that are located in the NuGet.
To perform a production deployment as specified in https://aka.ms/portalfx/howtodeploy, or deployment by using VSRM that is specified in https://onebranch.visualstudio.com/onebranch/_wiki/wikis/OneBranch.wiki/936/Release, we assume that you have already onboarded to OneBranch VSRM. If not please see the OneBranch VSRM onboarding guidance located at https://aka.ms/portalfx/vsrm.
For Ev2 specific questions please visit https://ev2docs.azure.net or reach out to ev2sup@microsoft.com.
The preceding configuration will result in a build output as required by Ev2 and the hosting service. The following is a sample of the output that gets generated.
out\retail-amd64\ServiceGroupRoot
\HostingSvc\1.2.1.0.zip
\buildver.txt
\Production.RolloutSpec.PT6H.json
\Production.RolloutSpec.P1D.json
\Production.RolloutSpec.stage1.json
\Production.RolloutSpec.stage2.json
\Production.RolloutSpec.stage3.json
\Production.RolloutSpec.stage4.json
\Production.RolloutSpec.stage5.json
\Production.RolloutSpec.SetFriendlyName.friendlyname_1.json
\Production.RolloutSpec.SetFriendlyName.friendlyname_2.json
\Production.RolloutSpec.SetFriendlyName.friendlyname_3.json
\Production.RolloutSpec.RemoveFriendlyName.friendlyname_1.json
\Production.RolloutSpec.RemoveFriendlyName.friendlyname_2.json
\Production.RolloutSpec.RemoveFriendlyName.friendlyname_3.json
\Production\Production.ServiceModel.PT6H.json
\Production\Production.ServiceModel.P1D.json
\Production\Production.RolloutParameters.PT6H.json
\Production\Production.RolloutParameters.P1D.json
\Production\Production.ServiceModel.stage1.json
\Production\Production.ServiceModel.stage2.json
\Production\Production.ServiceModel.stage3.json
\Production\Production.ServiceModel.stage4.json
\Production\Production.ServiceModel.stage5.json
\Production\Production.RolloutParameters.stage1.json
\Production\Production.RolloutParameters.stage2.json
\Production\Production.RolloutParameters.stage3.json
\Production\Production.RolloutParameters.stage4.json
\Production\Production.RolloutParameters.stage5.json
\Production\Production.ServiceModel.SetFriendlyName.friendlyname_1.json
\Production\Production.ServiceModel.SetFriendlyName.friendlyname_2.json
\Production\Production.ServiceModel.SetFriendlyName.friendlyname_3.json
\Production\Production.RolloutParameters.SetFriendlyName.friendlyname_1.json
\Production\Production.RolloutParameters.SetFriendlyName.friendlyname_2.json
\Production\Production.RolloutParameters.SetFriendlyName.friendlyname_3.json
\Production\Production.ServiceModel.RemoveFriendlyName.friendlyname_1.json
\Production\Production.ServiceModel.RemoveFriendlyName.friendlyname_2.json
\Production\Production.ServiceModel.RemoveFriendlyName.friendlyname_3.json
\Production\Production.RolloutParameters.RemoveFriendlyName.friendlyname_1.json
\Production\Production.RolloutParameters.RemoveFriendlyName.friendlyname_2.json
\Production\Production.RolloutParameters.RemoveFriendlyName.friendlyname_3.json
The Ev2 template files that are shipped are now formatted to accept customized monitor durations, which is the time to wait between each stage of a deployment, also known as bake time. To accommodate this, the files have been renamed from Blackforest.RolloutParameters.PT6H.json
to
Blackforest.RolloutParameters.{MonitorDuration}.json
.
The monitor duration can be specified by updating the ServiceGroupRootReplacements.json
file to include a new array called "MonitorDuration", as in the following example.
{
"production": {
"ServiceGroupRootReplacementsVersion": 3,
"AzureSubscriptionId": "<SubscriptionId>",
"CertKeyVaultUri": "https://sometest.vault.azure.net/secrets/PortalHostingServiceDeploymentCertificate",
"StorageAccountCredentialsType": "<ConnectionString | AccountKey | SASToken>",
"TargetStorageCredentialsKeyVaultUri": "<https://sometest.vault.azure.net/secrets/PortalHostingServiceStorageConnectionString | https://sometest.vault.azure.net/secrets/PortalHostingServiceStorageAccountKey>",
"TargetContainerName": "hostingservice",
"ContactEmail": "youremail@microsoft.com",
"PortalExtensionName": "Microsoft_Azure_Monitoring",
"FriendlyNames": [ "friendlyname_1", "friendlyname_2", "friendlyname_3" ],
"MonitorDuration": [ "PT30M", "PT1H" ],
}
}
If no monitor durations are specified, then the Ev2 generation will default to 6 hours (PT6H) and 1 day (P1D).
The Ev2 templates support generating SDP and Hotfix deployment files that include Last Known Good (LKG) support. LKG support enables keeping track of the last two deployments that completed the SDP deployment process.
Enabling LKG support, along with HotfixDeployment, enables keeping track of the last two completed Hotfix deployments. Additionally, when LKG support is enabled, LKG Revert is supported.
LKG support is enabled by adding the key/value pair "LastKnownGoodSupport": "true"
More information about LKG and other improvements https://msit.microsoftstream.com/video/22893a4c-975c-4aed-b01e-bfd5c61e73b5?st=1065
How does LKG work
After final stage or hotfix deployment has satisfied the monitoring duration, the build is promoted to LastKnownGoodBuild / LastHotfixBuild.
If LastKnownGoodBuild or LastHotfixBuild has an existing value, this existing value (version) will be promoted (copied) to PreviousLastKnownGoodBuild / PreviousLastHotfixBuild.
How does LKG Revert work
LKG revert is a mechanism to quickly revert your extension version back to a previously known good verion without requiring a redeployment of the specific build. LKG Revert can revert all stages to the version from LastKnownGoodBuild, LastHotfixBuild, PreviousLastKnownGoodBuild, or PreviousLastHotfixBuild.
An LKG revert will update the stages sequentially with no delay between stages.
Ev2 templates support generating deployment files that do not include a delay between stages. This can be enabled by adding the key/value pair "SkipSafeDeployment": "true"
in the corresponding environment in the ServiceGroupRootReplacements.json
file. The following example adds the SkipSafeDeployment key/value pair to the extension named Microsoft_MyExtension
in the Production environment.
{
"production": {
"ServiceGroupRootReplacementsVersion": 2,
"AzureSubscriptionId": "<SubscriptionId>",
"CertKeyVaultUri": "https://sometest.vault.azure.net/secrets/PortalHostingServiceDeploymentCertificate",
"StorageAccountCredentialsType": "<ConnectionString | AccountKey | SASToken>",
"TargetStorageCredentialsKeyVaultUri": "<https://sometest.vault.azure.net/secrets/PortalHostingServiceStorageConnectionString | https://sometest.vault.azure.net/secrets/PortalHostingServiceStorageAccountKey>",
"TargetContainerName": "hostingservice",
"ContactEmail": "youremail@microsoft.com",
"PortalExtensionName": "Microsoft_Azure_Monitoring",
"FriendlyNames": [ "friendlyname_1", "friendlyname_2", "friendlyname_3" ],
"SkipSafeDeployment": "true"
}
}
To remove a friendly name, just run an Ev2 deployment with the Rolloutspec.RemoveFriendlyName.<friendlyName>.json
file.