Each extension team develops their UI in terms of Blades and Parts. Further, they use FX APIs like container.openBlade(…)
to link between their Blades and (less commonly) the pin
function to affix Parts to user Dashboards. This is how they create the larger user experience for their service from their Blades and Parts, similar to the way they'd stitch together the pages of their own site via links.
Now, rather than perceiving each service like an island, Azure users value the ability to navigate between related services directly (from service A directly to service B) without the need to drill from the top-level (from "Browse", for instance). Users are accustomed to links in team A's UI leading to Blades from team B. They like to click a Pin
command in a team C's resource Blade and pin a monitoring Part from the Monitor extension.
When developing your Blades and Parts, it's fairly simply to mark Blades/Parts to be "exported" for reuse by other teams. Once a team "imports" Blades and Parts from another team and rebuilds their extension, they'll find code-generated BladeReferences
and PartReferences
corresponding to the newly imported Blades and Parts. These BladeReferences
and PartReferences
work with Azure Portal FX APIs just like those generated for internally developed Blades/Parts:
import { BladeReferences, PartReferences } from "Fx/Composition";
public onOpenImportedBladeClick() {
const { container } = this.context;
container.openBlade(BladeReferences.forExtension("SamplesExtension").forBlade("ExportedBlade").createReference({
parameters: { parameter1: "42" },
}));
}
public onPinImportedPartClick() {
pin([PartReferences.forExtension("SamplesExtension").forPart("ExportedPart").createReference({ parameters: { parameter1: "42" } })]);
}
This is the effect of importing Blades/Parts that have been exported by some other team. To achieve this between your extension A and some extension built by a partner team B, this is the set of steps to follow.
Exporting a Blade or Part from your extension requires a couple of simple steps. First, apply the forExport
option to your Blade or Part like so:
@TemplateBlade.Decorator({
htmlTemplate: "" +
"<div class='msportalfx-padding'>" +
" <div>This Blade is exported for use by other extensions.</div>" +
" <div data-bind='visible: bladeParameterValue'>The Blade was passed parameter value '<span data-bind='text: bladeParameterValue'></span>'.</div>" +
"</div>",
forExport: true,
})
@TemplateBlade.InjectableModel.Decorator(BladesArea.DataContext)
export class ExportedBlade {
Second, relocate the Parameters interface type for your Blade or Part into a new <YourExtensionProjectRootFolder>/Client/ForExport/<YourExtensionName>.d.ts
file that you'll share with your partner team B:
/**
* API types for Blades and Parts exported from this extension.
* This TypeScript definition file will be redistributed to extension teams who reuse this extension's exported Blades
* and Parts.
* The types here are referenced by code-generated BladeReferences and PartReferences, enabling compiler verification for
* parameters passed to the exported Blades and Parts.
*/
declare namespace SamplesExtension {
/**
* API types for the exported 'ExportedBlade' Blade.
*/
namespace ExportedBlade {
/**
* Defines the parameters that can be passed to 'ExportedBlade' when, for instance,
* it is programmatically pinned using the 'pin' function in the 'Fx/Pinner' module.
*/
interface Parameters {
/**
* An optional, sample parameter passed to 'ExportedBlade'.
*/
parameter1?: string;
}
}
Notice that you'll partition your types using TypeScript namespaces, following a structure that reflects <YourExtensionName>.<YourBladeOrPartName>.Parameters
. This makes your types discoverable to teams that make use of this .d.ts
file. It also allows you to cleanly add new types as you export more Blades and Parts in the future.
You'll find that you have to fix up any type references to your Parameters type, illustrated here:
public context: TemplateBlade.Context<SamplesExtension.ExportedBlade.Parameters, BladesArea.DataContext>;
With this, once you recompile your extension, you'll find the generated <YourExtensionName>.pde
will be expanded to include entries for your newly exported Blade/Part.
Once you've completed step 1, you'll have two artifacts that you must share with your partner teams:
- The .d.ts developed in step (1) above;
- The compiler-generated .pde file.
For this next step, follow the instructions in this doc to produce and publish a NuGet package that your partner teams can use to import your Blades and Parts.
Once a partner team applies your NuGet package to their extension and rebuilds, they can verify that new BladeReference
and PartReferences
are now generated for the newly imported Blades/Parts. The generated TypeScript for this resides in:
- /Client/_generated//BladeReferences.ts
- /Client/_generated//PartReferences.ts
Now, partner team's UI can make use of these new BladeReferences
/PartReferences
in their code, repeated from above:
import { BladeReferences, PartReferences } from "Fx/Composition";
public onOpenImportedBladeClick() {
const { container } = this.context;
container.openBlade(BladeReferences.forExtension("SamplesExtension").forBlade("ExportedBlade").createReference({
parameters: { parameter1: "42" },
}));
}
public onPinImportedPartClick() {
pin([PartReferences.forExtension("SamplesExtension").forPart("ExportedPart").createReference({ parameters: { parameter1: "42" } })]);
}
Once you've successfully published a NuGet package for your exported Blades and Parts, your team is responsible for versioning these Blades and Parts in a responsible manner, so Azure users don't experience broken links or broken experiences in the Azure Portal. You will take steps to avoid breaking your partner teams consuming your NuGet package over time. These steps include:
- Updating semantic version numbers for your NuGet package, so teams consuming your NuGet can update their version in a predictable manner
- Avoiding removing Blades and Parts from your NuGet package. For instance,
- Don't remove the
forExport
option from your Blades and Parts - Don't remove the Blade and Part classes themselves
- Don't remove the
- Modify the Parameters types in your
<YourExtensionName>.d.ts
file only in a manner that retains backwards compatibility with old versions. For instance,- Don't add non-optional properties. Only add optional properties.
- Don't changes the types of properties in a way that breaks compilation. Consider using TypeScript
union types
so you continue to support old property types along with any newly supported property types.
In situations where you must break Blades/Parts that you've already published via NuGet (for instance, you're retiring an old Blade), you will likely reach out to your partner teams to coordinate deployments around the breaking change, so no Azure users encounter disruptions.