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(iotevents): allow setting description, evaluation method and key of DetectorModel #18644

Merged
merged 4 commits into from
Jan 27, 2022
Merged
Show file tree
Hide file tree
Changes from 1 commit
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
5 changes: 4 additions & 1 deletion packages/@aws-cdk/aws-iotevents/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -51,7 +51,7 @@ import * as iotevents from '@aws-cdk/aws-iotevents';

const input = new iotevents.Input(this, 'MyInput', {
inputName: 'my_input', // optional
attributeJsonPaths: ['payload.temperature'],
attributeJsonPaths: ['payload.deviceId', 'payload.temperature'],
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why this change - seems unrelated to the rest of this PR...?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I modified this for the sake of reality.
The path passed in detectorKey must be declared in this attributeJsonPaths. And the value passed in detectorKey is often the unique value of the device or device group.
This is because IoT Events is often used to monitor devices and device groups.

I modified integ test for the same reason.

});

const onlineState = new iotevents.State({
Expand All @@ -64,6 +64,9 @@ const onlineState = new iotevents.State({

new iotevents.DetectorModel(this, 'MyDetectorModel', {
detectorModelName: 'test-detector-model', // optional
detectorModelDescription: 'test-detector-model-description', // optional property, default is none
evaluationMethod: iotevents.EvaluationMethod.SERIAL, // optional property, default is iotevents.EvaluationMethod.BATCH
key: 'payload.deviceId', // optional property, default is none and single detector instance will be created and all inputs will be routed to it
initialState: onlineState,
});
```
51 changes: 51 additions & 0 deletions packages/@aws-cdk/aws-iotevents/lib/detector-model.ts
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,22 @@ export interface IDetectorModel extends IResource {
readonly detectorModelName: string;
}

/**
* Information about the order in which events are evaluated and how actions are executed.
*/
export enum EvaluationMethod {
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Is this name too generic? Should this be EventEvaluationMethod, or StateEvaluationMethod, or DetectorActionEvaluationMethod, or something?

Actually, can we even kill the "Method" suffix here? I don't think it adds much, and just makes the name longer.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

EventEvaluation is good I think!

/**
* When setting to SERIAL, variables are updated and event conditions are evaluated in the order
* that the events are defined.
*/
BATCH = 'BATCH',
yamatatsu marked this conversation as resolved.
Show resolved Hide resolved
/**
* When setting to BATCH, variables within a state are updated and events within a state are
* performed only after all event conditions are evaluated.
*/
SERIAL = 'SERIAL',
}

/**
* Properties for defining an AWS IoT Events detector model
*/
Expand All @@ -27,6 +43,38 @@ export interface DetectorModelProps {
*/
readonly detectorModelName?: string;

/**
* A brief description of the detector model.
*
* @default none
*/
readonly detectorModelDescription?: string;
yamatatsu marked this conversation as resolved.
Show resolved Hide resolved

/**
* Information about the order in which events are evaluated and how actions are executed.
*
* When setting to SERIAL, variables are updated and event conditions are evaluated in the order
* that the events are defined.
* When setting to BATCH, variables within a state are updated and events within a state are
* performed only after all event conditions are evaluated.
*
* @default EvaluationMethod.BATCH
*/
readonly evaluationMethod?: EvaluationMethod;

/**
* The value used to identify a detector instance. When a device or system sends input, a new
* detector instance with a unique key value is created. AWS IoT Events can continue to route
* input to its corresponding detector instance based on this identifying information.
*
* This parameter uses a JSON-path expression to select the attribute-value pair in the message
* payload that is used for identification. To route the message to the correct detector instance,
* the device must send a message payload that contains the same attribute-value.
*
* @default - none (single detector instance will be created and all inputs will be routed to it)
*/
readonly key?: string;
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This name, however, is way, way too generic. How about calling it detectorKey, or something like that?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

detectorKey is good! Hmm... I should think for more understandability instead of CFn copy. Thank you!


/**
* The state that is entered at the creation of each detector.
*/
Expand Down Expand Up @@ -70,6 +118,9 @@ export class DetectorModel extends Resource implements IDetectorModel {

const resource = new CfnDetectorModel(this, 'Resource', {
detectorModelName: this.physicalName,
detectorModelDescription: props.detectorModelDescription,
evaluationMethod: props.evaluationMethod,
key: props.key,
detectorModelDefinition: {
initialStateName: props.initialState.stateName,
states: [props.initialState._toStateJson()],
Expand Down
59 changes: 59 additions & 0 deletions packages/@aws-cdk/aws-iotevents/test/detector-model.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -95,6 +95,65 @@ test('can set physical name', () => {
});
});

test('can set description', () => {
// WHEN
new iotevents.DetectorModel(stack, 'MyDetectorModel', {
detectorModelDescription: 'test-detector-model-description',
initialState: new iotevents.State({
stateName: 'test-state',
onEnter: [{
eventName: 'test-eventName',
condition: iotevents.Expression.fromString('test-eventCondition'),
}],
}),
});

// THEN
Template.fromStack(stack).hasResourceProperties('AWS::IoTEvents::DetectorModel', {
DetectorModelDescription: 'test-detector-model-description',
});
});

test('can set evaluationMethod', () => {
// WHEN
new iotevents.DetectorModel(stack, 'MyDetectorModel', {
detectorModelDescription: 'test-detector-model-description',
evaluationMethod: iotevents.EvaluationMethod.SERIAL,
initialState: new iotevents.State({
stateName: 'test-state',
onEnter: [{
eventName: 'test-eventName',
condition: iotevents.Expression.fromString('test-eventCondition'),
}],
}),
});

// THEN
Template.fromStack(stack).hasResourceProperties('AWS::IoTEvents::DetectorModel', {
EvaluationMethod: 'SERIAL',
});
});

test('can set key', () => {
// WHEN
new iotevents.DetectorModel(stack, 'MyDetectorModel', {
detectorModelDescription: 'test-detector-model-description',
key: 'payload.deviceId',
initialState: new iotevents.State({
stateName: 'test-state',
onEnter: [{
eventName: 'test-eventName',
condition: iotevents.Expression.fromString('test-eventCondition'),
}],
}),
});

// THEN
Template.fromStack(stack).hasResourceProperties('AWS::IoTEvents::DetectorModel', {
Key: 'payload.deviceId',
});
});
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Let's just do all 3 in one test. Spreading them out like that just makes the tests longer, it doesn't really improve coverage in any way.


test('can set multiple events to State', () => {
// WHEN
new iotevents.DetectorModel(stack, 'MyDetectorModel', {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -5,6 +5,9 @@
"Properties": {
"InputDefinition": {
"Attributes": [
{
"JsonPath": "payload.deviceId"
},
{
"JsonPath": "payload.temperature"
}
Expand Down Expand Up @@ -70,7 +73,10 @@
"Arn"
]
},
"DetectorModelName": "test-detector-model"
"DetectorModelDescription": "test-detector-model-description",
"DetectorModelName": "test-detector-model",
"EvaluationMethod": "SERIAL",
"Key": "payload.deviceId"
}
}
}
Expand Down
5 changes: 4 additions & 1 deletion packages/@aws-cdk/aws-iotevents/test/integ.detector-model.ts
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@ class TestStack extends cdk.Stack {

const input = new iotevents.Input(this, 'MyInput', {
inputName: 'test_input',
attributeJsonPaths: ['payload.temperature'],
attributeJsonPaths: ['payload.deviceId', 'payload.temperature'],
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Again, why are we changing existing things?

});

const onlineState = new iotevents.State({
Expand All @@ -27,6 +27,9 @@ class TestStack extends cdk.Stack {

new iotevents.DetectorModel(this, 'MyDetectorModel', {
detectorModelName: 'test-detector-model',
detectorModelDescription: 'test-detector-model-description',
evaluationMethod: iotevents.EvaluationMethod.SERIAL,
key: 'payload.deviceId',
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can't we use payload.temperature here?

initialState: onlineState,
});
}
Expand Down