A comprehensive Python library and command-line tool for managing Apache Guacamole users, groups, connections, and connection groups. Provides direct MySQL database access with secure operations and YAML output.
- User management (create, delete, modify, list)
- User group management (create, delete, modify membership)
- Connection management (create, delete, modify parameters)
- Connection group management (create, delete, modify hierarchy)
- Comprehensive listing commands with YAML output
- Data dump functionality
- Version information
- Secure database operations with parameterized queries
- Detailed error handling and validation
- Full programmatic access to all management features
- Context manager for automatic connection handling
- Type-safe parameter handling
- Comprehensive error reporting
- Advanced permission management with atomic operations
- Entity resolution with both name and ID support
- Connection group cycle detection
- Debug utilities for troubleshooting
- Transaction-safe operations
Just install it with pip:
pip install guacalib- Clone the repository:
git clone https://github.com/burbilog/guacalib.git
cd guacalib- Install the library:
pip install .Create a database configuration file in $HOME called .guacaman.ini for guacaman command line utility:
[mysql]
host = localhost
user = guacamole_user
password = your_password
database = guacamole_dbEnsure permissions are strict:
chmod 0600 $HOME/.guacaman.iniAlternatively, you can configure the database connection using environment variables. Environment variables take precedence over the configuration file:
export GUACALIB_HOST=localhost
export GUACALIB_USER=guacamole_user
export GUACALIB_PASSWORD=your_password
export GUACALIB_DATABASE=guacamole_dbPriority Order:
- Environment variables (if all required variables are set)
- Configuration file (fallback if environment variables are incomplete)
Usage in Python Library:
from guacalib import GuacamoleDB
import os
# Configure with environment variables
os.environ['GUACALIB_HOST'] = 'localhost'
os.environ['GUACALIB_USER'] = 'guacamole_user'
os.environ['GUACALIB_PASSWORD'] = 'your_password'
os.environ['GUACALIB_DATABASE'] = 'guacamole_db'
# Will automatically use environment variables if available
with GuacamoleDB() as db:
users = db.list_users()Environment Variables:
GUACALIB_HOST: MySQL server hostnameGUACALIB_USER: MySQL usernameGUACALIB_PASSWORD: MySQL passwordGUACALIB_DATABASE: MySQL database name
All four variables must be set for environment-based configuration to be used.
The guacalib library includes comprehensive API documentation with Google-style docstrings throughout the codebase. You can:
- View Online Documentation: Generate static HTML docs with
make docs - Start Documentation Server: Run interactive docs with
make docs-serve - Direct Python Access: Use
help(guacalib.GuacamoleDB)in Python
# Generate static documentation in docs/ directory
make docs
# Start interactive documentation server (clickable URL)
make docs-serveThe documentation includes detailed method signatures, parameter descriptions, return values, and usage examples for all library functions.
The guacalib library provides programmatic access to all Guacamole management features. Here are some examples:
from guacalib import GuacamoleDB
# Initialize with config file
guacdb = GuacamoleDB('~/.guacaman.ini')
# Initialize with environment variables (if set, will override config file)
with GuacamoleDB('~/.guacaman.ini') as guacdb:
# Your code here
# Initialize without specifying config file (will check environment variables first)
with GuacamoleDB() as guacdb:
# Your code here# Create user
guacdb.create_user('john.doe', 'secretpass')
# Check if user exists
if guacdb.user_exists('john.doe'):
print("User exists")
# Delete user
guacdb.delete_existing_user('john.doe')# Create user group
guacdb.create_usergroup('developers')
# Check if user group exists
if guacdb.usergroup_exists('developers'):
print("User group exists")
# Add user to a user group
guacdb.add_user_to_usergroup('john.doe', 'developers')
# Delete user group
guacdb.delete_existing_usergroup('developers')# Create VNC connection
conn_id = guacdb.create_connection(
'vnc',
'dev-server',
'192.168.1.100',
5901,
'vncpass'
)
# Grant connection to group
guacdb.grant_connection_permission(
'developers',
'USER_GROUP',
conn_id
)
# Check if connection exists
if guacdb.connection_exists('dev-server'):
print("Connection exists")
# Delete connection
guacdb.delete_existing_connection('dev-server')Connection groups allow organizing connections into hierarchical groups. Here are the key methods:
# Create a top-level connection group
guacdb.create_connection_group('production')
# Create a nested connection group
guacdb.create_connection_group('vnc_servers', parent_group_name='production')
# List all connection groups with their hierarchy
groups = guacdb.list_connection_groups()
for group_name, data in groups.items():
print(f"{group_name}:")
print(f" Parent: {data['parent']}")
print(f" Connections: {data['connections']}")
# Check if a connection group exists
if guacdb.connection_group_exists('production'):
print("Group exists")
# Modify a connection group's parent
guacdb.modify_connection_group_parent('vnc_servers', 'infrastructure')
# Remove a connection group's parent
guacdb.modify_connection_group_parent('vnc_servers', '')
# Delete a connection group
guacdb.delete_connection_group('vnc_servers')The GuacamoleDB class provides additional methods for complex operations:
# Connection permissions
guacdb.grant_connection_permission_to_user('john.doe', 'dev-server')
guacdb.revoke_connection_permission_from_user('john.doe', 'dev-server')
# Connection group permissions (advanced)
guacdb.grant_connection_group_permission_to_user('john.doe', 'production')
guacdb.grant_connection_group_permission_to_user_by_id('john.doe', 42)
guacdb.revoke_connection_group_permission_from_user('john.doe', 'production')
guacdb.revoke_connection_group_permission_from_user_by_id('john.doe', 42)# Resolve entities by name or ID with validation
conn_id = guacdb.resolve_connection_id(connection_name='dev-server')
conn_id = guacdb.resolve_connection_id(connection_id=42)
group_id = guacdb.resolve_conngroup_id(group_name='production')
group_id = guacdb.resolve_conngroup_id(group_id=15)
usergroup_id = guacdb.resolve_usergroup_id(group_name='developers')
usergroup_id = guacdb.resolve_usergroup_id(group_id=8)# Get specific entities by ID
connection = guacdb.get_connection_by_id(42)
conngroup = guacdb.get_connection_group_by_id(15)
# Get names by ID
name = guacdb.get_connection_name_by_id(42)
name = guacdb.get_connection_group_name_by_id(15)
name = guacdb.get_usergroup_name_by_id(8)
# Check existence by ID
exists = guacdb.usergroup_exists_by_id(8)
exists = guacdb.connection_group_exists('production')
exists = guacdb.connection_exists('dev-server')# Debug permission issues
guacdb.debug_connection_permissions('dev-server')
# Validate positive IDs
guacdb.validate_positive_id(42, "connection")# List users with their groups
users = guacdb.list_users_with_usergroups()
# List groups with their users and connections
groups = guacdb.list_usergroups_with_users_and_connections()
# List all connections
connections = guacdb.list_connections_with_conngroups_and_parents()
# List connection groups with hierarchy
groups = guacdb.list_connection_groups()
# List groups with users
groups_users = guacdb.list_groups_with_users()# Basic user creation
guacaman user new \
--name john.doe \
--password secretpass
# Create with group memberships (comma-separated)
guacaman user new \
--name john.doe \
--password secretpass \
--usergroup developers,managers,qa # Add to multiple groups
# Note: Will fail if user already existsModifies user's settings or changes password. Allowed parameters are:
| Parameter | Type | Default | Description |
|---|---|---|---|
| access_window_end | time | NULL | End of allowed access time window (HH:MM:SS) |
| access_window_start | time | NULL | Start of allowed access time window (HH:MM:SS) |
| disabled | tinyint | 0 | Whether the user is disabled (0=enabled, 1=disabled) |
| email_address | string | NULL | User's email address |
| expired | tinyint | 0 | Whether the user account is expired (0=active, 1=expired) |
| full_name | string | NULL | User's full name |
| organization | string | NULL | User's organization |
| organizational_role | string | NULL | User's role within the organization |
| timezone | string | NULL | User's timezone (e.g., "America/New_York") |
| valid_from | date | NULL | Date when account becomes valid (YYYY-MM-DD) |
| valid_until | date | NULL | Date when account expires (YYYY-MM-DD) |
guacaman user modify --name john.doe --set disabled=1
guacaman user modify --name john.doe --password newsecretShows all users and their group memberships:
guacaman user listRemoves a user:
guacaman user del --name john.doeCheck user existence (returns 0 if exists, 1 if not):
guacaman user exists --name john.doeguacaman usergroup new --name developersShows all groups and their members:
guacaman usergroup listSupports both name and ID-based deletion:
# Delete by name
guacaman usergroup del --name developers
# Delete by ID
guacaman usergroup del --id 42Supports both name and ID-based existence checking:
# Check by name
guacaman usergroup exists --name developers
# Check by ID
guacaman usergroup exists --id 42Add or remove users from a group with comprehensive ID support:
# Add user to group by name
guacaman usergroup modify --name developers --adduser john.doe
# Remove user from group by name
guacaman usergroup modify --name developers --rmuser john.doe
# Use ID-based group selectors
guacaman usergroup modify --id 42 --adduser john.doe
guacaman usergroup modify --id 42 --rmuser john.doe# Create a VNC connection
guacaman conn new \
--type vnc \
--name dev-server \
--hostname 192.168.1.100 \
--port 5901 \
--password vncpass \
--group developers,qa # Comma-separated list of groups
# Create other types of connections
guacaman conn new --type rdp ...
guacaman conn new --type ssh ... # Basic support availableguacaman conn list
# List specific connection by ID
guacaman conn list --id 42# Change basic connection parameters
guacaman conn modify --name dev-server \
--set hostname=192.168.1.200 \
--set port=5902 \
--set max_connections=5 \
--set max_connections_per_user=2
# Change protocol-specific parameters
guacaman conn modify --name dev-server \
--set color-depth=32 \
--set enable-sftp=true \
--set enable-audio=true
# Grant permission to user
guacaman conn modify --name dev-server --permit john.doe
# Revoke permission from user
guacaman conn modify --name dev-server --deny john.doe
# Set parent connection group
guacaman conn modify --name dev-server --parent "vnc_servers"
# Remove parent connection group
guacaman conn modify --name dev-server --parent ""
# Invalid parameter handling example (will fail)
guacaman conn modify --name dev-server --set invalid_param=valueAvailable Connection Parameters:
The --set option supports 50+ parameters organized by category:
Connection Table Parameters:
protocol- Connection protocol (vnc, rdp, ssh)max_connections- Maximum concurrent connectionsmax_connections_per_user- Max connections per userproxy_hostname- Guacamole proxy hostnameproxy_port- Guacamole proxy portconnection_weight- Load balancing weightfailover_only- Use for failover only
Protocol-Specific Parameters:
- VNC:
hostname,port,password,color-depth,clipboard-encoding - RDP:
hostname,port,username,password,domain,color-depth - SSH:
hostname,port,username,password,private-key,passphrase
Advanced Features:
- SFTP:
enable-sftp,sftp-hostname,sftp-port,sftp-username,sftp-password - Audio:
enable-audio,audio-servername - Session Recording:
recording-path,recording-name - Network:
autoretry,server-alive-interval - Security:
host-key,disable-copy,disable-paste
Supports both name and ID-based existence checking:
# Check by name
guacaman conn exists --name dev-server
# Check by ID
guacaman conn exists --id 42Supports both name and ID-based deletion:
# Delete by name
guacaman conn del --name dev-server
# Delete by ID
guacaman conn del --id 42Connection groups allow you to organize connections into hierarchical groups and manage user permissions for entire groups of connections. Features include cycle detection, atomic operations, and comprehensive permission management.
# Create a top-level group
guacaman conngroup new --name production
# Create a nested group under production
guacaman conngroup new --name vnc_servers --parent productionShows all groups and their hierarchy with database IDs:
guacaman conngroup list
# List specific connection group by ID
guacaman conngroup list --id 15Supports both name and ID-based deletion with automatic cleanup of child references:
# Delete by name
guacaman conngroup del --name vnc_servers
# Delete by ID
guacaman conngroup del --id 42Supports both name and ID-based existence checking:
# Check by name
guacaman conngroup exists --name vnc_servers
# Check by ID
guacaman conngroup exists --id 42Advanced modification with cycle detection, atomic operations, and comprehensive ID support:
Parent Group Management:
# Move group to new parent (with cycle detection)
guacaman conngroup modify --name vnc_servers --parent "infrastructure"
# Remove parent (make top-level)
guacaman conngroup modify --name vnc_servers --parent ""
# Use ID-based selectors
guacaman conngroup modify --id 42 --parent "infrastructure"Connection Management within Groups:
# Add connection to group by name
guacaman conngroup modify --name vnc_servers --addconn-by-name dev-server
# Add connection to group by ID
guacaman conngroup modify --name vnc_servers --addconn-by-id 42
# Remove connection from group by name
guacaman conngroup modify --name vnc_servers --rmconn-by-name dev-server
# Remove connection from group by ID
guacaman conngroup modify --name vnc_servers --rmconn-by-id 42
# Use ID-based group selectors
guacaman conngroup modify --id 15 --addconn-by-name dev-serverAdvanced User Permission Management: The system supports atomic permission operations with comprehensive validation:
# Grant permission to user for connection group using name selector
guacaman conngroup modify --name vnc_servers --permit john.doe
# Grant permission to user for connection group using ID selector
guacaman conngroup modify --id 42 --permit jane.doe
# Revoke permission from user for connection group using name selector
guacaman conngroup modify --name vnc_servers --deny john.doe
# Revoke permission from user for connection group using ID selector
guacaman conngroup modify --id 42 --deny jane.doe
# Multiple users in single operation (repeat flags)
guacaman conngroup modify --name vnc_servers --permit john.doe --permit jane.doeCombined Operations:
# Combine parent modification with permission operations
guacaman conngroup modify --name vnc_servers --parent "infrastructure" --permit john.doe
# Add connection and grant permission in one command
guacaman conngroup modify --name vnc_servers --addconn-by-name dev-server --permit jane.doe
# Complex operation with ID-based selectors
guacaman conngroup modify --id 42 --parent "infrastructure" --addconn-by-id 15 --permit john.doeCheck the installed version:
guacaman versionEnable debug output to see SQL queries and internal state:
guacaman --debug conn list
guacaman --debug user new --name test --password testCheck if a user, group, connection, or connection group exists (returns 0 if exists, 1 if not):
# Check user
guacaman user exists --name john.doe
# Check user group (supports both name and ID)
guacaman usergroup exists --name developers
guacaman usergroup exists --id 42
# Check connection (supports both name and ID)
guacaman conn exists --name dev-server
guacaman conn exists --id 42
# Check connection group (supports both name and ID)
guacaman conngroup exists --name production
guacaman conngroup exists --id 15These commands are silent and only return an exit code, making them suitable for scripting.
Dumps all groups, users and connections in YAML format:
guacaman dumpAll list commands (user list, usergroup list, conn list, conngroup list, dump) output data in YAML-like format. The output includes additional fields that may be useful for scripting and integration.
Connection List Output:
connections:
dev-server:
id: 42
protocol: vnc
parameters:
hostname: 192.168.1.100
port: "5901"
password: "******"
parent: vnc_servers
permissions:
- user: john.doe
permission: READConnection Group List Output:
conngroups:
vnc_servers:
id: 15
parent: production
connections:
- dev-server
- test-serverUser List Output:
users:
john.doe:
groups:
- developers
- qaExample parsing with yq:
# Get all connection names
guacaman conn list | yq '.connections | keys[]'
# Get connections in a specific group
guacaman conngroup list | yq '.conngroups.vnc_servers.connections[]'
# Get user groups
guacaman user list | yq '.users[].groups[]'The $HOME/.guacaman.ini file should contain MySQL connection details:
[mysql]
host = localhost
user = guacamole_user
password = your_password
database = guacamole_dbThe tool provides comprehensive error handling for:
- Database connection issues
- Missing users or groups
- Duplicate entries
- Permission problems
- Invalid configurations
Validation Rules:
- Exactly one of
--nameor--idmust be provided for most operations - IDs must be positive integers greater than 0
- Names cannot be empty strings
- Configuration files must have secure permissions (0600)
- Mutual exclusion validation for conflicting options
- Cycle detection for connection group parent relationships
- Type validation for connection parameters (colors, booleans, integers)
Transaction Handling:
- Connection group modifications use explicit transactions
- Automatic rollback on errors
- Proper error messages with context
- Atomic permission operations to prevent partial state corruption
- Comprehensive input validation before database operations
Advanced Features:
- Connection group cycle detection prevents infinite hierarchies
- Centralized entity resolution with consistent error messages
- Debug mode for troubleshooting (
--debugflag) - Parameter validation based on protocol-specific requirements
All errors are reported with clear messages to help diagnose issues. Use --debug flag to see detailed SQL queries and internal state.
- Database credentials are stored in a separate configuration file
- Configuration file must have strict permissions (0600/-rw-------)
- Script will exit with error code 2 if permissions are too open
- Passwords are properly hashed before storage
- The tool handles database connections securely
- All SQL queries use parameterized statements to prevent SQL injection
- Requires MySQL client access to the Guacamole database
- SSH connections have basic support with limited parameter validation
- Connection creation only supports basic parameters (use
modifyfor advanced parameters)
Current limitations and planned improvements:
-
Separate connection management from user creation ✓
- Implemented in
conncommand:# Create VNC connection guacaman conn new --type vnc --name dev-server --hostname 192.168.1.100 --port 5901 --password somepass # List connections guacaman conn list # Delete connection guacaman conn del --name dev-server
- Implemented in
-
GuacamoleDB initialization without configuration file, via variables ✓
- Environment variables: GUACALIB_HOST, GUACALIB_USER, GUACALIB_PASSWORD, GUACALIB_DATABASE
- Environment variables take precedence over configuration file
- Falls back to config file if environment variables are incomplete
-
Support for other connection types
- RDP (Remote Desktop Protocol)
- SSH (basic support available)
-
User permissions management ✓
- Grant/revoke permissions for individual connections
- Grant/revoke permissions for connection groups
- Support for both name-based and ID-based operations
- Atomic permission operations with rollback
- Advanced validation and cycle detection
- More granular permissions control
- Permission templates
-
Connection parameters management ✓
- Full parameter modification support via
--setoption - Support for all Guacamole connection parameters
- Connection group hierarchy management
- Custom parameters for different connection types (protocol-specific validation)
- Full parameter modification support via
-
Implement dumping RDP connections ✓
- All connection types (VNC, RDP) are included in dump command
- Connection parameters are properly exported in YAML format
PRs implementing any of these features are welcome!
For all AI agents: use rules in AGENTS.md
This is version 0.23
Contributions are welcome! Please feel free to submit a Pull Request.
Copyright Roman V. Isaev rm@isaeff.net 2024-2025, though 95% of the code is written with Aider/DeepSeek/Sonnet/etc
This software is distributed under the terms of the GNU General Public license, version 3.
For bugs, questions, and discussions please use the GitHub Issues.