|  | .. _timerfd-howto: | 
|  |  | 
|  | ***************************** | 
|  | timer file descriptor HOWTO | 
|  | ***************************** | 
|  |  | 
|  | :Release: 1.13 | 
|  |  | 
|  | This HOWTO discusses Python's support for the linux timer file descriptor. | 
|  |  | 
|  |  | 
|  | Examples | 
|  | ======== | 
|  |  | 
|  | The following example shows how to use a timer file descriptor | 
|  | to execute a function twice a second: | 
|  |  | 
|  | .. code-block:: python | 
|  |  | 
|  | # Practical scripts should use really use a non-blocking timer, | 
|  | # we use a blocking timer here for simplicity. | 
|  | import os, time | 
|  |  | 
|  | # Create the timer file descriptor | 
|  | fd = os.timerfd_create(time.CLOCK_REALTIME) | 
|  |  | 
|  | # Start the timer in 1 second, with an interval of half a second | 
|  | os.timerfd_settime(fd, initial=1, interval=0.5) | 
|  |  | 
|  | try: | 
|  | # Process timer events four times. | 
|  | for _ in range(4): | 
|  | # read() will block until the timer expires | 
|  | _ = os.read(fd, 8) | 
|  | print("Timer expired") | 
|  | finally: | 
|  | # Remember to close the timer file descriptor! | 
|  | os.close(fd) | 
|  |  | 
|  | To avoid the precision loss caused by the :class:`float` type, | 
|  | timer file descriptors allow specifying initial expiration and interval | 
|  | in integer nanoseconds with ``_ns`` variants of the functions. | 
|  |  | 
|  | This example shows how :func:`~select.epoll` can be used with timer file | 
|  | descriptors to wait until the file descriptor is ready for reading: | 
|  |  | 
|  | .. code-block:: python | 
|  |  | 
|  | import os, time, select, socket, sys | 
|  |  | 
|  | # Create an epoll object | 
|  | ep = select.epoll() | 
|  |  | 
|  | # In this example, use loopback address to send "stop" command to the server. | 
|  | # | 
|  | # $ telnet 127.0.0.1 1234 | 
|  | # Trying 127.0.0.1... | 
|  | # Connected to 127.0.0.1. | 
|  | # Escape character is '^]'. | 
|  | # stop | 
|  | # Connection closed by foreign host. | 
|  | # | 
|  | sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM) | 
|  | sock.bind(("127.0.0.1", 1234)) | 
|  | sock.setblocking(False) | 
|  | sock.listen(1) | 
|  | ep.register(sock, select.EPOLLIN) | 
|  |  | 
|  | # Create timer file descriptors in non-blocking mode. | 
|  | num = 3 | 
|  | fds = [] | 
|  | for _ in range(num): | 
|  | fd = os.timerfd_create(time.CLOCK_REALTIME, flags=os.TFD_NONBLOCK) | 
|  | fds.append(fd) | 
|  | # Register the timer file descriptor for read events | 
|  | ep.register(fd, select.EPOLLIN) | 
|  |  | 
|  | # Start the timer with os.timerfd_settime_ns() in nanoseconds. | 
|  | # Timer 1 fires every 0.25 seconds; timer 2 every 0.5 seconds; etc | 
|  | for i, fd in enumerate(fds, start=1): | 
|  | one_sec_in_nsec = 10**9 | 
|  | i = i * one_sec_in_nsec | 
|  | os.timerfd_settime_ns(fd, initial=i//4, interval=i//4) | 
|  |  | 
|  | timeout = 3 | 
|  | try: | 
|  | conn = None | 
|  | is_active = True | 
|  | while is_active: | 
|  | # Wait for the timer to expire for 3 seconds. | 
|  | # epoll.poll() returns a list of (fd, event) pairs. | 
|  | # fd is a file descriptor. | 
|  | # sock and conn[=returned value of socket.accept()] are socket objects, not file descriptors. | 
|  | # So use sock.fileno() and conn.fileno() to get the file descriptors. | 
|  | events = ep.poll(timeout) | 
|  |  | 
|  | # If more than one timer file descriptors are ready for reading at once, | 
|  | # epoll.poll() returns a list of (fd, event) pairs. | 
|  | # | 
|  | # In this example settings, | 
|  | #    1st timer fires every 0.25 seconds in 0.25 seconds. (0.25, 0.5, 0.75, 1.0, ...) | 
|  | #    2nd timer every 0.5 seconds in 0.5 seconds. (0.5, 1.0, 1.5, 2.0, ...) | 
|  | #    3rd timer every 0.75 seconds in 0.75 seconds. (0.75, 1.5, 2.25, 3.0, ...) | 
|  | # | 
|  | #    In 0.25 seconds, only 1st timer fires. | 
|  | #    In 0.5 seconds, 1st timer and 2nd timer fires at once. | 
|  | #    In 0.75 seconds, 1st timer and 3rd timer fires at once. | 
|  | #    In 1.5 seconds, 1st timer, 2nd timer and 3rd timer fires at once. | 
|  | # | 
|  | # If a timer file descriptor is signaled more than once since | 
|  | # the last os.read() call, os.read() returns the nubmer of signaled | 
|  | # as host order of class bytes. | 
|  | print(f"Signaled events={events}") | 
|  | for fd, event in events: | 
|  | if event & select.EPOLLIN: | 
|  | if fd == sock.fileno(): | 
|  | # Check if there is a connection request. | 
|  | print(f"Accepting connection {fd}") | 
|  | conn, addr = sock.accept() | 
|  | conn.setblocking(False) | 
|  | print(f"Accepted connection {conn} from {addr}") | 
|  | ep.register(conn, select.EPOLLIN) | 
|  | elif conn and fd == conn.fileno(): | 
|  | # Check if there is data to read. | 
|  | print(f"Reading data {fd}") | 
|  | data = conn.recv(1024) | 
|  | if data: | 
|  | # You should catch UnicodeDecodeError exception for safety. | 
|  | cmd = data.decode() | 
|  | if cmd.startswith("stop"): | 
|  | print(f"Stopping server") | 
|  | is_active = False | 
|  | else: | 
|  | print(f"Unknown command: {cmd}") | 
|  | else: | 
|  | # No more data, close connection | 
|  | print(f"Closing connection {fd}") | 
|  | ep.unregister(conn) | 
|  | conn.close() | 
|  | conn = None | 
|  | elif fd in fds: | 
|  | print(f"Reading timer {fd}") | 
|  | count = int.from_bytes(os.read(fd, 8), byteorder=sys.byteorder) | 
|  | print(f"Timer {fds.index(fd) + 1} expired {count} times") | 
|  | else: | 
|  | print(f"Unknown file descriptor {fd}") | 
|  | finally: | 
|  | for fd in fds: | 
|  | ep.unregister(fd) | 
|  | os.close(fd) | 
|  | ep.close() | 
|  |  | 
|  | This example shows how :func:`~select.select` can be used with timer file | 
|  | descriptors to wait until the file descriptor is ready for reading: | 
|  |  | 
|  | .. code-block:: python | 
|  |  | 
|  | import os, time, select, socket, sys | 
|  |  | 
|  | # In this example, use loopback address to send "stop" command to the server. | 
|  | # | 
|  | # $ telnet 127.0.0.1 1234 | 
|  | # Trying 127.0.0.1... | 
|  | # Connected to 127.0.0.1. | 
|  | # Escape character is '^]'. | 
|  | # stop | 
|  | # Connection closed by foreign host. | 
|  | # | 
|  | sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM) | 
|  | sock.bind(("127.0.0.1", 1234)) | 
|  | sock.setblocking(False) | 
|  | sock.listen(1) | 
|  |  | 
|  | # Create timer file descriptors in non-blocking mode. | 
|  | num = 3 | 
|  | fds = [os.timerfd_create(time.CLOCK_REALTIME, flags=os.TFD_NONBLOCK) | 
|  | for _ in range(num)] | 
|  | select_fds = fds + [sock] | 
|  |  | 
|  | # Start the timers with os.timerfd_settime() in seconds. | 
|  | # Timer 1 fires every 0.25 seconds; timer 2 every 0.5 seconds; etc | 
|  | for i, fd in enumerate(fds, start=1): | 
|  | os.timerfd_settime(fd, initial=i/4, interval=i/4) | 
|  |  | 
|  | timeout = 3 | 
|  | try: | 
|  | conn = None | 
|  | is_active = True | 
|  | while is_active: | 
|  | # Wait for the timer to expire for 3 seconds. | 
|  | # select.select() returns a list of file descriptors or objects. | 
|  | rfd, wfd, xfd = select.select(select_fds, select_fds, select_fds, timeout) | 
|  | for fd in rfd: | 
|  | if fd == sock: | 
|  | # Check if there is a connection request. | 
|  | print(f"Accepting connection {fd}") | 
|  | conn, addr = sock.accept() | 
|  | conn.setblocking(False) | 
|  | print(f"Accepted connection {conn} from {addr}") | 
|  | select_fds.append(conn) | 
|  | elif conn and fd == conn: | 
|  | # Check if there is data to read. | 
|  | print(f"Reading data {fd}") | 
|  | data = conn.recv(1024) | 
|  | if data: | 
|  | # You should catch UnicodeDecodeError exception for safety. | 
|  | cmd = data.decode() | 
|  | if cmd.startswith("stop"): | 
|  | print(f"Stopping server") | 
|  | is_active = False | 
|  | else: | 
|  | print(f"Unknown command: {cmd}") | 
|  | else: | 
|  | # No more data, close connection | 
|  | print(f"Closing connection {fd}") | 
|  | select_fds.remove(conn) | 
|  | conn.close() | 
|  | conn = None | 
|  | elif fd in fds: | 
|  | print(f"Reading timer {fd}") | 
|  | count = int.from_bytes(os.read(fd, 8), byteorder=sys.byteorder) | 
|  | print(f"Timer {fds.index(fd) + 1} expired {count} times") | 
|  | else: | 
|  | print(f"Unknown file descriptor {fd}") | 
|  | finally: | 
|  | for fd in fds: | 
|  | os.close(fd) | 
|  | sock.close() | 
|  | sock = None | 
|  |  |