-
Notifications
You must be signed in to change notification settings - Fork 78
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Along with the IOT2050 SM variant, some IOT2050 device specific events are recognized as being important for field applications, including: - Device power up - Device power loss - Device case open (uncover) - Device tilting - Device watchdog reset - External module events Although it's possible to collect all these events from variant source, such as syslog, or by customized coding to get the sensor event, it's valuable to provide an unified method or portal to collect all these predefined events from one place. Then this service is added to serve the above purpose, it reads the power up, power loss, eio, tilt, uncover, and (possibly) the watchdog reset events, then writes them into syslog and makes them be readable from gRPC interface. This service could be used directly, or as a base/reference for customization. Signed-off-by: Li Hua Qian <[email protected]>
- Loading branch information
1 parent
0863ae7
commit e78c39f
Showing
16 changed files
with
831 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,94 @@ | ||
# IOT2050 Event Record | ||
|
||
IOT2050 Event Record is using for reading and recording events, such as | ||
power up, power loss, tilted, uncovered, watchdog reset and eio events. | ||
|
||
The core is a RPC service implemented with the help of gPRC. | ||
|
||
## Event record services | ||
|
||
The `iot2050-event-record.service` and `iot2050-event-serve.service` are systemd | ||
services, they could be managed by `systemctl`. The `iot2050-event-record` collects events from various source then consume the API exposed by `iot2050-event-serve` | ||
to wrap the collected events as `IOT2050-EventRecord` events and save them to | ||
syslog. Then these wrapped events could be read by `journalctl` or by gRPC APIs. | ||
|
||
## Predefined events | ||
|
||
### Power events and Extended IO(EIO) events | ||
|
||
Power events and EIO events are injected to `journal(syslog)` by default. | ||
|
||
To check them on IOT2050: | ||
|
||
```sh | ||
root@iot2050-debian:~# journalctl SYSLOG_IDENTIFIER=IOT2050-EventRecord | ||
Oct 23 22:36:12 iot2050-debian IOT2050-EventRecord[323]: IOT2050_EVENTS.power: 2023-10-23 22:36:00 the device is powered up | ||
Oct 23 22:40:21 iot2050-debian IOT2050-EventRecord[323]: IOT2050_EVENTS.power: 2023-10-23 22:34:4 [2] power loss | ||
Oct 23 22:40:21 iot2050-debian IOT2050-EventRecord[323]: IOT2050_EVENTS.eio: 2023-10-23 22:12:54 [11] slot1 lost | ||
``` | ||
|
||
### Sensor events | ||
|
||
By default, the sensor events, i.e. tilted and uncovered events, are disabled. | ||
To enable them, please create a systemd drop-in for `iot2050-event-record.service`, | ||
as follows: | ||
|
||
```sh | ||
cp /usr/lib/iot2050/event/iot2050-event-record.conf /etc/systemd/system/iot2050-event-record.service.d/ | ||
``` | ||
|
||
### Watchdog events | ||
|
||
If watchdog event recording is expected, please refer to [WATCHDOG.md](./WATCHDOG.md). | ||
|
||
## Development | ||
|
||
### How to inject a new event? | ||
|
||
First, please link the `EventInterface` to the customized application. | ||
```sh | ||
# In IOT DUT | ||
ln -s /usr/lib/iot2050/event/gRPC/EventInterface /path/to/customized-app/gRPC/EventInterface | ||
|
||
# In Source code | ||
ln -s recipes-app/iot2050-event-record/files/gRPC/EventInterface /path/to/customized-app/gRPC/EventInterface | ||
``` | ||
|
||
Then, use the `Write` and `Read` functions to communicate with | ||
`iot2050-event-serve.service`, as follows: | ||
|
||
```python | ||
import grpc | ||
from gRPC.EventInterface.iot2050_event_pb2 import ( | ||
WriteRequest, | ||
ReadRequest | ||
) | ||
from gRPC.EventInterface.iot2050_event_pb2_grpc import EventRecordStub | ||
|
||
|
||
def write_event(event_type, event): | ||
with grpc.insecure_channel(iot2050_event_api_server) as channel: | ||
stub = EventRecordStub(channel) | ||
response = stub.Write(WriteRequest(event_type=event_type, event=event)) | ||
|
||
if response.status: | ||
print(f'Event Record writes event result: {response.status}') | ||
print(f'Event Record writes event message: {response.message}') | ||
|
||
def read_event(event_type): | ||
with grpc.insecure_channel(iot2050_event_api_server) as channel: | ||
stub = EventRecordStub(channel) | ||
response = stub.Read(ReadRequest(event_type=event_type)) | ||
return response.event | ||
``` | ||
|
||
And, please find the api definition in `gRPC/EventInterface/iot2050-event.proto`. | ||
|
||
### Regenerate the gRPC python modules if proto file changed | ||
|
||
If the `proto` file needs to be changed when customizing a new application | ||
to inject event, please update the gRPC in the original path as follows. | ||
|
||
```sh | ||
python3 -m grpc_tools.protoc -I. --python_out=. --pyi_out=. --grpc_python_out=. gRPC/EventInterface/iot2050-event.proto | ||
``` |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,38 @@ | ||
This README file explains how to get the watchdog reset status and how to | ||
inject it into `iot2050-event-record` service. | ||
|
||
# How to get the watchdog reset status? | ||
|
||
The `wdt_example.py` below shows how to get the watchdog reset status. | ||
|
||
```py | ||
import psutil | ||
import time | ||
from datetime import datetime | ||
|
||
WDIOF_CARDRESET = "32" | ||
WDT_PATH = "/sys/class/watchdog/watchdog0/bootstatus" | ||
|
||
EVENT_STRINGS = { | ||
"wdt": "{} watchdog reset is detected", | ||
"no-wdt": "{} watchdog reset isn't detected" | ||
} | ||
|
||
def record_wdt_events(): | ||
status = "" | ||
with open(WDT_PATH, "r") as f: | ||
status = f.read().strip() | ||
|
||
boot_time = datetime.fromtimestamp(psutil.boot_time()) | ||
if WDIOF_CARDRESET == status: | ||
print(EVENT_STRINGS["wdt"].format(boot_time)) | ||
else: | ||
print(EVENT_STRINGS["no-wdt"].format(boot_time)) | ||
|
||
if __name__ == "__main__": | ||
record_wdt_events() | ||
``` | ||
|
||
# How to inject it into iot2050-event-record? | ||
|
||
Please refer to [README.md](./README.md). |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1 @@ | ||
../../../iot2050-eio-manager/files/gRPC/EIOManager |
61 changes: 61 additions & 0 deletions
61
recipes-app/iot2050-event-record/files/gRPC/EventInterface/iot2050-event.proto
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,61 @@ | ||
/* | ||
* Copyright (c) Siemens AG, 2023 | ||
* | ||
* Authors: | ||
* Li Hua Qian <[email protected]> | ||
* | ||
* SPDX-License-Identifier: MIT | ||
*/ | ||
|
||
syntax = "proto3"; | ||
|
||
package eventrecord; | ||
|
||
service EventRecord { | ||
rpc Write (WriteRequest) returns (WriteReply) {} | ||
rpc Read (ReadRequest) returns (ReadReply) {} | ||
} | ||
|
||
/* ----------------- Write event ----------------- */ | ||
/* WriteRequest | ||
* - event_type: a string to present event type | ||
* - "IOT2050_EVENT.xxx" means IOT2050 standard events | ||
* - "" or other strings mean customized events | ||
* - event: the event content to write | ||
*/ | ||
message WriteRequest { | ||
string event_type = 1; | ||
string event = 2; | ||
} | ||
|
||
/* WriteReply | ||
* - status: 0 means successful | ||
* others mean error | ||
* - message: the detail write message | ||
*/ | ||
message WriteReply { | ||
int32 status = 1; | ||
string message = 2; | ||
} | ||
|
||
/* ----------------- Read event ----------------- */ | ||
/* ReadRequest | ||
* - event_type: a string to present event type | ||
* - "IOT2050_EVENT.xxx" means IOT2050 standard events | ||
* - "" means to all types of events | ||
*/ | ||
message ReadRequest { | ||
string event_type = 1; | ||
} | ||
|
||
/* ReadReply | ||
* - status: 0 means successful | ||
* others mean error | ||
* - message: the detail write message | ||
* - event: the read back event content | ||
*/ | ||
message ReadReply { | ||
int32 status = 1; | ||
string message = 2; | ||
string event = 3; | ||
} |
33 changes: 33 additions & 0 deletions
33
recipes-app/iot2050-event-record/files/gRPC/EventInterface/iot2050_event_pb2.py
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Oops, something went wrong.
37 changes: 37 additions & 0 deletions
37
recipes-app/iot2050-event-record/files/gRPC/EventInterface/iot2050_event_pb2.pyi
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,37 @@ | ||
from google.protobuf import descriptor as _descriptor | ||
from google.protobuf import message as _message | ||
from typing import ClassVar as _ClassVar, Optional as _Optional | ||
|
||
DESCRIPTOR: _descriptor.FileDescriptor | ||
|
||
class WriteRequest(_message.Message): | ||
__slots__ = ["event_type", "event"] | ||
EVENT_TYPE_FIELD_NUMBER: _ClassVar[int] | ||
EVENT_FIELD_NUMBER: _ClassVar[int] | ||
event_type: str | ||
event: str | ||
def __init__(self, event_type: _Optional[str] = ..., event: _Optional[str] = ...) -> None: ... | ||
|
||
class WriteReply(_message.Message): | ||
__slots__ = ["status", "message"] | ||
STATUS_FIELD_NUMBER: _ClassVar[int] | ||
MESSAGE_FIELD_NUMBER: _ClassVar[int] | ||
status: int | ||
message: str | ||
def __init__(self, status: _Optional[int] = ..., message: _Optional[str] = ...) -> None: ... | ||
|
||
class ReadRequest(_message.Message): | ||
__slots__ = ["event_type"] | ||
EVENT_TYPE_FIELD_NUMBER: _ClassVar[int] | ||
event_type: str | ||
def __init__(self, event_type: _Optional[str] = ...) -> None: ... | ||
|
||
class ReadReply(_message.Message): | ||
__slots__ = ["status", "message", "event"] | ||
STATUS_FIELD_NUMBER: _ClassVar[int] | ||
MESSAGE_FIELD_NUMBER: _ClassVar[int] | ||
EVENT_FIELD_NUMBER: _ClassVar[int] | ||
status: int | ||
message: str | ||
event: str | ||
def __init__(self, status: _Optional[int] = ..., message: _Optional[str] = ..., event: _Optional[str] = ...) -> None: ... |
99 changes: 99 additions & 0 deletions
99
recipes-app/iot2050-event-record/files/gRPC/EventInterface/iot2050_event_pb2_grpc.py
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,99 @@ | ||
# Generated by the gRPC Python protocol compiler plugin. DO NOT EDIT! | ||
"""Client and server classes corresponding to protobuf-defined services.""" | ||
import grpc | ||
|
||
from gRPC.EventInterface import iot2050_event_pb2 as gRPC_dot_EventInterface_dot_iot2050__event__pb2 | ||
|
||
|
||
class EventRecordStub(object): | ||
"""Missing associated documentation comment in .proto file.""" | ||
|
||
def __init__(self, channel): | ||
"""Constructor. | ||
Args: | ||
channel: A grpc.Channel. | ||
""" | ||
self.Write = channel.unary_unary( | ||
'/eventrecord.EventRecord/Write', | ||
request_serializer=gRPC_dot_EventInterface_dot_iot2050__event__pb2.WriteRequest.SerializeToString, | ||
response_deserializer=gRPC_dot_EventInterface_dot_iot2050__event__pb2.WriteReply.FromString, | ||
) | ||
self.Read = channel.unary_unary( | ||
'/eventrecord.EventRecord/Read', | ||
request_serializer=gRPC_dot_EventInterface_dot_iot2050__event__pb2.ReadRequest.SerializeToString, | ||
response_deserializer=gRPC_dot_EventInterface_dot_iot2050__event__pb2.ReadReply.FromString, | ||
) | ||
|
||
|
||
class EventRecordServicer(object): | ||
"""Missing associated documentation comment in .proto file.""" | ||
|
||
def Write(self, request, context): | ||
"""Missing associated documentation comment in .proto file.""" | ||
context.set_code(grpc.StatusCode.UNIMPLEMENTED) | ||
context.set_details('Method not implemented!') | ||
raise NotImplementedError('Method not implemented!') | ||
|
||
def Read(self, request, context): | ||
"""Missing associated documentation comment in .proto file.""" | ||
context.set_code(grpc.StatusCode.UNIMPLEMENTED) | ||
context.set_details('Method not implemented!') | ||
raise NotImplementedError('Method not implemented!') | ||
|
||
|
||
def add_EventRecordServicer_to_server(servicer, server): | ||
rpc_method_handlers = { | ||
'Write': grpc.unary_unary_rpc_method_handler( | ||
servicer.Write, | ||
request_deserializer=gRPC_dot_EventInterface_dot_iot2050__event__pb2.WriteRequest.FromString, | ||
response_serializer=gRPC_dot_EventInterface_dot_iot2050__event__pb2.WriteReply.SerializeToString, | ||
), | ||
'Read': grpc.unary_unary_rpc_method_handler( | ||
servicer.Read, | ||
request_deserializer=gRPC_dot_EventInterface_dot_iot2050__event__pb2.ReadRequest.FromString, | ||
response_serializer=gRPC_dot_EventInterface_dot_iot2050__event__pb2.ReadReply.SerializeToString, | ||
), | ||
} | ||
generic_handler = grpc.method_handlers_generic_handler( | ||
'eventrecord.EventRecord', rpc_method_handlers) | ||
server.add_generic_rpc_handlers((generic_handler,)) | ||
|
||
|
||
# This class is part of an EXPERIMENTAL API. | ||
class EventRecord(object): | ||
"""Missing associated documentation comment in .proto file.""" | ||
|
||
@staticmethod | ||
def Write(request, | ||
target, | ||
options=(), | ||
channel_credentials=None, | ||
call_credentials=None, | ||
insecure=False, | ||
compression=None, | ||
wait_for_ready=None, | ||
timeout=None, | ||
metadata=None): | ||
return grpc.experimental.unary_unary(request, target, '/eventrecord.EventRecord/Write', | ||
gRPC_dot_EventInterface_dot_iot2050__event__pb2.WriteRequest.SerializeToString, | ||
gRPC_dot_EventInterface_dot_iot2050__event__pb2.WriteReply.FromString, | ||
options, channel_credentials, | ||
insecure, call_credentials, compression, wait_for_ready, timeout, metadata) | ||
|
||
@staticmethod | ||
def Read(request, | ||
target, | ||
options=(), | ||
channel_credentials=None, | ||
call_credentials=None, | ||
insecure=False, | ||
compression=None, | ||
wait_for_ready=None, | ||
timeout=None, | ||
metadata=None): | ||
return grpc.experimental.unary_unary(request, target, '/eventrecord.EventRecord/Read', | ||
gRPC_dot_EventInterface_dot_iot2050__event__pb2.ReadRequest.SerializeToString, | ||
gRPC_dot_EventInterface_dot_iot2050__event__pb2.ReadReply.FromString, | ||
options, channel_credentials, | ||
insecure, call_credentials, compression, wait_for_ready, timeout, metadata) |
6 changes: 6 additions & 0 deletions
6
recipes-app/iot2050-event-record/files/iot2050-event-record.conf
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,6 @@ | ||
# In IoT2050-SM, there are sensors for tilted and uncovered detection. If you | ||
# want to enable logging of sensor events, i.e. tilted and uncovered events. | ||
# Please copy this file to /etc/systemd/system/iot2050-event-record.service.d/. | ||
|
||
[Service] | ||
Environment="RECORD_SENSOR_EVENTS=True" |
Oops, something went wrong.