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

Draft: Create Network driver instead of directly using sockets. #330

Draft
wants to merge 289 commits into
base: main
Choose a base branch
from

Conversation

Jakio815
Copy link
Collaborator

@Jakio815 Jakio815 commented Jan 15, 2024

Open problems

[] OPENSSL_init_crypto(OPENSSL_INIT_NO_ATEXIT, NULL); - the "atexit" problem.

Netdriver

Current LF code of the reactor-c uses TCP and UDP sockets directly, limiting to scale network options. This PR introduces a network driver to support various network types.

typedef struct netdrv_t {
  void* priv;
  unsigned int read_remaining_bytes;
  int federate_id;
} netdrv_t;

The void* priv will be casted depending on the network type.

APIs.

The main design is as a client and server. The RTI is always a server. Federates can be servers and clients. As a client, it should be able to connect to the RTI, and also connect to other federates. As a server, it should be able to accept connections from other federates for two reasons: decentralized coordination and physical connections.
So, the main design follows TCP socket server-client design.
The RTI will have a netdriver that accepts other federate connections, and a netdriver will be allocated for each federate.
The federate will have a netdriver to connect to the RTI as a client.
Next, a netdriver to accept other federate's connection as a server.
Finally netdrivers to communicate with each inbound and outbound p2p connections.

The description below separates the APIs as functions for clients, servers, and common each.

Common

netdrv_t* initialize_netdrv();

Initializes netdriver, and allocates memory.

void close_netdrv(netdrv_t* drv);

Frees memory allocated by netdriver.

int write_to_netdrv(netdrv_t* drv, size_t num_bytes, unsigned char* buffer);

This sends the buffer to the netdriver.
TCP: write()
MQTT: publish()
SST: encapsulate message with SST, and send.

The below two each has additional error handling.
int write_to_netdrv_close_on_error(netdrv_t* drv, size_t num_bytes, unsigned char* buffer);
void write_to_netdrv_fail_on_error(netdrv_t* drv, size_t num_bytes, unsigned char* buffer, lf_mutex_t* mutex,char* format, ...);

ssize_t read_from_netdrv(netdrv_t* drv, unsigned char* buffer, size_t buffer_length);

ssize_t read_from_netdrv_close_on_error(netdrv_t* drv, unsigned char* buffer, size_t buffer_length);
void read_from_netdrv_fail_on_error(netdrv_t* drv, unsigned char* buffer, size_t buffer_length, lf_mutex_t* mutex, char* format, ...);

Server

int create_server(netdrv_t* drv, int server_type, uint16_t port);

TCP: creates server socket, binds, and listens to server socket.
MQTT: Connects to broker. Subscribe to Topic “{Federation_ID}_RTI”
SST: creates server socket, init_SST(): loads keys

netdrv_t* establish_communication_session(netdrv_t* netdrv);

TCP: accept() for federate socket requests, return fed_socket.
Mqtt:

  1. (RTI is already subscribing to “{Federation_ID}_RTI”)
  2. (Each federate publishes fed_id to “{Federation_ID}_RTI”)
  3. RTI subscribes to “{Federation_ID}_fed{fed_id}_to_RTI“
  4. (Federate subscribes to ”{Federation_ID}_RTI_to_fed{fed_id}”)
  • Now there are two one-way channels for each RTI-fed connection.

SST: accept() sockets. RTI does handshake with federates. SST_session_ctx is created for each federate.

Client

int create_client(netdrv_t* drv, int federate_id);

TCP: Creates a real-time TCP socket.
MQTT: Federate connects to broker.
SST: init_SST().

int connect_to_netdrv(netdrv_t* drv);

TCP: connect()
MQTT: Federate publishes to “{Federation_ID}_RTI”, then subscribes to ”{Federation_ID}_RTI_to_fed{fed_id}”
SST: Requests session key from Auth, and request connection to RTI or federate server.

Major changes

MQTT

_RTI _RTI_to_fedID _fedID_to_RTI _fedServID _fedServID_to_fedCliID _fedCliID_to_fedServID
RTI rti_netdrv sub
fed_netdrv pub sub
Fed rti_netdrv pub sub pub
serv_netdrv sub pub sub
cli_netdrv pub sub pub

SST

Netdriver supports SST.

reactor C {
  comm-type:SST
}

RTI must be built with -DCOMM_TYPE=SST.

Instructions

  1. Install sst-c-api
    OpenSSL 3.0 above must be installed.
$ git clone https://github.com/iotauth/sst-c-api.git
$ cd sst-c-api && mkdir build && cd build
$ cmake ../
$ make
$ sudo make install
  1. Generate keys and execute Auth.
    Java 11 or above must be installed.
$ git clone https://github.com/iotauth/iotauth.git
$ cd iotauth/examples
$ ./generateAll.sh -g configs/lf_sst.graph 
$ cd ../auth/auth-server/
$ java -jar target/auth-server-jar-with-dependencies.jar -p ../properties/exampleAuth101.properties

The path must be in the same directory with lingua-franca for the script to work. This is for key paths, and auto generated .config files.

.
├── iotauth
├── lingua-franca
  1. Install RTI with -DCOMM_TYPE=SST.
  2. Add comm-type:SST to .lf code.

Input for configuration is required.

Each federate binaries and the RTI needs a config as an input with the -sst or --sst option.

The code generator copies the skeleton config from reactor-c/core/federated/network/SSTskeleton.config.
It writes three components.

  • entityInfo.name
  • authInfo.pubkey.path
  • entityInfo.privkey.path

The generated federate's sst config will be saved such as test/C/fed-gen/SimpleFederated/src-gen/federate__fed1/core/federated/network/federate__fed1.config

The RTI's config will be saved such as test/C/fed-gen/SimpleFederated/src-gen/rti.config. However, this RTI config is only for local environments.

./federate__fed1 -sst federate__fed1.config
./RTI -sst rti.config

Refactoring

federate.c

lf_create_server():
Separate the part sending the address_advertisement message to the RTI. The implementation is in each lf_STACK_support.c. In TCP connections, it will send the port as it did. In MQTT connections, it will do nothing, because it does not need the RTI to connect to another federate. (The MQTT will do a handshake using the federateID, which is already known).

lf_connect_to_federate():
This is a very large function having three parts.
1-1. Send the RTI an ADDRESS_QUERY message, sending the remote federate's ID which it wants to connect to.
1-2. Receive ADDRESS_QUERY_REPLY message from RTI, getting the port number and IP address of the requested federate.
2. Connect to the remote federate, using the port and IP address.
3-1. Send my federateID and federationID
3-2. Receive an ACK.

I splitted this into three parts, using helper functions. However, part 1, sending address query messages, will have different implementations depending on the network communication type. So, it is defined in netdriver.h, and implemented in lf_STACK_support.c

Maximum length of Federation ID is needed.

Change response message of RTI to federate MSG_TYPE_ADDRESS_QUERY

** THIS IS ALREADY FIXED IN #392
The federate first sends MSG_TYPE_ADDRESS_QUERY with total 5bytes (1 Header + 2 federate id).
Then, the RTI responds with the same header MSG_TYPE_ADDRESS_QUERY with (1 Header + 4 port_num + 4 IP address).

Due to my approach needing unique message headers, the response of the RTI will be MSG_TYPE_ADDRESS_QUERY_REPLY.

Copy link
Contributor

@edwardalee edwardalee left a comment

Choose a reason for hiding this comment

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

I realize this is just an early draft, but this looks like a good start, working in the right direction. A couple of suggestions that may save time later:

  1. You may want to base your changes on the federated-cleanup branch of reactor-c, which will be merged soon, I think. It includes quite a few changes to the networking code.
  2. Our coding style uses spaces only for indentation, no tabs.
  3. Our coding style puts opening brackets on the same line and always encloses if statements in brackets when a new line is involved. E.g., instead of:
int  netdrv_open(netdrv_t *drv)
{
	if (!drv)
		return -1;
	return drv->open(drv);
}

it should be

int  netdrv_open(netdrv_t *drv) {
    if (!drv) {
        return -1;
    }
    return drv->open(drv);
}

or

int  netdrv_open(netdrv_t *drv) {
    if (!drv) return -1;
    return drv->open(drv);
}
  1. Header files should always document functions using JavaDoc style documentation.

@Jakio815
Copy link
Collaborator Author

@edwardalee Thank you for having a look. I'll keep in my what you mentioned. Especially for the indents and coding style, it was just the auto formatted. I'll fix it. Also I'll add up the documentation for header files. Thank you.

Copy link
Contributor

coderabbitai bot commented Jun 21, 2024

Important

Review skipped

Draft detected.

Please check the settings in the CodeRabbit UI or the .coderabbit.yaml file in this repository. To trigger a single review, invoke the @coderabbitai review command.

You can disable this status message by setting the reviews.review_status to false in the CodeRabbit configuration file.


Thank you for using CodeRabbit. We offer it for free to the OSS community and would appreciate your support in helping us grow. If you find it useful, would you consider giving us a shout-out on your favorite social media?

Share
Tips

Chat

There are 3 ways to chat with CodeRabbit:

  • Review comments: Directly reply to a review comment made by CodeRabbit. Example:
    • I pushed a fix in commit <commit_id>.
    • Generate unit testing code for this file.
    • Open a follow-up GitHub issue for this discussion.
  • Files and specific lines of code (under the "Files changed" tab): Tag @coderabbitai in a new review comment at the desired location with your query. Examples:
    • @coderabbitai generate unit testing code for this file.
    • @coderabbitai modularize this function.
  • PR comments: Tag @coderabbitai in a new PR comment to ask questions about the PR branch. For the best results, please provide a very specific query, as very limited context is provided in this mode. Examples:
    • @coderabbitai generate interesting stats about this repository and render them as a table.
    • @coderabbitai show all the console.log statements in this repository.
    • @coderabbitai read src/utils.ts and generate unit testing code.
    • @coderabbitai read the files in the src/scheduler package and generate a class diagram using mermaid and a README in the markdown format.
    • @coderabbitai help me debug CodeRabbit configuration file.

Note: Be mindful of the bot's finite context window. It's strongly recommended to break down tasks such as reading entire modules into smaller chunks. For a focused discussion, use review comments to chat about specific files and their changes, instead of using the PR comments.

CodeRabbit Commands (invoked as PR comments)

  • @coderabbitai pause to pause the reviews on a PR.
  • @coderabbitai resume to resume the paused reviews.
  • @coderabbitai review to trigger an incremental review. This is useful when automatic reviews are disabled for the repository.
  • @coderabbitai full review to do a full review from scratch and review all the files again.
  • @coderabbitai summary to regenerate the summary of the PR.
  • @coderabbitai resolve resolve all the CodeRabbit review comments.
  • @coderabbitai configuration to show the current CodeRabbit configuration for the repository.
  • @coderabbitai help to get help.

Additionally, you can add @coderabbitai ignore anywhere in the PR description to prevent this PR from being reviewed.

CodeRabbit Configration File (.coderabbit.yaml)

  • You can programmatically configure CodeRabbit by adding a .coderabbit.yaml file to the root of your repository.
  • Please see the configuration documentation for more information.
  • If your editor has YAML language server enabled, you can add the path at the top of this file to enable auto-completion and validation: # yaml-language-server: $schema=https://coderabbit.ai/integrations/schema.v2.json

Documentation and Community

  • Visit our Documentation for detailed information on how to use CodeRabbit.
  • Join our Discord Community to get help, request features, and share feedback.
  • Follow us on X/Twitter for updates and announcements.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

5 participants