Python os.setsid Function
Last modified April 11, 2025
This comprehensive guide explores Python's os.setsid function,
which creates new process sessions. We'll cover Unix process groups, session
management, and practical daemonization examples.
Basic Definitions
The os.setsid function creates a new session and sets the
process as its leader. It's a Unix-specific system call that detaches from
the controlling terminal.
Key features: creates new session, becomes session leader, becomes process group leader, has no controlling terminal. Returns the new session ID.
Creating a New Session
This basic example demonstrates creating a new session with os.setsid. The child process becomes a session leader independent of the parent.
import os
import time
pid = os.fork()
if pid == 0: # Child process
print(f"Child PID: {os.getpid()}")
print(f"Child PGID before setsid: {os.getpgid(0)}")
# Create new session
sid = os.setsid()
print(f"New SID: {sid}")
print(f"Child PGID after setsid: {os.getpgid(0)}")
time.sleep(10) # Keep process alive to observe
else: # Parent process
print(f"Parent PID: {os.getpid()}")
print(f"Parent PGID: {os.getpgid(0)}")
os.waitpid(pid, 0) # Wait for child
The child process creates a new session with os.setsid, becoming independent. The parent's process group remains unchanged while the child gets a new one.
Note that os.setsid will fail if the calling process is already a process group leader, which is why we fork first.
Creating a Daemon Process
A common use of os.setsid is daemon creation. This example shows the full daemonization process including session creation and file descriptor handling.
import os
import sys
import time
def daemonize():
# Fork first time to background
pid = os.fork()
if pid > 0:
sys.exit(0) # Exit parent
# Create new session
os.setsid()
# Fork second time to ensure not session leader
pid = os.fork()
if pid > 0:
sys.exit(0)
# Change working directory
os.chdir('/')
# Close file descriptors
for fd in range(3, 1024):
try:
os.close(fd)
except OSError:
pass
# Redirect stdio to /dev/null
os.open('/dev/null', os.O_RDWR) # stdin
os.dup2(0, 1) # stdout
os.dup2(0, 2) # stderr
if __name__ == '__main__':
daemonize()
while True:
with open('/tmp/daemon.log', 'a') as f:
f.write(f"Daemon running at {time.ctime()}\n")
time.sleep(5)
This creates a proper daemon process by forking twice, creating a new session, and handling file descriptors. The daemon writes to a log file periodically.
The double fork ensures the daemon cannot reacquire a controlling terminal. The working directory is changed to prevent filesystem unmounting issues.
Handling Session Leader Restrictions
This example demonstrates the restriction that process group leaders cannot call setsid, and how to work around it with proper forking.
import os
try:
# This will fail because we're already a process group leader
sid = os.setsid()
print(f"Created session {sid}")
except OSError as e:
print(f"Error calling setsid: {e}")
# Proper approach: fork first
pid = os.fork()
if pid == 0: # Child
try:
sid = os.setsid()
print(f"Child created session {sid}")
print(f"New PGID: {os.getpgid(0)}")
except OSError as e:
print(f"Child error: {e}")
os._exit(0)
else: # Parent
os.waitpid(pid, 0)
print("Parent exiting")
The first attempt fails because the main process is a process group leader. The second approach forks first, allowing the child (not a group leader) to successfully create a new session.
This demonstrates why daemonization typically involves forking before calling setsid, to avoid the process group leader restriction.
Process Group Management
This example shows how setsid affects process groups and demonstrates process group management with os.setpgid and os.getpgid.
import os
import time
def show_ids(label):
print(f"{label}: PID={os.getpid()}, PGID={os.getpgid(0)}, SID={os.getsid(0)}")
show_ids("Parent before fork")
pid = os.fork()
if pid == 0: # Child
show_ids("Child before setsid")
# Create new session
sid = os.setsid()
show_ids("Child after setsid")
# Fork again to create a process group
pid2 = os.fork()
if pid2 == 0: # Grandchild
show_ids("Grandchild before setpgid")
os.setpgid(0, 0) # Create new process group
show_ids("Grandchild after setpgid")
time.sleep(10)
else:
time.sleep(10)
else: # Parent
show_ids("Parent after fork")
os.waitpid(pid, 0)
This demonstrates the process hierarchy changes when creating new sessions and process groups. The grandchild creates its own process group within the child's session.
The output shows how PID, PGID, and SID values change at each step, illustrating the process relationships.
Terminal Control and Session Leadership
This example demonstrates how setsid detaches from the controlling terminal and how session leaders interact with terminal signals.
import os
import signal
import time
def handler(signum, frame):
print(f"Received signal {signum}")
# Set up signal handler
signal.signal(signal.SIGHUP, handler)
pid = os.fork()
if pid == 0: # Child
print(f"Child PID: {os.getpid()}")
# Create new session (detaches from terminal)
os.setsid()
print("Child in new session, ignoring SIGHUP")
signal.signal(signal.SIGHUP, signal.SIG_IGN)
time.sleep(30) # Keep process alive
else: # Parent
print(f"Parent PID: {os.getpid()}")
time.sleep(1) # Let child set up
# Send SIGHUP to child's process group
os.kill(-pid, signal.SIGHUP)
os.waitpid(pid, 0)
The child process creates a new session and ignores SIGHUP signals. The parent attempts to send SIGHUP to the child's original process group.
This demonstrates how session leaders handle terminal signals differently and how process groups affect signal delivery.
Session ID Inheritance
This example shows how session IDs are inherited across fork and exec calls, and how setsid creates new independent sessions.
import os
import sys
def show_session():
print(f"PID: {os.getpid()}, SID: {os.getsid(0)}")
show_session() # Original process
pid = os.fork()
if pid == 0: # Child
show_session() # Inherited session
# Create new session
os.setsid()
show_session()
# Execute new program
os.execvp("python3", ["python3", "-c",
"import os; print(f'Exec PID: {os.getpid()}, SID: {os.getsid(0)}')"])
else: # Parent
os.waitpid(pid, 0)
show_session() # Still original session
The child process first inherits the parent's session, then creates a new one with setsid, and finally executes a new program that shows its session.
This demonstrates that exec preserves the session ID, while setsid creates a completely new independent session.
Security Considerations
- Privilege separation: New sessions may affect privilege inheritance
- Signal handling: Session leaders handle terminal signals differently
- Resource limits: New sessions may have different resource constraints
- Process isolation: Sessions provide basic process isolation
- Platform limitations: Windows has different process grouping
Best Practices
- Fork first: Always fork before calling setsid
- Handle stdio: Redirect or close file descriptors after setsid
- Change directory: Avoid filesystem unmounting issues
- Signal handling: Set up appropriate signal handlers
- Error checking: Verify setsid return value
Source References
Author
List all Python tutorials.