lmr/duo_auth

Duo Security Universal Prompt two-factor authentication for Roundcube. Supports multiple authentication methods including push notifications, SMS, phone calls, and hardware tokens.

Maintainers

Package info

github.com/leonardomarino/duo_auth

Documentation

Type:roundcube-plugin

pkg:composer/lmr/duo_auth

Statistics

Installs: 388

Dependents: 0

Suggesters: 0

Stars: 4

Open Issues: 0

v2.0.6 2026-04-11 17:11 UTC

This package is auto-updated.

Last update: 2026-04-11 18:28:04 UTC


README

Version License PHP

This is a Roundcube webmail plugin that enables Duo Security Two Factor Authentication using the modern Universal Prompt.

image

It redirects to Duo's secure authentication page after successful username/password authentication, requiring a 2nd Factor of Authentication using Duo Security (push, SMS, call, hardware token code).

🚀 What's New in v2.0.6

  • IPv4-mapped IPv6 normalization - Dual-stack deployments (PHP-FPM/Apache) no longer silently fail CIDR bypass matching when REMOTE_ADDR is presented as ::ffff:x.x.x.x
  • CIDR prefix bounds validation - Invalid prefix lengths (e.g. /200 on an IPv6 range) now return false immediately rather than producing undefined index behavior in the byte loop
  • Tightened startup() callback exemption - The plugin.duo_callback action exemption is now scoped to the login task, closing a surface where a crafted _action parameter on a non-login task could suppress the pending-auth guard
  • Single cleanup ownership - fail_login() is now the sole authoritative caller of cleanup_duo_session() on error paths; redundant calls removed from catch blocks
  • Proactive stale-state removal - duo_state is cleared immediately when the user returns to the login page, rather than persisting until the next protected-task request

INSTALLATION

============

Prerequisites

  • PHP 7.4 or higher (PHP 8.x supported)
  • Roundcube 1.4.0 or higher
  • Composer
  • Duo Security account with Admin API access

Install via Composer

From the root directory of your Roundcube installation:

# Update Composer
composer update

# 1. Allow the plugin installer
composer config allow-plugins.roundcube/plugin-installer true

# 2. Install the plugin
composer require "lmr/duo_auth:^2.0"

# Or for manual installation
cd plugins/
git clone https://github.com/leonardomarino/duo_auth.git
cd duo_auth
composer install

CONFIGURATION

=============

1. Configure Duo Settings

Copy and edit the configuration file:

cp config.inc.php.dist config.inc.php
nano config.inc.php

Update with your Duo credentials from the Duo Admin Panel:

// Required settings
$config['duo_client_id'] = 'YOUR_CLIENT_ID';
$config['duo_client_secret'] = 'YOUR_CLIENT_SECRET';
$config['duo_api_hostname'] = 'api-xxxxxxxx.duosecurity.com';
$config['duo_redirect_uri'] = 'https://your-domain.com/roundcube/?_task=login&_action=plugin.duo_callback';

// Optional: Bypass settings
$config['duo_bypass_users'] = ['service_account'];  // Users who always skip Duo
$config['duo_bypass_ips'] = ['192.168.1.0/24'];    // IPs where everyone skips Duo
$config['duo_bypass_rules'] = [                     // Conditional bypasses
    'admin' => ['127.0.0.1', '::1'],
];

2. Configure PHP Session

Ensure your php.ini has:

session.save_handler = files
session.save_path = "/var/lib/php/sessions"

MIGRATION FROM v1.x

==================

If upgrading from the legacy iframe-based version:

  1. Update Duo Application in Duo Admin Panel for Universal Prompt
  2. Update Configuration Format:
    • Old: duo_auth.conf → New: config.inc.php
    • ikeyduo_client_id
    • skeyduo_client_secret
    • hostduo_api_hostname
  3. Test with duo_failmode = 'open' first
  4. Switch to duo_failmode = 'secure' after verification

FEATURES

========

Three-Tier Bypass System

  1. Global User Bypass - Specific users skip Duo from any location
  2. Global IP Bypass - All users skip Duo from trusted networks
  3. Conditional Bypass - Specific users from specific IPs only

Advanced Options

  • Failmode - Choose behavior when Duo service is unavailable
  • Session Timeout - Configurable Duo session duration
  • Proxy Support - Trust headers from reverse proxies
  • Debug Logging - Detailed logs for troubleshooting

TROUBLESHOOTING

===============

Check Logs

tail -f plugins/duo_auth/logs/duo_auth.log

Test Configuration

cd plugins/duo_auth/
php -l duo_auth.php
composer validate

Common Issues

Problem Solution
500 Error Check PHP error logs, ensure Composer dependencies installed
"Client ID not found" Verify credentials in config.inc.php
Bypass not working Check IP detection, enable debug logging
Session timeout Adjust duo_session_timeout in config

CREDITS

=======

Author: Alexios Polychronopoulos - Original duo_auth for Roundcube

Author: Leonardo Mariño-Ramírez - Updated for Roundcube 1.3.0+ compatibility, v2.0.0 Universal Prompt migration

Author: Johnson Chow - Added IPv4 CIDR matching and user-specific 2FA override

Author: Pavlo Lyha - Rewrote plugin for Duo Web v4 SDK compatibility

LICENSE

=======

This project is licensed under the GPL-3.0-or-later License - see the LICENSE file for details.

SUPPORT

=======

CHANGELOG

=========

v2.0.6 (2026-04-11)

  • IPv4-mapped IPv6 normalization in get_client_ip() for dual-stack CIDR bypass correctness
  • CIDR prefix length bounds checks in ip_in_cidr() (max 32 for IPv4, 128 for IPv6)
  • startup() callback exemption scoped to login task only
  • Proactive duo_state cleanup on login page early-return
  • Redundant cleanup_duo_session() calls removed from callback_handler() catch blocks

v2.0.5 (2026)

  • Security fix: back-button bypass — startup hook enforces Duo completion on every request

v2.0.4 (2025)

  • Complete rewrite for Duo Universal Prompt (Web SDK v4)
  • Added three-tier bypass system
  • IPv4/IPv6 with CIDR support
  • Proxy detection and failmode
  • Configuration moved to config.inc.php
  • PHP 8.2 compatibility

v1.0.9 (2023)

  • Initial Duo Web v4 SDK support by Pavlo Lyha

v1.0.8 (2023)

  • IPv4 CIDR matching by Johnson Chow
  • User-specific 2FA override

v1.0.3 (2020)

  • Roundcube 1.3.0 compatibility by Leonardo Mariño-Ramírez

v1.0.0 (2019)

  • Initial release by Alexios Polychronopoulos

⚠️ Important: The traditional Duo Prompt (iframe-based) reached end of support on March 30, 2024. All installations must use Universal Prompt (SDK v4).