Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat(measuremodels): add optional local names and unit to measure definitions #343

Merged
merged 12 commits into from
May 31, 2024
Merged
Show file tree
Hide file tree
Changes from 11 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
49 changes: 48 additions & 1 deletion doc/2/concepts/models/index.md
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@
code: false
type: page
title: Models
description: Assets and Devices Models
description: Assets, Measure and Devices Models
---

# Models
Expand Down Expand Up @@ -58,6 +58,53 @@ The API also allows to:
- list available models `device-manager/models:listDevices`
- get a model `device-manager/models:getDevices`


## Measure Model

A measure model contains the following information:

- `model`: model name
- `measure`: type of the measure
- `valuesMappings`: measurements mappings (See [Collection Mappings](https://docs.kuzzle.io/core/2/guides/main-concepts/data-storage/#collection-mappings))
- `valuesDetails`: (optional) Metadata and translations of measurements. You can use it to keep consistency on translations between your apps

It is possible to create new models on the Kuzzle IoT Platform using either:

- the API through the action `device-manager/models:writeMeasure`
- the framework with the method `deviceManager.models.registerMeasure`

**Example: declaration of a model via API**

```typescript
await sdk.query({
controller: 'device-manager/models',
action: 'writeMeasure',
body: {
type: 'light',
valuesMappings: {
light: { type: 'integer' },
},
valuesDetails: {
light: {
en: {
friendlyName: 'Light intensity',
unit: 'lux',
},
fr: {
friendlyName: 'Intensité lumineuse',
unit: 'lux',
},
},
},
},
});
```

The API also allows to:

- list registered measures `device-manager/models:listMeasures`
- get a measure model `device-manager/models:getMeasure`

## Asset Model

Unlike sensors and metrics which are available for all engine groups, asset models are specific to a particular engine group.
Expand Down
5 changes: 5 additions & 0 deletions doc/2/controllers/models/write-measure/index.md
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,10 @@ Method: POST
"valuesMappings": {
// Values mappings
},
// Optional
"valuesDetails":{
// Values details and translation
}
}
}
```
Expand All @@ -41,6 +45,7 @@ Method: POST

- `model`: Measure model name
- `valuesMappings`: Mappings of the measure values in Elasticsearch format
- `valuesDetails`: (optional) Measurement translations and units

---

Expand Down
13 changes: 13 additions & 0 deletions lib/modules/measure/measures/BatteryMeasure.ts
Original file line number Diff line number Diff line change
Expand Up @@ -8,4 +8,17 @@ export type BatteryMeasurement = {

export const batteryMeasureDefinition: MeasureDefinition = {
valuesMappings: { battery: { type: "integer" } },

valuesDetails: {
battery: {
en: {
friendlyName: "Battery level",
unit: "%",
},
fr: {
friendlyName: "Niveau de batterie",
unit: "%",
},
},
},
};
12 changes: 12 additions & 0 deletions lib/modules/measure/measures/HumidityMeasure.ts
Original file line number Diff line number Diff line change
Expand Up @@ -8,4 +8,16 @@ export type HumidityMeasurement = {

export const humidityMeasureDefinition: MeasureDefinition = {
valuesMappings: { humidity: { type: "float" } },
valuesDetails: {
humidity: {
en: {
friendlyName: "Relative humidity",
unit: "%",
},
fr: {
friendlyName: "Humidité relative",
unit: "%",
},
},
},
};
10 changes: 10 additions & 0 deletions lib/modules/measure/measures/MovementMeasure.ts
Original file line number Diff line number Diff line change
Expand Up @@ -8,4 +8,14 @@ export type MovementMeasurement = {

export const movementMeasureDefinition: MeasureDefinition = {
valuesMappings: { movement: { type: "boolean" } },
valuesDetails: {
movement: {
en: {
friendlyName: "Movement detection",
},
fr: {
friendlyName: "Détection de mouvement",
},
},
},
};
22 changes: 22 additions & 0 deletions lib/modules/measure/measures/PositionMeasure.ts
Original file line number Diff line number Diff line change
Expand Up @@ -17,4 +17,26 @@ export const positionMeasureDefinition: MeasureDefinition = {
accuracy: { type: "float" },
altitude: { type: "float" },
},
valuesDetails: {
position: {
en: {
friendlyName: "Localization",
unit: "(lat,lon)",
},
fr: {
friendlyName: "Localisation",
unit: "(lat,lon)",
},
},
altitude: {
en: {
friendlyName: "Altitude",
unit: "m",
},
fr: {
friendlyName: "Altitude",
unit: "m",
},
},
},
};
12 changes: 12 additions & 0 deletions lib/modules/measure/measures/TemperatureMeasure.ts
Original file line number Diff line number Diff line change
Expand Up @@ -8,4 +8,16 @@ export type TemperatureMeasurement = {

export const temperatureMeasureDefinition: MeasureDefinition = {
valuesMappings: { temperature: { type: "float" } },
valuesDetails: {
temperature: {
en: {
friendlyName: "Temperature",
unit: "°C",
},
fr: {
friendlyName: "Température",
unit: "°C",
},
},
},
};
39 changes: 39 additions & 0 deletions lib/modules/measure/types/MeasureDefinition.ts
Original file line number Diff line number Diff line change
@@ -1,16 +1,55 @@
import { JSONObject } from "kuzzle-sdk";

/* *
* Represents a measure information and localization
*
* @example
* {
* locales: {
Juiced66 marked this conversation as resolved.
Show resolved Hide resolved
* en:{
* name:"Temperature",
* unit:"°C"
* },fr:{
* name:"Température",
* unit:"°C"
* }
* },
* }
*/
interface MeasureLocales {
[localeString: string]: {
friendlyName: string;
unit?: string;
};
}
export interface MeasureValuesDetails {
[valueName: string]: MeasureLocales;
}
/**
* Represents a measure definition registered by the Device Manager
*
* @example
* {
* valuesMappings: { temperature: { type: 'float' } },
* valuesDetails: {
* temperature:{
* en:{
* name:"Temperature",
* unit:"°C"
* },fr:{
QuentinRousselet marked this conversation as resolved.
Show resolved Hide resolved
* name:"Température",
* unit:"°C"
* }
* }
* },
* }
*/
export interface MeasureDefinition {
/**
* Mappings for the measurement values in order to index the fields
*/
valuesMappings: JSONObject;
valuesDetails?: {
[valueName: string]: MeasureLocales;
};
}
8 changes: 7 additions & 1 deletion lib/modules/model/ModelService.ts
Original file line number Diff line number Diff line change
Expand Up @@ -36,6 +36,7 @@ import {
AskModelMeasureGet,
} from "./types/ModelEvents";
import { MappingsConflictsError } from "./MappingsConflictsError";
import { MeasureValuesDetails } from "../measure";

export class ModelService extends BaseService {
constructor(plugin: DeviceManagerPlugin) {
Expand Down Expand Up @@ -316,9 +317,14 @@ export class ModelService extends BaseService {
async writeMeasure(
type: string,
valuesMappings: JSONObject,
valuesDetails?: MeasureValuesDetails,
): Promise<KDocument<MeasureModelContent>> {
const modelContent: MeasureModelContent = {
measure: { type, valuesMappings },
measure: {
type,
valuesDetails,
valuesMappings,
},
type: "measure",
};

Expand Down
3 changes: 2 additions & 1 deletion lib/modules/model/ModelsController.ts
Original file line number Diff line number Diff line change
Expand Up @@ -154,10 +154,11 @@ export class ModelsController {
): Promise<ApiModelWriteMeasureResult> {
const type = request.getBodyString("type");
const valuesMappings = request.getBodyObject("valuesMappings");

const valuesDetails = request.getBodyObject("valuesDetails", {});
const measureModel = await this.modelService.writeMeasure(
type,
valuesMappings,
valuesDetails,
);

return measureModel;
Expand Down
6 changes: 5 additions & 1 deletion lib/modules/model/ModelsRegister.ts
Original file line number Diff line number Diff line change
Expand Up @@ -132,7 +132,11 @@ export class ModelsRegister {

registerMeasure(type: string, measureDefinition: MeasureDefinition) {
this.measureModels.push({
measure: { type, valuesMappings: measureDefinition.valuesMappings },
measure: {
type,
valuesDetails: measureDefinition.valuesDetails,
valuesMappings: measureDefinition.valuesMappings,
},
type: "measure",
});
}
Expand Down
4 changes: 4 additions & 0 deletions lib/modules/model/collections/modelsMappings.ts
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,10 @@ export const modelsMappings: CollectionMappings = {
dynamic: "false",
properties: {},
},
valuesDetails: {
dynamic: "false",
properties: {},
},
},
},

Expand Down
2 changes: 2 additions & 0 deletions lib/modules/model/types/ModelApi.ts
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ import {
MetadataGroups,
MetadataMappings,
} from "./ModelContent";
import { MeasureValuesDetails } from "../../measure/types/MeasureDefinition";

interface ModelsControllerRequest {
controller: "device-manager/models";
Expand Down Expand Up @@ -67,6 +68,7 @@ export interface ApiModelWriteMeasureRequest extends ModelsControllerRequest {
body: {
type: string;
valuesMappings: JSONObject;
valuesDetails?: MeasureValuesDetails;
};
}
export type ApiModelWriteMeasureResult = KDocument<MeasureModelContent>;
Expand Down
31 changes: 31 additions & 0 deletions tests/fixtures/rights.ts
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,14 @@ export default {
},
},
},
"default-user": {
content: {
profileIds: ["default-user"],
},
credentials: {
local: { username: "default-user", password: "password" },
},
},
},
profiles: {
"ayse-admin": {
Expand Down Expand Up @@ -70,7 +78,21 @@ export default {
},
],
},
"default-user": {
rateLimit: 0,
policies: [
{
roleId: "default-user",
},
],
optimizedPolicies: [
{
roleId: "default-user",
},
],
},
},

roles: {
tests: {
controllers: {
Expand All @@ -92,5 +114,14 @@ export default {
},
},
},
"default-user": {
controllers: {
"device-manager/assets": {
actions: {
"*": true,
},
},
},
},
},
};
2 changes: 2 additions & 0 deletions tests/hooks/engines.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
import { BaseRequest, JSONObject, Kuzzle } from "kuzzle-sdk";
import { loadSecurityDefault } from "./security";

async function createEngineIfNotExists(
sdk: Kuzzle,
Expand All @@ -24,6 +25,7 @@ async function createEngineIfNotExists(
}

export async function beforeAllCreateEngines(sdk: Kuzzle) {
await loadSecurityDefault(sdk);
await Promise.all([
createEngineIfNotExists(sdk, "engine-ayse"),
createEngineIfNotExists(sdk, "engine-kuzzle"),
Expand Down
Loading
Loading