Options
All
  • Public
  • Public/Protected
  • All
Menu

Module SelfServe

Self Serve Model

The Self Serve Model allows you to write classes that auto generate UI components for your feature. The idea is to allow developers from other feature teams, who may not be familiar with writing UI, to develop and own UX components. This is accomplished by just writing simpler TypeScript classes for their features.

What this means for the feature team

  • Can concentrate just on the logic behind showing, hiding and disabling UI components
  • Need not worry about specifics of the UI language or UX requirements (Accessibility, Localization, Themes, etc.)
  • Can own the REST API calls made as part of the feature, which can change in the future
  • Quicker turn around time for development and bug fixes since they have deeper knowledge of the feature

What this means for the UI team

  • No need to ramp up on the intricacies of every feature which requires UI changes
  • Own only the framework and not every feature, giving more bandwidth to prioritize inhouse features as well

Getting Started

Clone the cosmos-explorer repo and run

  • npm install
  • npm run build

Click here for more info on setting up the cosmos-explorer repo.

Code Changes

Code changes need to be made only in the following files

  • A JSON file - for strings to be displayed
  • A Types File - for defining the data models
  • A RP file - for defining the REST calls
  • A Class file - for defining the UI
  • SelfServeUtils.tsx and SelfServe.tsx - for defning the entrypoint for the UI

1. JSON file for UI strings

Naming Convention

Localization/en/<FEATURE_NAME>.json
Please place your files only under "Localization/en" folder. If not, the UI strings will not be picked up by the framework.

Example

SelfServeExample.json

Description

This is a JSON file where the values are the strings that needs to be displayed in the UI. These strings are referenced using their corresponding unique keys.

For example, If your class file defines properties as follows

  @Values({
    labelTKey: "stringPropertylabel"
  })
  stringProperty: string;

  @Values({
    labelTKey: "booleanPropertyLabel",
    trueLabelTKey: "trueLabel",
    falseLabelTKey: "falseLabel",
  })
  booleanProperty: boolean;

Then the content of Localization/en/FeatureName.json should be

{
    stringPropertyLabel: "string property",
    booleanPropertyLabel: "boolean property",
    trueLabel: "Enable",
    falseLabel: "Disable"
}

You can learn more on how to define the class file here.

2. Types file

Naming Convention

<FEATURE_NAME>.types.ts

Example

SelfServeExample.types.ts

Description

This file contains the definitions of all the data models to be used in your Class file and RP file.

For example, if your RP call takes/returns the stringProperty and booleanProperty of your SelfServe class, then you can define an interface in your FeatureName.types.ts file like this.

export RpDataModel {
  stringProperty: string,
  booleanProperty: boolean
}

3. RP file

Naming Convention

<FEATURE_NAME>.rp.ts

Example

SelfServeExample.rp.ts

Description

The RP file will host the REST calls needed for the initialize, save and refresh functions. This decouples the view and the model of the feature.

To make the ARM call, we need some information about the Azure Cosmos DB databaseAccount - the subscription id, resource group name and database account name. These are readily available through the userContext object, exposed through

  • userContext.subscriptionId
  • userContext.resourceGroup
  • userContext.databaseAccount.name

You can use the armRequestWithoutPolling function to make the ARM api call.

Your FeatureName.rp.ts file can look like the following.

import { userContext } from "../../UserContext";
import { armRequestWithoutPolling } from "../../Utils/arm/request";
import { configContext } from "../../ConfigContext";

const apiVersion = "2020-06-01-preview";

export const saveData = async (properties: RpDataModel): Promise<string> => {
  const path = `/subscriptions/${userContext.subscriptionId}/resourceGroups/${userContext.resourceGroup}/providers/Microsoft.DocumentDB/databaseAccounts/${userContext.databaseAccount.name}/<REST_OF_THE_PATH>`
  const body = {
      data : properties
  }
  const armRequestResult = await armRequestWithoutPolling({
    host: configContext.ARM_ENDPOINT,
    path,
    method: "PUT",
    apiVersion,
    body,
  });

  return armRequestResult.operationStatusUrl;
};

4. Class file

Naming Convention

<FEATURE_NAME>.tsx

Example

SelfServeExample.tsx

Description

This file will contain the actual code that is translated into the UI component by the Self Serve framework.

  • Each Self Serve class

    • Needs to extends the SelfServeBase class.
    • Needs to have the @IsDisplayable() decorator to tell the compiler that UI needs to be generated from this class.
    • Needs to define an initialize() function, to set default values for the inputs.
    • Needs to define an onSave() function, a callback for when the save button is clicked.
    • Needs to define an onRefresh() function, a callback for when the refresh button is clicked.
    • Can have an optional @RefreshOptions() decorator that determines how often the auto refresh of the UI component should take place.
  • For every UI element needed, add a property to the Self Serve class. Each of these properties

    • Needs to have a @Values() decorator.
    • Can have an optional @PropertyInfo() decorator that describes it's info bubble.
    • Can have an optional @OnChange() decorator that dictates the effects of the change of the UI element tied to this property.

Your FeatureName.tsx file will look like the following.

@IsDisplayable()
@RefreshOptions({ retryIntervalInMs: 2000 })
export default class FeatureName extends SelfServeBaseClass {

  public initialize = async (): Promise<Map<string, SmartUiInput>> => {
      // initialize RP call and processing logic
  }

  public onSave = async (
    currentValues: Map<string, SmartUiInput>,
    baselineValues: ReadonlyMap<string, SmartUiInput>
  ): Promise<OnSaveResult> => {
      // onSave RP call and processing logic
  }

  public onRefresh = async (): Promise<RefreshResult> => {
      // refresh RP call and processing logic
  };

  @Values(...)
  stringProperty: string;

  @OnChange(...)
  @PropertyInfo(...)
  @Values(...)
  booleanProperty: boolean;
}

5. Update SelfServeType

Once you have written your Self Serve Class, add a corresponding type to SelfServeType

export enum SelfServeType {
  invalid = "invalid",
  example = "example",
  ...
  // Add the type for your new feature
  featureName = "featurename"
}

6. Update SelfServe.tsx (landing page)

Once the SelfServeType has been updated, update SelfServe.tsx for your feature. This ensures that the framework picks up your SelfServe Class.

const getDescriptor = async (selfServeType: SelfServeType): Promise<SelfServeDescriptor> => {
  switch (selfServeType) {
    case SelfServeType.example: {
        ....
    }
    ...
    ...
    ...
    // Add this for your new feature
    case SelfServeType.featureName: {
      // The 'webpackChunkName' is used during debugging, to identify if the correct class has been loaded
      const FeatureName = await import(/* webpackChunkName: "FeatureName" */ "./FeatureName/FeatureName");
      const featureName = new FeatureName.default();
      await loadTranslations(featureName.constructor.name);
      return featureName.toSelfServeDescriptor();
    }
    ...
    ...
    default:
      return undefined;
  }
};

Telemetry

You can add telemetry for your feature using the functions in SelfServeTelemetryProcessor

For example, in your SelfServe class, you can call the trace method in your onSave function.

import { saveData } from "./FeatureName.rp"
import { RpDataModel } from "./FeatureName.types"

@IsDisplayable()
export default class FeatureName extends SelfServeBaseClass {

  .
  .
  .

  public onSave = async (
    currentValues: Map<string, SmartUiInput>,
    baselineValues: ReadonlyMap<string, SmartUiInput>
  ): Promise<OnSaveResult> => {

    stringPropertyValue = currentValues.get("stringProperty")
    booleanPropertyValue = currentValues.get("booleanProperty")
    
    const propertiesToSave : RpDataModel = { 
      stringProperty: stringPropertyValue,
      booleanProperty: booleanPropertyValue
    }
    const telemetryData = { ...propertiesToSave, selfServeClassName: FeatureName.name }
    const onSaveTimeStamp = selfServeTraceStart(telemetryData)

    await saveData(propertiesToSave)

    selfServeTraceSuccess(telemetryData, onSaveTimeStamp)

    // return required values
  }

  .
  .
  .

  @Values(...)
  stringProperty: string;

  @Values(...)
  booleanProperty: boolean;
}

Portal Notifications

You can enable portal notifications for your feature by passing in the required strings as part of the portalNotification property of the onSaveResult.

@IsDisplayable()
export default class SqlX extends SelfServeBaseClass {

.
.
.

  public onSave = async (
      currentValues: Map<string, SmartUiInput>,
      baselineValues: Map<string, SmartUiInput>
  ): Promise<OnSaveResult> => {

    stringPropertyValue = currentValues.get("stringProperty")
    booleanPropertyValue = currentValues.get("booleanProperty")
    
    const propertiesToSave : RpDataModel = { 
      stringProperty: stringPropertyValue,
      booleanProperty: booleanPropertyValue
    }

    const operationStatusUrl = await saveData(propertiesToSave);
    return {
      operationStatusUrl: operationStatusUrl,
      portalNotification: {
        initialize: {
          titleTKey: "DeleteInitializeTitle",
          messageTKey: "DeleteInitializeMessage",
        },
        success: {
          titleTKey: "DeleteSuccessTitle",
          messageTKey: "DeleteSuccesseMessage",
        },
        failure: {
          titleTKey: "DeleteFailureTitle",
          messageTKey: "DeleteFailureMessage",
        },
      },
    };
  }

  .
  .
  .

  @Values(...)
  stringProperty: string;

  @Values(...)
  booleanProperty: boolean;
}

Execution

Watch mode

Run npm start to start the development server and automatically rebuild on changes

Local Development

Ensure that you have made the Code changes.

  • Go to https://ms.portal.azure.com/
  • Add the query string feature.showSelfServeExample=true&feature.selfServeSource=https://localhost:1234/selfServe.html?selfServeType%3D<SELF_SERVE_TYPE>
  • Click on the Self Serve Example menu item on the left panel.

For example, if you want to open up the the UI of a class with the type sqlx, then visit https://ms.portal.azure.com/?feature.showSelfServeExample=true&feature.selfServeSource=https://localhost:1234/selfServe.html?selfServeType%3Dsqlx

Generated using TypeDoc