The MessageStorage
component is a passive F' component responsible for storing and loading messages of type
SpacePost
on/from the satellite's file system.
Other components which want to use MessageStorage
to store a message need to call its ports.
storeMessage
: Stores a single given message.loadMessageLastN
: Loads a given number of the most recently stored messages. I.e., messages are handled in last-in-first-out order. The loaded messages are returned in a batch which is defined as a type.loadMessageFromIndex
: Loads a single message from a provided index. The index is an identifier number internal to the component. This port is only useful if the user knows what index they are looking for, e.g. from an event or telemetry data emitted by the component.
The component emits an event every time
- it successfully stores a message (
MESSAGE_STORE_COMPLETE
), - it fails to store a message (
MESSAGE_STORE_FAILED
), - it successfully loads a message (
MESSAGE_LOAD_COMPLETE
), - it fails to load a message (
MESSAGE_LOAD_FAILED
).
The name in the brackets is the type of the emitted event. For every cause of an event, a different type is used.
Furthermore, the component emits additional events and telemetry upon the success or failure of some internal operations. For a full definition, refer to MessageStorage.fpp.
This component depends on the Operating System Abstraction Layer (OSAL) of F'. More specifically, it uses OS::File
,
OS::Directory
, and OS::FileSystem
which means that these three classes must be implemented for the underlying
Operating System.
Challenge
The component should store each message in a separate file. The component can assume that it is given access to a storage directory on the file system to store these files.
The component should not crash even if
- files other than the ones written by the component are (accidentally) put into the storage directory
- files the component wrote to the storage directory are corrupted. This is a plausible scenario as radiation can cause bit errors on the storage device.
Resulting Design Decision
Every message is stored with a unique file name in the storage directory (see Indexing).
Each message file follows the following format consisting of the following.
- Delimiter: A unique byte value that is expected as the first byte of every stored message file. Thus, we provide basic protection against trying to load files that do not originate from the
MessageStorage
component as message files. - Message Length: A
U32
in little-endian order that indicates how long the byte-serial representation of the message is. It helps to verify that the correct number of bytes is read and deserialized when loading the actual message from the file. - Message Content: The byte-serial representation of the message data. It contains everything needed to fully restore a message so that the message object obtained from loading is the same as the one provided for storing.
Challenge
- The component needs to give messages a file name that is different from the messages already stored.
- It also must be able to iterate the messages in reverse order of arrival to provide the capability to load the last n stored messages.
- Assume no index can be inferred from the message content.
Resulting Design Decision
The component assigns every message an index when starts an attempt to store the message. The index is a U32
and simply counted upwards every time a new message is about to be stored.
The index bijectively defines the file name. The file name format is '<id>.spacepost
' where <id>
is replaced with the index number.
The index starts counting at 1 past the last index of a message found in the storage directory upon initialization of the component. If the component does not find any messages upon initialization, the index starts at 1. Restoring the index is implemented in restoreIndexFromHighestStoredIndexFoundInDirectory()
.
For the sake of simplicity, we assume that the index counter never exceeds the maximum of U32
. If it does, it is wrapped around to 0. The assumption is realistic as U32
can count over 4 trillion messages which are multiple magnitudes more than what we expect as defined in the mission success criteria.
Challenge
At the time this component was developed, the decision of what content will make up a message was still pending. Furthermore, this decision might change in the future.
Resulting Design Decision
The component encapsulates a message in the type SpacePost
.
To make the component as independent of this type as possible, the component uses knowledge about this type only in a few very consciously chosen locations:
- The component receives
SpacePost
s via its ports for communication with other components. Thus, the handler methods for port invocations know that the concrete type of a message isSpacePost
. - In all other places of the component, the concrete type
SpacePost
is hidden behind the abstract classSerializable
. Consequently, in all of these places, the only assumption the component makes about the messages it is supposed to store is that they can be serialized into and deserialized from a raw byte buffer. No other assumptions are needed to store it in a file.
Therefore, the handler methods receive a message as a SpacePost
but only call other methods in the component by passing them a base class pointer of type Serializable
to the message.
Challenge
To write a message to a file, the message type must be serialized into a raw byte buffer. The address of that raw buffer must then be used to make a system call to the operating system.
Resulting Design Decision
- Serialize messages and other data by calling the framework's
Serializable
interface on the type which is to be serialized - Serialize data to a buffer that is allocated on the stack to avoid dynamic memory allocation. The buffer is implemented as a local class
StackBuffer
inside theMessageStorage
component.
- The MessageStorage component has been unit tested to 100% line coverage and 91% branch coverage.
- The unit tests follow the data-driven unit test style.
- The unit tests are implemented with the GoogleTest testing library.
- Both black-box and white-box unit tests are used to make the unit tests as independent from the component as possible while still covering all internal error-handling branches.
- An object-oriented model is used to simplify the unit tests by modeling helper data and functionality in separate classes.
For a detailed report of the unit tests, refer to the unit test documentation.