MoonNet guide_en

MoonNet Guide


Overview

MoonNet is a lightweight, high-performance, event-driven network library based on C++. It aims to simplify the development of network applications, supporting both TCP and UDP protocols. It provides an efficient event loop mechanism for handling asynchronous I/O operations. With its modular design, MoonNet offers flexible interfaces that allow developers to quickly build high-performance network services.


Table of Contents

  1. Module Overview
  2. Classes and Interfaces
  3. Usage Examples
  4. Error Handling
  5. Frequently Asked Questions (FAQs)
  6. Conclusion

Module Overview

The core modules of MoonNet include:

  • Event Loop (eventloop): Manages the registration, deletion, and distribution of events, implementing the core Reactor model.
  • Event (event): Represents an event on a file descriptor, encapsulating event callbacks and trigger mechanisms.
  • Buffer (buffer): Handles data buffering for non-blocking I/O, implementing data caching and processing.
  • Buffered Event (bfevent): A buffer-based event handling class, encapsulating read/write buffers and callbacks.
  • UDP Event (udpevent): Handles the sending and receiving of UDP packets.
  • Timer Event (timerevent): Provides timer functionality, supporting one-time and periodic timers.
  • Signal Event (signalevent): Handles UNIX signals, integrating signal events into the event loop.
  • Acceptor (acceptor): Listens on TCP ports and accepts new connections.
  • Server (server): Encapsulates TCP and UDP server functionalities, managing connections, events, and thread pools.
  • Thread Pool (threadpool): Implement a simple thread pool with either static or dynamic modes, providing straightforward and easy-to-use interface functions.
  • Lock-Free Ring Buffer (ringbuff): Implements a lock-free circular buffer, providing simple and easy-to-use interface functions.
  • Lock-Free Task Thread (lfthread): Encapsulates ringbuff as a task queue and a thread into the lfthread class, facilitating management.
  • Lock-Free Thread Pool (lfthreadpool): Based on an architecture where each thread has one ringbuff as a task queue (encapsulated as lfthread), this implements a lock-free thread pool. It provides static or dynamic modes, features dynamic backing-off strategies for thread sleeping adjustments, and includes a rejection strategy when the task queue is full.

Note: The commented-out section labeled /** v1.0.0 **/ contains deprecated or previous versions of event handling functions. These have been superseded by the generalized functions to provide a unified approach to event management.


Classes and Interfaces

base_event

Description:

base_event is the base class for all event types, defining the basic interfaces and behaviors of an event.

Interface:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
namespace moon {

class eventloop;
class base_event {
public:
virtual ~base_event(){}
virtual eventloop* getloop() const=0;
virtual void close()=0;
virtual void disable_cb()=0;
/** v1.0.1 **/
virtual void enable_listen()=0;
virtual void del_listen()=0;
virtual void update_ep()=0;
};

}

Function Descriptions:

  • virtual eventloop* getloop() const = 0;
    Retrieves the associated event loop object.

  • virtual void close() = 0;
    Closes the event and releases related resources.

  • virtual void disable_cb() = 0;
    Disables the event’s callback functions to prevent it from being triggered again.

  • virtual void enable_listen() = 0;

    Enables event listening.

  • virtual void del_listen() = 0;

    Removes the event listening.

  • virtual void update_ep() = 0;

    Updates the event’s status in epoll.


event

Description:

The event class represents an event on a file descriptor, encapsulating the event type, callback functions, and trigger mechanisms.

Interface:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
namespace moon {

class eventloop;

// Event class
class event : public base_event {
public:
using Callback = std::function<void()>;

event(eventloop* base, int fd, uint32_t events);
~event();

int getfd() const; // Gets the file descriptor
uint32_t getevents() const; // Gets the event types being listened to
eventloop* getloop() const override;

void setcb(const Callback& rcb, const Callback& wcb, const Callback& ecb); // Sets the callback functions
void setrcb(const Callback& rcb); // Sets the read event callback
void setwcb(const Callback& wcb); // Sets the write event callback
void setecb(const Callback& ecb); // Sets the error event callback

void setrevents(const uint32_t revents); // Sets the triggered event types
void enable_events(uint32_t op); // Enables specific event types
void disable_events(uint32_t op); // Disables specific event types
void update_ep() override; // Updates the event listening
void handle_cb(); // Handles the event callback

bool readable(); // Checks if the event is readable
bool writeable(); // Checks if the event is writable
void enable_read(); // Enables read events
void disable_read(); // Disables read events
void enable_write(); // Enables write events
void disable_write(); // Disables write events
void enable_ET(); // Enables edge-triggered mode
void disable_ET(); // Disables edge-triggered mode

void reset_events(); // Resets the event types
void del_listen() override; // Removes the event listening
void enable_listen() override; // Enables event listening
void disable_cb() override; // Disables the callback functions
void close() override; // Closes the event

private:
eventloop* loop_;
int fd_;
uint32_t events_; // Events being listened to
uint32_t revents_; // Triggered events
Callback readcb_; // Read event callback
Callback writecb_; // Write event callback
Callback eventcb_; // Error event callback
};

}

Function Descriptions:

  • event(eventloop* base, int fd, uint32_t events);
    Constructor that initializes the event object.

  • ~event();
    Destructor that releases resources.

  • int getfd() const;
    Gets the file descriptor associated with the event.

  • uint32_t getevents() const;
    Gets the event types being listened to.

  • eventloop* getloop() const override;
    Retrieves the associated event loop object.

  • void setcb(const Callback& rcb, const Callback& wcb, const Callback& ecb);
    Sets the read, write, and error event callback functions.

  • void setrcb(const Callback& rcb);
    Sets the read event callback function.

  • void setwcb(const Callback& wcb);
    Sets the write event callback function.

  • void setecb(const Callback& ecb);
    Sets the error event callback function.

  • void setrevents(const uint32_t revents);
    Sets the triggered event types.

  • void enable_events(uint32_t op);
    Enables specific event types.

  • void disable_events(uint32_t op);
    Disables specific event types.

  • void update_ep() override;
    Updates the event’s status in epoll.

  • void handle_cb();
    Handles the event by calling the appropriate callback functions.

  • bool readable();
    Checks if the event is readable.

  • bool writeable();
    Checks if the event is writable.

  • void enable_read();
    Enables read event listening.

  • void disable_read();
    Disables read event listening.

  • void enable_write();
    Enables write event listening.

  • void disable_write();
    Disables write event listening.

  • void enable_ET();
    Enables edge-triggered mode.

  • void disable_ET();
    Disables edge-triggered mode.

  • void reset_events();
    Resets the event types.

  • void del_listen() override;
    Removes the event listening.

  • void enable_listen() override;
    Enables event listening.

  • void disable_cb() override;
    Disables the event’s callback functions.

  • void close() override;
    Closes the event.


eventloop

Description:

The eventloop class is the core of the event loop, implementing the Reactor model and managing the registration, deletion, and distribution of all events.

Interface:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
namespace moon {

class base_event;
class event;
class loopthread;

// Event loop class
class eventloop {
public:
using Callback = std::function<void()>;

eventloop(loopthread* base = nullptr, int timeout = -1);
~eventloop();

loopthread* getbaseloop();
int getefd() const;
int getevfd() const;
int getload() const;

// Event control functions
void add_event(event* event);
void del_event(event* event);
void mod_event(event* event);

void loop(); // Starts the event loop
void loopbreak(); // Stops the event loop
void getallev(std::list<event*>& list);

void create_eventfd(); // Creates a notification file descriptor
void read_eventfd();
void write_eventfd();

void add_pending_del(base_event* ev); // Adds an event to the pending deletion queue

private:
void updateload(int n); // Updates the load

private:
int epfd_; // epoll file descriptor
int eventfd_; // Event notification file descriptor
int timeout_; // epoll timeout
std::atomic<int> load_; // Load (number of active events)
std::atomic<bool> shutdown_; // Whether to shut down the event loop
std::list<event*> evlist_; // List of events
std::vector<epoll_event> events_; // Array of epoll events
std::vector<base_event*> delque_; // Queue of events pending deletion
loopthread* baseloop_; // Associated thread
};

}

Function Descriptions:

  • eventloop(loopthread* base = nullptr, int timeout = -1);
    Constructor that initializes the event loop object.

  • ~eventloop();
    Destructor that releases resources.

  • loopthread* getbaseloop();
    Gets the associated thread object.

  • int getefd() const;
    Gets the epoll file descriptor.

  • int getevfd() const;
    Gets the event notification file descriptor.

  • int getload() const;
    Gets the current load (number of active events) of the event loop.

  • void add_event(event* event);
    Adds an event to be listened to by epoll.

  • void del_event(event* event);
    Removes an event from epoll.

  • void mod_event(event* event);
    Modifies the event’s listening types.

  • void loop();
    Starts the event loop and handles events.

  • void loopbreak();
    Stops the event loop.

  • void getallev(std::list<event*>& list);
    Retrieves all events in the list.

  • void create_eventfd();
    Creates an event notification file descriptor.

  • void read_eventfd();
    Reads from the event notification file descriptor to handle loop termination signals.

  • void write_eventfd();
    Writes to the event notification file descriptor to notify the event loop to terminate.

  • void add_pending_del(base_event* ev);
    Adds an event to the pending deletion queue.


loopthread

Description:

The loopthread class encapsulates an event loop thread used to run an eventloop.

Interface:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
namespace moon {

class eventloop;

class loopthread {
public:
loopthread(int timeout = -1);
~loopthread();

eventloop* getloop(); // Gets the event loop object

private:
void _init_(); // Initializes the event loop

private:
eventloop* loop_;
std::thread t_;
std::mutex mx_;
std::condition_variable cv_;
int timeout_;
};

}

Function Descriptions:

  • loopthread(int timeout = -1);
    Constructor that initializes the thread and creates the event loop.

  • ~loopthread();
    Destructor that terminates the thread and releases resources.

  • eventloop* getloop();
    Gets the event loop object.

  • void _init_();
    Internal function that initializes the event loop and starts looping.


looptpool

Description:

The looptpool class manages a group of event loop threads (loopthread), implementing thread pool functionality and providing static/dynamic load balancing.

Interface:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
namespace moon {

class eventloop;
class loopthread;

class looptpool {
public:
looptpool(eventloop* base, bool dispath = false); // By default, dynamic load balancing is disabled
~looptpool();

void create_pool(int timeout = -1); // Creates the thread pool
void create_pool(int n, int timeout); // Creates the thread pool with specified number of threads
void create_pool_noadjust(int n, int timeout); // Creates the thread pool without dynamic adjustment

eventloop* ev_dispatch(); // Dispatches events to event loops in the thread pool
void delloop_dispatch(); // Deletes a sub-reactor and redistributes events
void addloop(); // Adds a new event loop thread
void adjust_task(); // Management thread task, dynamically adjusting sub-reactors
int getscale(); // Gets the average load
void enable_adjust(); // Enables dynamic load balancing
void stop(); // Stops the thread pool

// Disables copy and assignment operations
looptpool(const looptpool&) = delete;
looptpool& operator=(const looptpool&) = delete;

private:
void init_pool(int timeout = -1); // Initializes the thread pool
eventloop* getminload(); // Gets the event loop with the minimum load
int getmaxidx(); // Gets the index of the event loop with the maximum load

private:
eventloop* baseloop_; // Main event loop
std::thread manager_; // Management thread
std::vector<eventloop*> loadvec_;// List of event loops
int next_; // Round-robin index
int t_num_; // Number of threads
int timeout_; // Timeout duration
int max_tnum_; // Maximum number of threads
int min_tnum_; // Minimum number of threads
bool dispath_; // Whether dynamic dispatch is enabled
int coolsec_; // Cool-down time
int timesec_; // Dispatch interval
int scale_max_; // Maximum load scale
int scale_min_; // Minimum load scale
};

}

Function Descriptions:

  • looptpool(eventloop* base, bool dispath = false);
    Constructor that initializes the thread pool and specifies whether to enable dynamic load balancing.

  • ~looptpool();
    Destructor that releases resources.

  • void create_pool(int timeout = -1);
    Creates the thread pool with the default number of threads.

  • void create_pool(int n, int timeout);
    Creates the thread pool with a specified number of threads.

  • void create_pool_noadjust(int n, int timeout);
    Creates the thread pool without dynamic adjustment.

  • eventloop* ev_dispatch();
    Dispatches events to event loops in the thread pool.

  • void delloop_dispatch();
    Deletes the event loop with the maximum load and redistributes its events.

  • void addloop();
    Adds a new event loop thread.

  • void adjust_task();
    Management thread task for dynamically adjusting the number of threads.

  • int getscale();
    Gets the average load scale.

  • void enable_adjust();
    Enables dynamic load balancing.

  • void stop();
    Stops the thread pool and terminates all event loops.


threadpool

Description:

The threadpool class implements a general-purpose thread pool for executing arbitrary task functions.

Interface:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
namespace moon {

class threadpool {
public:
threadpool(int num);
~threadpool();

// Adds a task to the thread pool
template<typename _Fn, typename... _Args>
void add_task(_Fn&& fn, _Args&&... args);

private:
void init(); // Initializes the thread pool
void t_shutdown(); // Shuts down the thread pool
void t_task(); // Task thread entry function
void adjust_task(); // Management thread entry function

private:
std::thread adjust_thr; // Management thread
std::vector<std::thread> threads; // Worker threads
std::queue<std::function<void()>> tasks; // Task queue
std::mutex mx; // Mutex
std::condition_variable task_cv; // Condition variable
int min_thr_num; // Minimum number of threads
int max_thr_num; // Maximum number of threads
std::atomic<int> run_num; // Number of threads executing tasks
std::atomic<int> live_num; // Number of live threads
std::atomic<int> exit_num; // Number of threads to terminate
bool shutdown; // Whether the thread pool is shut down
};

}

Function Descriptions:

  • threadpool(int num);
    Constructor that initializes the thread pool with a specified minimum number of threads.

  • ~threadpool();
    Destructor that shuts down the thread pool.

  • void add_task(_Fn&& fn, _Args&&... args);
    Adds a task to the thread pool, specifying the task function and its arguments.

  • void init();
    Initializes the thread pool, creating worker and management threads.

  • void t_shutdown();
    Shuts down the thread pool, terminating all threads.

  • void t_task();
    Entry function for worker threads to execute tasks.

  • void adjust_task();
    Entry function for the management thread to dynamically adjust the number of threads.


buffer

Description:

The buffer class implements an auto-expanding buffer for handling data caching and operations in non-blocking I/O.

Interface:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
namespace moon {

class buffer {
public:
buffer();
~buffer();

void append(const char* data, size_t len); // Appends data to the buffer
size_t remove(char* data, size_t len); // Reads data from the buffer
std::string remove(size_t len); // Reads data from the buffer and returns a string
void retrieve(size_t len); // Moves the read pointer without copying data
size_t readbytes() const; // Gets the size of readable data
size_t writebytes() const; // Gets the size of writable space
const char* peek() const; // Gets the data at the current read pointer
void reset(); // Resets the buffer
ssize_t readiov(int fd, int& errnum); // Reads data from a file descriptor into the buffer

private:
void able_wirte(size_t len); // Ensures there is enough writable space

private:
std::vector<char> buffer_;
uint64_t reader_; // Read pointer position
uint64_t writer_; // Write pointer position
};

}

Function Descriptions:

  • buffer();
    Constructor that initializes the buffer.

  • ~buffer();
    Destructor that releases resources.

  • void append(const char* data, size_t len);
    Appends data to the buffer.

  • size_t remove(char* data, size_t len);
    Reads data from the buffer into a specified memory location.

  • std::string remove(size_t len);
    Reads data from the buffer and returns it as a string.

  • void retrieve(size_t len);
    Moves the read pointer forward, marking data as read without copying.

  • size_t readbytes() const;
    Gets the number of bytes available for reading.

  • size_t writebytes() const;
    Gets the number of bytes available for writing.

  • const char* peek() const;
    Gets a pointer to the data at the current read position.

  • void reset();
    Resets the buffer, clearing all data.

  • ssize_t readiov(int fd, int& errnum);
    Reads data from a file descriptor into the buffer using scatter/gather I/O.


bfevent

Description:

The bfevent class is a buffer-based event handling class, encapsulating read/write buffers and callbacks, handling data transmission for TCP connections.

Interface:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
namespace moon {

class eventloop;
class event;

class bfevent : public base_event {
public:
using RCallback = std::function<void(bfevent*)>;
using Callback = std::function<void()>;

bfevent(eventloop* base, int fd, uint32_t events);
~bfevent();

int getfd() const;
eventloop* getloop() const override;
buffer* getinbuff();
buffer* getoutbuff();
bool writeable() const;

void setcb(const RCallback& rcb, const Callback& wcb, const Callback& ecb); // Sets the callback functions
void setrcb(const RCallback& rcb);
void setwcb(const Callback& wcb);
void setecb(const Callback& ecb);
RCallback getrcb();
Callback getwcb();
Callback getecb();

void update_ep() override; // Updates the event listening
void del_listen() override; // Cancels event listening
void enable_listen() override; // Enables event listening

void sendout(const char* data, size_t len); // Sends data
void sendout(const std::string& data); // Sends data
size_t receive(char* data, size_t len); // Receives data into specified memory
std::string receive(size_t len); // Receives a specified length of data
std::string receive(); // Receives all available data

void enable_events(uint32_t op); // Enables specific events
void disable_events(uint32_t op); // Disables specific events
void enable_read(); // Enables read events
void disable_read(); // Disables read events
void enable_write(); // Enables write events
void disable_write(); // Disables write events
void enable_ET(); // Enables edge-triggered mode
void disable_ET(); // Disables edge-triggered mode
void disable_cb() override; // Disables callback functions

void close() override; // Closes the event

private:
void close_event(); // Internal function to close the event
void handle_read(); // Handles read events
void handle_write(); // Handles write events
void handle_event(); // Handles error events

private:
eventloop* loop_;
int fd_;
event* ev_;
buffer inbuff_;
buffer outbuff_;
RCallback readcb_;
Callback writecb_;
Callback eventcb_;
bool closed_;
};

}

Function Descriptions:

  • bfevent(eventloop* base, int fd, uint32_t events);
    Constructor that initializes the buffered event object.

  • ~bfevent();
    Destructor that closes the event and releases resources.

  • int getfd() const;
    Gets the file descriptor.

  • eventloop* getloop() const override;
    Retrieves the associated event loop.

  • buffer* getinbuff();
    Gets the input buffer.

  • buffer* getoutbuff();
    Gets the output buffer.

  • bool writeable() const;
    Checks if the event is writable.

  • void setcb(const RCallback& rcb, const Callback& wcb, const Callback& ecb);
    Sets the read, write, and error event callback functions.

  • void update_ep() override;
    Updates the event listening.

  • void del_listen() override;
    Cancels event listening.

  • void enable_listen() override;
    Enables event listening.

  • void sendout(const char* data, size_t len);
    Sends data.

  • void sendout(const std::string& data);
    Sends data.

  • size_t receive(char* data, size_t len);
    Receives data into specified memory.

  • std::string receive(size_t len);
    Receives a specified length of data.

  • std::string receive();
    Receives all available data.

  • void enable_events(uint32_t op);
    Enables specific events.

  • void disable_events(uint32_t op);
    Disables specific events.

  • void enable_read();
    Enables read events.

  • void disable_read();
    Disables read events.

  • void enable_write();
    Enables write events.

  • void disable_write();
    Disables write events.

  • void enable_ET();
    Enables edge-triggered mode.

  • void disable_ET();
    Disables edge-triggered mode.

  • void disable_cb() override;
    Disables callback functions.

  • void close() override;
    Closes the event.


udpevent

Description:

The udpevent class handles the transmission and reception of UDP protocol data, supporting non-blocking UDP communication.

Interface:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
cppCopy codenamespace moon {

class eventloop;
class event;

// UDP event handling class
class udpevent : public base_event {
public:
using Callback = std::function<void()>;
using RCallback = std::function<void(const sockaddr_in&, udpevent*)>;

udpevent(eventloop* base, int port);
~udpevent();

buffer* getinbuff();
eventloop* getloop() const override;
void setcb(const RCallback& rcb, const Callback& ecb);
void setrcb(const RCallback& rcb);
void setecb(const Callback& ecb);
void init_sock(int port);
/** v1.0.0 **/
//void start(); // Start listening
//void stop(); // Stop listening

/** v1.0.1 **/
void enable_listen() override; // Start listening
void del_listen() override; // Stop listening

void update_ep() override; // Update listening event
size_t receive(char* data, size_t len); // Receive data into specified memory
std::string receive(size_t len); // Receive specified length of data
std::string receive(); // Receive all readable data

void send_to(const std::string& data, const sockaddr_in& addr); // Send data to specified address

void enable_read();
void disable_read();
void enable_ET();
void disable_ET();
void disable_cb() override;
RCallback getrcb();
Callback getecb();
void close() override; // Close event

private:
void handle_receive(); // Handle receive event

private:
eventloop* loop_;
int fd_;
event* ev_;
buffer inbuff_;
RCallback receive_cb_;
Callback event_cb_;
bool started_;
};

}

Function Description:

  • udpevent(eventloop* base, int port);
    Constructor: Initializes the UDP event object with the specified event loop and port number.
  • ~udpevent();
    Destructor: Closes the event and releases resources.
  • buffer* getinbuff();
    Get Input Buffer: Retrieves the input buffer.
  • eventloop* getloop() const override;
    Get Event Loop: Retrieves the associated event loop.
  • void setcb(const RCallback& rcb, const Callback& ecb);
    Set Callbacks: Sets the receive and error event callback functions.
  • void setrcb(const RCallback& rcb);
    Set Receive Callback: Sets the receive event callback function.
  • void setecb(const Callback& ecb);
    Set Error Callback: Sets the error event callback function.
  • void init_sock(int port);
    Initialize Socket: Initializes the UDP socket.
  • void enable_listen() override;
    Enable Listening: Starts listening for UDP packets.
  • void del_listen() override;
    Disable Listening: Stops listening for UDP packets.
  • void update_ep() override;
    Update Event: Updates the listening event within the event loop.
  • size_t receive(char* data, size_t len);
    Receive Data: Receives data into the specified memory buffer.
  • std::string receive(size_t len);
    Receive Specified Length Data: Receives a specified length of data from the buffer.
  • std::string receive();
    Receive All Readable Data: Receives all readable data from the buffer.
  • void send_to(const std::string& data, const sockaddr_in& addr);
    Send Data To: Sends data to the specified address.
  • void enable_read();
    Enable Read Event: Enables the read event for the UDP socket.
  • void disable_read();
    Disable Read Event: Disables the read event for the UDP socket.
  • void enable_ET();
    Enable Edge Triggered: Enables edge-triggered behavior for the UDP socket.
  • void disable_ET();
    Disable Edge Triggered: Disables edge-triggered behavior for the UDP socket.
  • void disable_cb() override;
    Disable Callback: Disables the assigned callback functions.
  • RCallback getrcb();
    Get Receive Callback: Retrieves the currently assigned receive callback function.
  • Callback getecb();
    Get Error Callback: Retrieves the currently assigned error callback function.
  • void close() override;
    Close Event: Closes the UDP event and cleans up resources.
  • void handle_receive();
    Handle Receive Event: Processes incoming data when a receive event is triggered.

timerevent

Description:

The timerevent class implements timer functionalities, supporting both one-time and periodic timers for executing tasks at regular intervals.

Interface:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
cppCopy codenamespace moon {

class event;
class eventloop;

class timerevent : public base_event {
public:
using Callback = std::function<void()>;

timerevent(eventloop* loop, int timeout_ms, bool periodic);
~timerevent();

int getfd() const;
eventloop* getloop() const override;
void setcb(const Callback& cb);
/** v1.0.0 **/
//void start();
//void stop();
/** v1.0.1 **/
void enable_listen() override;
void del_listen() override;
void update_ep() override;

void close() override;
void disable_cb() override;
Callback getcb();

private:
void _init_();
void handle_timeout();

private:
eventloop* loop_;
event* ev_;
int fd_;
int timeout_ms_;
bool periodic_;
Callback cb_;
};

}

Function Description:

  • timerevent(eventloop* loop, int timeout_ms, bool periodic);
    Constructor: Initializes the timer event with the specified event loop, timeout in milliseconds, and a flag indicating whether the timer is periodic.
  • ~timerevent();
    Destructor: Closes the timer and releases associated resources.
  • int getfd() const;
    Get File Descriptor: Retrieves the file descriptor associated with the timer.
  • eventloop* getloop() const override;
    Get Event Loop: Retrieves the associated event loop.
  • void setcb(const Callback& cb);
    Set Callback: Assigns a callback function to be executed when the timer expires.
  • void enable_listen() override;
    Enable Listening: Starts the timer, enabling it to trigger events based on the specified timeout.
  • void del_listen() override;
    Disable Listening: Stops the timer from triggering further events.
  • void update_ep() override;
    Update Event: Updates the event’s state within the event loop, typically used after modifying timer settings.
  • void close() override;
    Close Event: Closes the timer event and cleans up resources.
  • void disable_cb() override;
    Disable Callback: Disables the assigned callback function, preventing it from being executed when the timer expires.
  • Callback getcb();
    Get Callback: Retrieves the currently assigned callback function.
  • void _init_();
    Initialize Timer: Internal method to initialize timer settings and configurations.
  • void handle_timeout();
    Handle Timeout: Internal method invoked when the timer expires, executing the assigned callback function.

signalevent

Description:

The signalevent class handles UNIX signals by integrating signal events into the event loop using a pipe mechanism.

Interface:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
cppCopy codenamespace moon {

class eventloop;
class event;

class signalevent : public base_event {
public:
using Callback = std::function<void(int)>;

signalevent(eventloop* base);
~signalevent();

eventloop* getloop() const override;
void add_signal(int signo);
void add_signal(const std::vector<int>& signals);
void setcb(const Callback& cb);

void enable_listen() override;
void del_listen() override;
void update_ep() override;
void disable_cb() override;
void close() override;
Callback getcb();

private:
static void handle_signal(int signo);
void handle_read();

private:
eventloop* loop_;
int pipe_fd_[2]; // Read and write ends of the pipe
event* ev_;
Callback cb_;
static signalevent* sigev_; // Singleton instance
};

}

Function Description:

  • signalevent(eventloop* base);
    Constructor: Initializes the signal event object with the given event loop.
  • ~signalevent();
    Destructor: Closes the event and releases resources.
  • eventloop* getloop() const override;
    Get Event Loop: Retrieves the associated event loop.
  • void add_signal(int signo);
    Add Single Signal: Adds a listener for a single signal.
  • void add_signal(const std::vector<int>& signals);
    Add Multiple Signals: Adds listeners for multiple signals.
  • void setcb(const Callback& cb);
    Set Callback: Assigns a callback function to handle signals.
  • void enable_listen() override;
    Enable Listening: Starts listening for the configured signals.
  • void del_listen() override;
    Disable Listening: Stops listening for signals.
  • void update_ep() override;
    Update Event: Updates the event’s state within the event loop.
  • void disable_cb() override;
    Disable Callback: Disables the assigned callback function.
  • void close() override;
    Close Event: Closes the signal event and cleans up resources.
  • Callback getcb();
    Get Callback: Retrieves the currently assigned callback function.
  • static void handle_signal(int signo);
    Handle Signal (Static): Static method to handle incoming signals and forward them to the appropriate instance.
  • void handle_read();
    Handle Read: Processes the signal read from the pipe.

acceptor

Description:

The acceptor class is responsible for listening on a specified port, accepting new TCP connections, and handing them off to a callback function.

Interface:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
namespace moon {

class eventloop;
class event;

// Acceptor class
class acceptor {
public:
using Callback = std::function<void(int)>;

acceptor(int port, eventloop* base);
~acceptor();

void listen(); // Starts listening
void stop(); // Stops listening
void init_sock(int port); // Initializes the listening socket
void setcb(const Callback& accept_cb); // Sets the callback function
void handle_accept(); // Handles new connections

private:
int lfd_; // Listening socket
eventloop* loop_; // Event loop
event* ev_; // Event object
Callback cb_; // Callback function
bool shutdown_; // Whether it has been shut down
};

}

Function Descriptions:

  • acceptor(int port, eventloop* base);
    Constructor that initializes the acceptor object.

  • ~acceptor();
    Destructor that stops listening and releases resources.

  • void listen();
    Starts listening.

  • void stop();
    Stops listening.

  • void init_sock(int port);
    Initializes the listening socket.

  • void setcb(const Callback& accept_cb);
    Sets the callback function for new connections.

  • void handle_accept();
    Handles new connection events.


server

Description:

The server class encapsulates TCP and UDP server functionalities, managing connections, events, and thread pools.

Interface:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
cppCopy codenamespace moon {

class udpevent;
class signalevent;
class timerevent;

class server {
public:
using SCallback = std::function<void(int)>;
using UCallback = std::function<void(const sockaddr_in&, udpevent*)>;
using RCallback = std::function<void(bfevent*)>;
using Callback = std::function<void()>;

server(int port = -1);
~server();

void start(); // Start the server
void stop(); // Stop the server
void init_pool(int timeout = -1); // Initialize the thread pool
void init_pool(int tnum, int timeout);// Initialize the thread pool with a specified number of threads
void init_pool_noadjust(int tnum, int timeout); // Initialize the thread pool with a specified number of threads without dynamic scheduling
void enable_tcp(int port); // Enable TCP service
void enable_tcp_accept(); // Enable TCP connection listening
void disable_tcp_accept(); // Disable TCP connection listening
eventloop* getloop(); // Get the main event loop
eventloop* dispatch(); // Dispatch events

// Set the callback functions for TCP connections
void set_tcpcb(const RCallback& rcb, const Callback& wcb, const Callback& ecb);

// Event operations
void addev(base_event *ev);
void delev(base_event *ev);
void modev(base_event *ev);
udpevent* add_udpev(int port, const UCallback& rcb, const Callback& ecb); // It is recommended to use this function to add a `udpevent`. Errors will automatically clean up.
signalevent* add_sev(int signo, const SCallback& cb);
signalevent* add_sev(const std::vector<int>& signals, const SCallback& cb);
timerevent* add_timeev(int timeout_ms, bool periodic, const Callback& cb);

/** v1.0.0 **/
/* void add_ev(event *ev);
void del_ev(event *ev);
void mod_ev(event *ev);
void add_bev(bfevent *bev);
void del_bev(bfevent *bev);
void mod_bev(bfevent *bev);
void add_udpev(udpevent *uev);
void del_udpev(udpevent *uev);
void mod_udpev(udpevent *uev);
void add_sev(signalevent* sev);
void del_sev(signalevent* sev);
void add_timeev(timerevent *tev);
void del_timeev(timerevent *tev); */

private:
void acceptcb_(int fd); // Handle new connections

void tcp_eventcb_(bfevent* bev); // Handle TCP error events

void handle_close(base_event* ev); // Handle event closure

private:
std::mutex events_mutex_;
eventloop base_; // Main event loop
looptpool pool_; // Thread pool
acceptor acceptor_; // Acceptor
int port_; // Port number
bool tcp_enable_; // Whether TCP is enabled
std::list<base_event*> events_; // List of events

// TCP connection callback functions
RCallback readcb_;
Callback writecb_;
Callback eventcb_;
};

}

Function Description:

  • server(int port = -1);
    Constructor: Initializes the server object.
  • ~server();
    Destructor: Shuts down the server and releases resources.
  • void start();
    Start the server: Begins the event loop.
  • void stop();
    Stop the server: Terminates the event loop.
  • void init_pool(int timeout = -1);
    Initialize the thread pool: Sets up the thread pool with an optional timeout.
  • void enable_tcp(int port);
    Enable TCP service: Activates TCP functionality on the specified port.
  • void enable_tcp_accept();
    Enable TCP connection listening: Starts listening for incoming TCP connections.
  • void disable_tcp_accept();
    Disable TCP connection listening: Stops listening for incoming TCP connections.
  • eventloop* getloop();
    Get the main event loop: Retrieves the primary event loop instance.
  • eventloop* dispatch();
    Dispatch events: Distributes events to the thread pool.
  • void set_tcpcb(const RCallback& rcb, const Callback& wcb, const Callback& ecb);
    Set TCP connection callbacks: Assigns callback functions for read, write, and error events on TCP connections.
  • void addev(base_event *ev);
    Add a general event: Adds a generic event (can pass any event type).
  • void delev(base_event *ev);
    Delete a general event: Removes a generic event (can pass any event type).
  • void modev(base_event *ev);
    Modify a general event: Modifies a generic event (can pass any event type).
  • udpevent* add_udpev(int port, const UCallback& rcb, const Callback& ecb);
    Add and initialize a UDP event: Adds a udpevent and initializes it. It is recommended to use this function to add a udpevent, as it will automatically clean up in case of errors.
  • signalevent* add_sev(int signo, const SCallback& cb);
    Add and initialize a signal event: Adds a signalevent for a specific signal number and assigns a callback function.
  • signalevent* add_sev(const std::vector<int>& signals, const SCallback& cb);
    Add and initialize multiple signal events: Adds signalevent instances for a list of signal numbers and assigns a callback function.
  • timerevent* add_timeev(int timeout_ms, bool periodic, const Callback& cb);
    Add and initialize a timer event: Adds a timerevent with a specified timeout in milliseconds, a flag indicating if it is periodic, and assigns a callback function.

wrap

Description:

The wrap module encapsulates cross-platform socket operation functions, providing a unified interface for both Windows and Unix systems.

Interface:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
#ifndef _WRAP_H_
#define _WRAP_H_

#ifdef _WIN32

// Socket encapsulation functions for Windows platform
void setreuse(SOCKET fd);
void setnonblock(SOCKET fd);
void perr_exit(const char* s);
// Other function definitions...

#else

// Socket encapsulation functions for Unix/Linux platform
void settcpnodelay(int fd);
void setreuse(int fd);
void setnonblock(int fd);
void perr_exit(const char* s);
int Accept(int fd, struct sockaddr* sa, socklen_t* salenptr);
int Bind(int fd, const struct sockaddr* sa, socklen_t salen);
int Connect(int fd, const struct sockaddr* sa, socklen_t salen);
int Listen(int fd, int backlog);
int Socket(int family, int type, int protocol);
ssize_t Read(int fd, void* ptr, size_t nbytes);
ssize_t Write(int fd, const void* ptr, size_t nbytes);
int Close(int fd);
ssize_t Readn(int fd, void* vptr, size_t n);
ssize_t Writen(int fd, const void* vptr, size_t n);
ssize_t Readline(int fd, void* vptr, size_t maxlen);
// Other function definitions...

#endif

#endif

Function Descriptions:

  • void settcpnodelay(int fd);
    Sets the socket to no-delay mode (TCP_NODELAY).

  • void setreuse(int fd);
    Sets the socket’s address and port to be reusable.

  • void setnonblock(int fd);
    Sets the socket to non-blocking mode.

  • void perr_exit(const char* s);
    Prints an error message and exits the program.

  • int Accept(int fd, struct sockaddr* sa, socklen_t* salenptr);
    Accepts a new connection.

  • int Bind(int fd, const struct sockaddr* sa, socklen_t salen);
    Binds the socket to a specified address and port.

  • int Connect(int fd, const struct sockaddr* sa, socklen_t salen);
    Connects to a specified address and port.

  • int Listen(int fd, int backlog);
    Starts listening on the socket.

  • int Socket(int family, int type, int protocol);
    Creates a new socket.

  • ssize_t Read(int fd, void* ptr, size_t nbytes);
    Reads data from the socket.

  • ssize_t Write(int fd, const void* ptr, size_t nbytes);
    Writes data to the socket.

  • int Close(int fd);
    Closes the socket.

  • ssize_t Readn(int fd, void* vptr, size_t n);
    Reads a specified number of bytes from the socket.

  • ssize_t Writen(int fd, const void* vptr, size_t n);
    Writes a specified number of bytes to the socket.

  • ssize_t Readline(int fd, void* vptr, size_t maxlen);
    Reads a line of data from the socket.


ringbuff

Description:

ringbuff is a lock-free, circular buffer designed for efficient, high-performance scenarios where multiple threads may be producing and consuming data concurrently. It is ideal for use cases where minimal latency and overhead are required, such as real-time data processing applications.

Interface:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
namespace moon {

// 无锁环形缓冲区 RingBuffer
// lock-free ringbuffer
template <class T>
class ringbuff {
public:
ringbuff(size_t size = 1024)
: head_(0),
tail_(0),
capacity_(adj_size(size)),
buffer_(new T[capacity_]) {}
// ~ringbuff()=default;
~ringbuff() { delete[] buffer_; }
/** push function **/
bool push(const T& item);
// push with move
bool push_move(T&& item);
/** pop function **/
bool pop(T& item);
// pop with move
bool pop_move(T& item);

/** performance function **/
size_t capacity() const;
size_t size() const;
bool empty();
bool full();
void swap(ringbuff& other) noexcept;

/** swap_to function **/
void swap_to_list(std::list<T>& list_);
void swap_to_vector(std::vector<T>& vec_);
std::list<T> swap_to_list();
std::vector<T> swap_to_vector();

private:
bool is_powtwo(size_t n) const;
size_t next_powtwo(size_t n) const;
size_t adj_size(size_t size) const;

private:
// avoid pseudo shareing
std::atomic<size_t> head_ alignas(64);
std::atomic<size_t> tail_ alignas(64);
// std::unique_ptr<T[]> buffer_;
size_t capacity_;
T* buffer_;
};

} // namespace moon

Function Description:

  • ringbuff(size_t size = 1024): Constructor that initializes the ring buffer with a specified capacity, adjusting it to the nearest power of two for optimal performance. The default capacity is 1024.
  • ~ringbuff(): Destructor that cleans up the allocated buffer.
  • bool push(const T& item): Attempts to add an item to the buffer. Returns true if successful; otherwise, returns false if the buffer is full. This method copies the item into the buffer.
  • bool push_move(T&& item): Similar to push, but uses move semantics to avoid copying the item. Returns true if successful; false if the buffer is full.
  • bool pop(T& item): Attempts to remove an item from the buffer and copy it into the provided variable. Returns true if successful; otherwise, false if the buffer is empty.
  • bool pop_move(T& item): Similar to pop, but uses move semantics to move the item out of the buffer. Returns true if successful; otherwise, false if the buffer is empty.
  • size_t capacity() const: Returns the current capacity of the buffer.
  • size_t size() const: Returns the number of items currently in the buffer.This operation requires additional atomic operations, resulting in performance loss. Please use with caution
  • bool empty() const: Checks if the buffer is empty.
  • bool full() const: Checks if the buffer is full.
  • void swap(ringbuff& other) noexcept: Exchanges the contents of this buffer with another ringbuff instance, including their internal state without copying elements.
  • void swap_to_list(std::list<T>& list_): Moves all elements from the buffer to a std::list, clearing the buffer.
  • void swap_to_vector(std::vector<T>& vec_): Moves all elements from the buffer to a std::vector, clearing the buffer.
  • std::list<T> swap_to_list(): Moves all elements from the buffer to a new std::list, returning it.
  • std::vector<T> swap_to_vector(): Moves all elements from the buffer to a new std::vector, returning it.

lfthread

Description:

lfthread is a lightweight, lock-free thread management class designed for high concurrency and low latency applications. It utilizes a ring buffer (ringbuff) to queue tasks which are functions encapsulated in std::function<void()>. The class handles task execution in its own thread and provides mechanisms for task enqueueing, thread shutdown, and buffer swapping.

Interface:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
namespace moon {

class lfthread {
public:
using task = std::function<void()>;
lfthread(size_t size)
: buffer_(size),
shutdown_(false),
t_(std::thread(&lfthread::t_task, this)) {}
~lfthread() { t_shutdown(); }
bool enqueue_task(task _task);
bool enqueue_task_move(task&& _task);
void t_shutdown();
int getload() const;
void swap_to_ringbuff(ringbuff<task>& rb_);
void swap_to_list(std::list<task>& list_);
void swap_to_vector(std::vector<task>& vec_);
std::list<task> swap_to_list();
std::vector<task> swap_to_vector();

/** move copy **/
lfthread& operator=(lfthread&& other) noexcept;
lfthread(lfthread&& other) noexcept;

private:
void t_task();

private:
ringbuff<task> buffer_; // Task Queue(lock-free Ring-Buffer)
bool shutdown_;
std::thread t_;
};

} // namespace moon

Function Description:

  • lfthread(size_t size): Constructs the thread with a specified size for the task buffer.
  • ~lfthread(): Destroys the thread, ensuring it is properly shut down.
  • bool enqueue_task(task _task): Attempts to enqueue a new task into the buffer. Returns true if successful; false if the buffer is full.
  • bool enqueue_task_move(task&& _task): Similar to enqueue_task, but uses move semantics to optimize performance.
  • void t_shutdown(): Shuts down the thread, ensuring all tasks are completed and the thread is joinable before exiting.
  • int getload() const: Returns the number of tasks currently in the buffer.
  • void swap_to_ringbuff(ringbuff<task>& rb_): Swaps the internal task buffer with another ring buffer.
  • void swap_to_list(std::list<task>& list_): Transfers all tasks from the buffer to a specified std::list.
  • void swap_to_vector(std::vector<task>& vec_): Transfers all tasks from the buffer to a specified std::vector.
  • std::list<task> swap_to_list(): Returns a std::list containing all tasks from the buffer.
  • std::vector<task> swap_to_vector(): Returns a std::vector containing all tasks from the buffer.
  • lfthread& operator=(lfthread&& other) noexcept: Move assignment operator.
  • lfthread(lfthread&& other) noexcept: Move constructor.

lfthreadpool

Description:

lfthreadpool is a versatile thread pool management class designed to handle dynamic or static allocation of threads for executing tasks. It leverages lfthread instances for managing individual threads and tasks, allowing for efficient task distribution and execution in a multi-threaded environment. The pool can operate in either a static mode, where the number of threads is fixed, or a dynamic mode, where threads can be adjusted based on workload.

Interface:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
#define ADJUST_TIMEOUT_SEC 5

namespace moon {

class lfthread;

enum class PoolMode {
Static, // 静态模式
Dynamic // 动态模式
};

class lfthreadpool {
public:
lfthreadpool(int tnum = -1, size_t buffsize = 1024,
PoolMode mode = PoolMode::Static);
~lfthreadpool();
void init();
void t_shutdown();

/** add_task function **/
template <typename _Fn, typename... _Args>
bool add_task(_Fn&& fn, _Args&&... args);

template <typename _Fn, typename... _Args>
bool add_task_move(_Fn&& fn, _Args&&... args);

lfthreadpool(const lfthreadpool&) = delete;
lfthreadpool& operator=(const lfthreadpool&) = delete;

private:
const size_t getnext();
void adjust_task();
void del_thread_dispath();
void add_thread();
const size_t getminidx();
const size_t getmaxidx();
const int getload();
void setnum();
void setnum(size_t n);
int getcores() const;

private:
std::atomic<size_t> tnum_;
// size_t tnum_;
std::thread mentor_;
std::vector<lfthread*> workers_;
// bool shutdown_;
std::atomic<bool> shutdown_;
PoolMode mode_;
size_t buffsize_;
size_t next_ = 0;
int timesec_ = 5;
int coolsec_ = 30;
int load_max = 80;
int load_min = 20;
int max_tnum = 0;
int min_tnum = 0;
};

} // namespace moon

Function Description:

  • lfthreadpool(int tnum, size_t buffsize, PoolMode mode): Constructs the thread pool with a specific number of threads, buffer size per thread, and operating mode.
  • ~lfthreadpool(): Destroys the thread pool, ensuring all threads are properly shut down.
  • void init(): Initializes the thread pool, creating the specified number of threads.
  • void t_shutdown(): Shuts down all threads in the pool safely.
  • bool add_task(_Fn&& fn, _Args&&... args): Adds a new task to the pool; tasks are distributed to threads based on current load and pool mode.
  • bool add_task_move(_Fn&& fn, _Args&&... args): Similar to add_task, but uses move semantics to optimize the handling of tasks that are movable.
  • void adjust_task(): Dynamically adjusts the number of threads based on the current load and predefined thresholds if the pool is in dynamic mode.
  • void del_thread_dispath(): Removes a thread from the pool when it is no longer needed and redistributes its tasks among remaining threads.
  • void add_thread(): Adds a new thread to the pool to handle increased load.
  • int getload(): Calculates the total load across all threads as a percentage of their capacity.

Usage Examples

The following examples demonstrate how to use the MoonNet network library to build TCP and UDP servers, as well as how to use timer and signal handling functionalities.

TCP Server Example

Description:

Creates a simple TCP server that listens on a specified port, accepts client connections, and echoes back received data.

Code Example:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
#include "server.h"
#include "bfevent.h"
#include <iostream>

// Read event callback function
void on_read(moon::bfevent* bev) {
std::string data = bev->receive();
if (!data.empty()) {
std::cout << "Received: " << data << std::endl;
// Echo the data back
bev->sendout(data);
}
}

// Write event callback function
void on_write(){
// Handle write events if necessary
}

// Other event callback function (e.g., error)
void on_event() {
std::cout << "An error occurred on the connection." << std::endl;
}

int main() {
// Create a server listening on port 8080
moon::server tcp_server(8080);

// Set the buffered event callback functions
tcp_server.set_tcpcb(on_read, on_write, on_event);

// Initialize the thread pool (optional)
tcp_server.init_pool();

// Start the server
tcp_server.start();

return 0;
}

Explanation:

  1. Create Server Instance:

    1
    moon::server tcp_server(8080);

    Creates a server object that listens on TCP port 8080.

  2. Set Callback Functions:

    1
    tcp_server.set_tcpcb(on_read, on_write, on_event);

    Sets the buffered event’s read, write, and error callback functions.

  3. Initialize Thread Pool:

    1
    tcp_server.init_pool();

    Initializes the thread pool to improve the server’s concurrency handling capabilities.

  4. Start Server:

    1
    tcp_server.start();

    Starts the server, begins the event loop, and listens for and handles connections.


UDP Server Example

Description:

Creates a simple UDP server that listens on a specified port, receives data packets, and replies to clients.

Code Example:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
#include "server.h"
#include "udpevent.h"
#include <iostream>

// UDP receive callback function
void on_udp_receive(const sockaddr_in& addr, moon::udpevent* uev) {
std::string data = uev->receive();
if (!data.empty()) {
std::cout << "UDP Received from " << inet_ntoa(addr.sin_addr)
<< ":" << ntohs(addr.sin_port) << " - " << data << std::endl;

// Send a reply
std::string reply = "Echo: " + data;
uev->send_to(reply, addr);
}
}

int main() {
// Create a server without specifying a TCP port
moon::server udp_server(-1);

// Add a UDP event listening on port 5005
udp_server.add_udpev(5005, on_udp_receive, nullptr);

// Start the server
udp_server.start();

return 0;
}

Explanation:

  1. Create Server Instance:

    1
    moon::server udp_server(-1);

    Creates a server object without enabling TCP service.

  2. Add UDP Event:

    1
    udp_server.add_udpev(5005, on_udp_receive, nullptr);

    Adds a UDP event that listens on port 5005 and sets the receive callback function.

  3. Start Server:

    1
    udp_server.start();

    Starts the server, begins the event loop, and listens for and handles UDP packets.


Timer Example

Description:

Uses a timer to periodically execute a task, such as printing a message every second.

Code Example:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
#include "server.h"
#include "timerevent.h"
#include <iostream>

// Timer callback function
void on_timer() {
std::cout << "Timer triggered!" << std::endl;
}

int main() {
// Create a server without specifying a TCP port
moon::server timer_server(-1);

// Add a periodic timer that triggers every 1000 milliseconds
timer_server.add_timeev(1000, true, on_timer);

// Start the server
timer_server.start();

return 0;
}

Explanation:

  1. Create Server Instance:

    1
    moon::server timer_server(-1);

    Creates a server object without enabling TCP service.

  2. Add Timer Event:

    1
    timer_server.add_timeev(1000, true, on_timer);

    Adds a timer event with a timeout of 1000 milliseconds (1 second). The true parameter indicates that the timer is periodic.

  3. Start Server:

    1
    timer_server.start();

    Starts the server, begins the event loop, and the timer starts working.


Signal Handling Example

Description:

Uses the signal event handling class to listen for and respond to specific UNIX signals, such as SIGINT (Ctrl+C).

Code Example:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
#include "server.h"
#include "signalevent.h"
#include <iostream>
#include <vector>
#include <csignal>

// Signal handling callback function
void on_signal(int signo) {
std::cout << "Received signal: " << signo << std::endl;
// Perform cleanup operations or graceful exit here
}

int main() {
// Create a server without specifying a TCP port
moon::server signal_server(-1);

// Add signal events to listen for SIGINT and SIGTERM
std::vector<int> signals = {SIGINT, SIGTERM};
signal_server.add_sev(signals, on_signal);

// Start the server
signal_server.start();

return 0;
}

Explanation:

  1. Create Server Instance:

    1
    moon::server signal_server(-1);

    Creates a server object without enabling TCP service.

  2. Add Signal Event:

    1
    2
    std::vector<int> signals = {SIGINT, SIGTERM};
    signal_server.add_sev(signals, on_signal);

    Adds a signal event to listen for SIGINT and SIGTERM signals and sets the callback function.

  3. Start Server:

    1
    signal_server.start();

    Starts the server, begins the event loop, and the signal handling starts working.


ringbuff Usage Example

Code Example:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
#include <moonnet/moonnet.h> // You can only reference the general header file
//#include <moonnet/ringbuff.h>
#include <iostream>
#include <string>
#include <vector>

int main() {
moon::ringbuff<std::string> buffer(32); // Initialize a ring buffer with capacity adjusted to the nearest power of two

// Push elements into the ring buffer using push and push_move
buffer.push("Hello");
buffer.push_move(std::string("World"));

// Pop elements from the ring buffer
std::string data;
if (buffer.pop(data)) {
std::cout << "Popped: " << data << std::endl;
}

// Check buffer size and capacity
std::cout << "Current buffer size: " << buffer.size() << std::endl;
std::cout << "Buffer capacity: " << buffer.capacity() << std::endl;

// Check if the buffer is empty or full
std::cout << "Is the buffer empty? " << (buffer.empty() ? "Yes" : "No") << std::endl;
std::cout << "Is the buffer full? " << (buffer.full() ? "Yes" : "No") << std::endl;

// Use swap_to_vector to export the buffer's contents
std::vector<std::string> vec;
buffer.swap_to_vector(vec);

// Display exported data
for (auto &str : vec) {
std::cout << "Exported: " << str << std::endl;
}

return 0;
}

lfthreadpool Usage Example

Code Example:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
#include <moonnet/moonnet.h> // You can only reference the general header file
//#include <moonnet/lfthreadpool.h>
#include <iostream>
#include <functional>

void printTask(int num) {
std::cout << "Task " << num << " completed." << std::endl;
}

int main() {
// If tnum is -1, it means that the system will automatically set the size of the thread pool
// moon::lfthreadpool pool(4, 1024, moon::PoolMode::Dynamic); // Initialize the thread pool in dynamic mode
// Use system settings for pool size and static mode
moon::lfthreadpool pool(-1, 32);

// Add tasks to the pool using add_task and add_task_move
for (int i = 0; i < 10; i++) {
pool.add_task(printTask, i);
pool.add_task_move([=] { printTask(i + 10); });
}

// Allow some time for tasks to execute
std::this_thread::sleep_for(std::chrono::seconds(2));

// Shutdown the pool
pool.t_shutdown();

return 0;
}

Error Handling

MoonNet uses standard POSIX error handling mechanisms. When an error occurs, the library calls the perr_exit function to print an error message and terminate the program. Users can customize error handling logic as needed, such as modifying callback functions to handle specific error situations.

Example:

In wrap.cpp, the perr_exit function:

1
2
3
4
void perr_exit(const char *s) {
perror(s);
exit(1);
}

Custom Error Handling:

If you want to avoid directly exiting the program when an error occurs, you can modify the perr_exit function or add error handling logic in the callback functions.

Example:

Handling errors in the callback function instead of exiting:

1
2
3
4
void on_event() {
std::cerr << "An error occurred on the connection." << std::endl;
// Perform cleanup operations or attempt recovery
}

Frequently Asked Questions (FAQs)

1. How do I enable both TCP and UDP services simultaneously?

Answer:

Create a server instance, enable TCP service, and add UDP events.

Example:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
#include "server.h"
#include "bfevent.h"
#include "udpevent.h"
#include <iostream>

// TCP read callback function
void on_tcp_read(moon::bfevent* bev) {
std::string data = bev->receive();
if (!data.empty()) {
std::cout << "TCP Received: " << data << std::endl;
// Echo the data back
bev->sendout(data);
}
}

// UDP receive callback function
void on_udp_receive(const sockaddr_in& addr, moon::udpevent* uev) {
std::string data = uev->receive();
if (!data.empty()) {
std::cout << "UDP Received from " << inet_ntoa(addr.sin_addr)
<< ":" << ntohs(addr.sin_port) << " - " << data << std::endl;

// Send a reply
std::string reply = "Echo: " + data;
uev->send_to(reply, addr);
}
}

int main() {
// Create a server listening on TCP port 8080
moon::server network_server(8080);

// Set the TCP read event callback function
network_server.set_tcpcb(on_tcp_read, nullptr, nullptr);

// Add a UDP event listening on port 9090
network_server.add_udpev(9090, on_udp_receive, nullptr);

// Initialize the thread pool
network_server.init_pool();

// Start the server
network_server.start();

return 0;
}

2. How do I gracefully shut down the server?

Answer:

Call the stop() method to interrupt the event loop and ensure all resources are properly released.

Example:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
#include "server.h"
#include "signalevent.h"
#include <iostream>
#include <csignal>

moon::server* global_server = nullptr;

// Signal handling callback
void on_signal(int signo) {
std::cout << "Received signal: " << signo << ", stopping server..." << std::endl;
if (global_server) {
global_server->stop();
}
}

int main() {
// Create a server listening on TCP port 8080
moon::server network_server(8080);
global_server = &network_server;

// Set TCP callback functions (omitted)

// Add signal handling for SIGINT
network_server.add_sev(SIGINT, on_signal);

// Initialize the thread pool
network_server.init_pool();

// Start the server
network_server.start();

std::cout << "Server stopped gracefully." << std::endl;
return 0;
}

3. How do I handle data race issues in multithreading?

Answer:

MoonNet minimizes data race issues through its design of thread pools and event loops. Each event loop runs in an independent thread. When processing data in callback functions, users should ensure thread-safe access to shared resources, using synchronization mechanisms like mutexes (std::mutex).

Example:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
#include "server.h"
#include "bfevent.h"
#include <iostream>
#include <mutex>

// Shared resource
std::mutex data_mutex;
int shared_data = 0;

// TCP read callback function
void on_tcp_read(moon::bfevent* bev) {
std::string data = bev->receive();
if (!data.empty()) {
std::lock_guard<std::mutex> lock(data_mutex);
shared_data += data.size();
std::cout << "Shared Data Size: " << shared_data << std::endl;
}
}

int main() {
// Create a server listening on TCP port 8080
moon::server network_server(8080);

// Set the buffered event callback functions
network_server.set_tcpcb(on_tcp_read, nullptr, nullptr);

// Initialize the thread pool
network_server.init_pool();

// Start the server
network_server.start();

return 0;
}

Conclusion

MoonNet provides an efficient and easy-to-use network programming framework suitable for building various types of network applications. With its modular design and flexible interfaces, developers can quickly implement high-performance network services. We hope this documentation helps you better understand and use the MoonNet network library.

If you have any questions or suggestions, feel free to contact the author or submit feedback.



MoonNet guide_en
https://moonfordream.github.io/posts/MoonNet-guide-en/
作者
Moon
发布于
2024年10月27日
许可协议