As always the privacyIDEA documentation is work in progress.
Note
Some parts are marked as “(TODO) Not yet implemented”. These are components that have not been migrated from 1.5 to 2.0. If you are missing an important, not-yet-migrated part, drop us a note!
If you are missing any information or descriptions file an issue at github (which would be the preferred way), drop a note to info(@)privacyidea.org or go to the Google group.
This will help us a lot to improve documentation to your needs.
Thanks a lot!
privacyIDEA is a system that is used to manage devices for two factor authentication. In the beginning there were OTP tokens, but other means to authenticate like SSH keys are added. Other concepts like handling of machines or enrolling certificates are coming up, you may monitor this development on Github.
privacyIDEA is a web application written in Python based on the flask micro framework. You can use any webserver with a wsgi interface to run privacyIDEA. E.g. this can be Apache, Nginx or even werkzeug.
A device or item used to authenticate is still called a “token”. All token information is stored in an SQL database, while you may choose, which database you want to use. privacyIDEA uses SQLAlchemy to map the database to internal objects. Thus you may choose to run privacyIDEA with SQLite, MySQL, PostgreSQL, Oracle, DB2 or other database.
The code is divided into three layers, the API, the library and the database layer. Read about it at The Code Documentation. privacyIDEA provides a clean REST API.
Administrators can use a Web UI or a command line client to manage authentication devices. Users can log in to the Web UI to manage their
own tokens.
Authentication is performed via the API or certain plugins for FreeRADIUS, simpleSAMLphp, Wordpress, Contao, Dokuwiki... to either provide default protocols like RADIUS or SAML or to integrate into applications directly.
Due to this flexibility there are also many different ways to install and setup privacyIDEA. We will take a look at common ways to setup privacyIDEA in the section Installation but there are still many others.
The two ways described here to install privacyIDEA are
- the installation via the python package index, which can be used on any Linux distribution and
- ready made debian packages for Ubuntu 14.04LTS.
If you want to upgrade from a privacyIDEA 1.5 installation please read Upgrade From privacyIDEA 1.5.
You can install privacyidea on usually any Linux distribution in a python virtual environment like this:
virtualenv /opt/privacyidea
cd /opt/privacyidea
source bin/activate
Now you are within the python virtual environment. Within the environment you can now run:
pip install privacyidea
Please see the section The Config File for a quick setup of your configuration.
Then create the encryption key and the signing keys:
pi-manage.py create_enckey
pi-manage.py create_signkey
Create the database and the first administrator:
pi-manage.py createdb
pi-manage.py admin add admin admin@localhost
Now you can run the server for your first test:
pi-manage.py runserver
Depending on the database you want to use, you may have to install additional packages.
There are ready made debian packages for Ubuntu 14.04 LTS. These are available in a public ppa repository [1], so that the installation will automatically resolve all dependencies. Install it like this:
add-apt-repository ppa:privacyidea/privacyidea
apt-get update
apt-get install python-privacyidea privacyideaadm
Optionally you can also install necessary configuration files to run privacyIDEA within the Nginx Webserver:
apt-get install privacyidea-nginx
Alternatively you can install privacyIDEA running in an Apache webserver:
apt-get install privacyidea-apache2
After installing in Nginx or Apache2 you only need to create your first administrator and you are done:
pi-manage.py admin add admin admin@localhost
privacyIDEA has a perl module to “translate” RADIUS requests to the API of the privacyIDEA server. This module plugs into FreeRADIUS. The FreeRADIUS does not have to run on the same machine like privacyIDEA. To install this module run:
apt-get install privacyidea-radius
For further details see FreeRADIUS Plugin.
Starting with 1.4 privacyIDEA also supports SAML via a plugin for simpleSAMLphp [2]. The simpleSAMLphp service does not need to run on the same machine like the privacyIDEA server.
To install it on a Ubuntu 14.04 system please run:
apt-get install privacyidea-simplesamlphp
For further details see simpleSAMLphp Plugin.
You can install privacyIDEA on debian Wheezy either via the Python Package Index or with a ready made Wheezy package.
The available Wheezy package privacyidea-venv_2.1~dev0_amd64.deb contains a complete virtual environment with all necessary dependent modules. To install it run:
dpkg -i privacyidea-venv_2.1~dev0_amd64.deb
This will install privacyIDEA into a virtual environment at /opt/privacyidea/privacyidea-venv.
You can enter the virtual environment by:
source /opt/privacyidea/privacyidea-venv/bin/activate
You need to create and fill the config directory /etc/privacyidea manually:
cp /opt/privacyidea/privacyidea-venv/etc/privacyidea/dictionary \
/etc/privacyidea/
Create a config /etc/privacyidea/pi.cfg like this:
# Your database
SQLALCHEMY_DATABASE_URI = 'mysql://pi:password@localhost/pi'
# This is used to encrypt the auth_token
SECRET_KEY = 'choose one'
# This is used to encrypt the admin passwords
PI_PEPPER = "choose one"
# This is used to encrypt the token data and token passwords
PI_ENCFILE = '/etc/privacyidea/enckey'
# This is used to sign the audit log
PI_AUDIT_KEY_PRIVATE = '/etc/privacyidea/private.pem'
PI_AUDIT_KEY_PUBLIC = '/etc/privacyidea/public.pem'
PI_LOGFILE = '/var/log/privacyidea/privacyidea.log'
#CRITICAL = 50
#ERROR = 40
#WARNING = 30
#INFO = 20
#DEBUG = 10
PI_LOGLEVEL = 20
You need to create the above mentioned logging directory /var/log/privacyidea.
You need to create the above mentioned database with the corresponding user access:
mysql -u root -p -e "create database pi"
mysql -u root -p -e "grant all privileges on pi.* to 'pi'@'localhost' \
identified by 'password'"
With this config file in place you can create the database tables, the encryption key and the audit keys:
pi-manage.py createdb
pi-manage.py create_enckey
pi-manage.py create_audit_keys
Now you can create the first administrator:
pi-manage.py admin add administrator email@domain.tld
The system is set up. You now only need to configure the Apache2 webserver.
The Apache2 needs a wsgi script that could be located at /etc/privacyidea/piapp.wsgi and look like this:
import sys
sys.stdout = sys.stderr
from privacyidea.app import create_app
# Now we can select the config file:
application = create_app(config_name="production", \
config_file="/etc/privacyidea/pi.cfg")
Finally you need to create a Apache2 configuration /etc/apache2/sites-available/privacyidea.conf which might look like this:
WSGIPythonHome /opt/privacyidea/privacyidea-venv
<VirtualHost _default_:443>
ServerAdmin webmaster@localhost
# You might want to change this
ServerName localhost
DocumentRoot /var/www
<Directory />
# For Apache 2.4 you need to set this:
# Require all granted
Options FollowSymLinks
AllowOverride None
</Directory>
# We can run several instances on different paths with different configurations
WSGIScriptAlias / /etc/privacyidea/piapp.wsgi
#
# The daemon is running as user 'privacyidea'
# This user should have access to the encKey database encryption file
WSGIDaemonProcess privacyidea processes=1 threads=15 display-name=%{GROUP} user=privacyidea
WSGIProcessGroup privacyidea
WSGIPassAuthorization On
ErrorLog /var/log/apache2/error.log
LogLevel warn
LogFormat "%h %l %u %t %>s \"%m %U %H\" %b \"%{Referer}i\" \"%{User-agent}i\"" privacyIDEA
CustomLog /var/log/apache2/ssl_access.log privacyIDEA
# SSL Engine Switch:
# Enable/Disable SSL for this virtual host.
SSLEngine on
# If both key and certificate are stored in the same file, only the
# SSLCertificateFile directive is needed.
SSLCertificateFile /etc/ssl/certs/privacyideaserver.pem
SSLCertificateKeyFile /etc/ssl/private/privacyideaserver.key
<FilesMatch "\.(cgi|shtml|phtml|php)$">
SSLOptions +StdEnvVars
</FilesMatch>
<Directory /usr/lib/cgi-bin>
SSLOptions +StdEnvVars
</Directory>
BrowserMatch ".*MSIE.*" \
nokeepalive ssl-unclean-shutdown \
downgrade-1.0 force-response-1.0
</VirtualHost>
The configuration assumes, a user privacyidea, which you need to create:
useradd -r -m privacyidea
The files in /etc/privacyidea and the logfiles in /var/log/privacyidea/ should be restricted to this user.
Footnotes
[1] | https://launchpad.net/~privacyidea |
[2] | https://github.com/privacyidea/privacyidea/tree/master/authmodules/simpleSAMLphp |
Warning
privacyIDEA 2.0 introduces many changes in database schema, so at least perform a database backup!
Be sure to stop your privacyIDEA server.
To upgrade the code enter your python virtualenv and run:
pip install --upgrade privacyidea
Read about the configuration in the The Config File.
You can use the old enckey, the old signing keys and the old database uri. The values can be found in your old ini-file as privacyideaSecretFile, privacyideaAudit.key.private, privacyideaAudit.key.public and sqlalchemy.url. Your new config file might look like this:
config_path = "/home/cornelius/tmp/pi20/etc/privacyidea/"
# This is your old database URI
# Note the three slashes!
SQLALCHEMY_DATABASE_URI = "sqlite:///" + config_path + "token.sqlite"
# This is new!
SECRET_KEY = 't0p s3cr3t'
# This is new
#This is used to encrypt the admin passwords
PI_PEPPER = "Never know..."
# This is used to encrypt the token data and token passwords
# This is your old encryption key!
PI_ENCFILE = config_path + 'enckey'
# THese are your old signing keys
# This is used to sign the audit log
PI_AUDIT_KEY_PRIVATE = config_path + 'private.pem'
PI_AUDIT_KEY_PUBLIC = config_path + 'public.pem'
To verify the new configuration run:
pi-manage.py create_enckey
It should say, that the enckey already exists!
You need to upgrade the database to the new database schema:
pi-manage.py db upgrade -d lib/privacyidea/migrations
Note
In the Ubuntu package the migrations folder is located at /usr/lib/privacyidea/migrations/.
With privacyIDEA 2.0 the administrators are stored in the database. The password of the administrator is salted and also peppered, to avoid having a database administrator slip in a rogue password.
You need to create new administrator accounts:
pi-manage.py addadmin <email-address> <admin-name>
Run the server:
pi-manage.py runserver
or add it to your Apache or Nginx configuration.
privacyIDEA reads its configuration from different locations:
- default configuration from the module privacyidea/config.py
- then from the config file /etc/privacyidea/pi.cfg if it exists and then
- from the file specified in the environment variable PRIVACYIDEA_CONFIGFILE.
The configuration is overwritten and extended in each step. I.e. values define in privacyidea/config.py that are not redefined in one of the other config files, stay the same.
You can create a new config file (either /etc/privacyidea/pi.cfg) or any other file at any location and set the environment variable. The file should contain the following contents:
# The realm, where users are allowed to login as administrators
SUPERUSER_REALM = ['super', 'administrators']
# Your database
SQLALCHEMY_DATABASE_URI = 'sqlite:////etc/privacyidea/data.sqlite'
# This is used to encrypt the auth_token
SECRET_KEY = 't0p s3cr3t'
# This is used to encrypt the admin passwords
PI_PEPPER = "Never know..."
# This is used to encrypt the token data and token passwords
PI_ENCFILE = '/etc/privacyidea/enckey'
# This is used to sign the audit log
PI_AUDIT_KEY_PRIVATE = '/home/cornelius/src/privacyidea/private.pem'
PI_AUDIT_KEY_PUBLIC = '/home/cornelius/src/privacyidea/public.pem'
# PI_LOGFILE = '....'
# PI_LOGLEVEL = 20
Note
The config file is parsed as python code, so you can use variables to set the path and you need to take care for indentations.
If you are using a config file other than /etc/privacyidea/pi.cfg you need to set the environment variable:
export PRIVACYIDEA_CONFIGFILE=/your/config/file
The SUPERUSER_REALM is a list of realms, in which the users get the role of an administrator.
Apache2 and Nginx are using a WSGI script to start the application.
This script is usually located at /etc/privacyidea/privacyideaapp.py or /etc/privacyidea/privacyideaapp.wsgi and has the following contents:
import sys
sys.stdout = sys.stderr
from privacyidea.app import create_app
# Now we can select the config file:
application = create_app(config_name="production",
config_file="/etc/privacyidea/pi.cfg")
In the create_app-call you can also select another config file.
Note
This way you can run several instances of privacyIDEA in one Apache2 server by defining several WSGIScriptAlias definitions pointing to different wsgi-scripts, that again reference different config files with different database definitions.
privacyIDEA has one login form for users to login and for administrators to login.
The login screen can contain a dropdown box with the list of existing realms. The dropdown box can be configured in the System Config dialog on the GUI settings tab.
You should enter your username and choose the right realm. If the dropdown box is not displayed, then you need to append the realm to the username like username@realm.
Normal users authenticate at the login form to be able to manage their own tokens. By default users need to authenticate with the password from their user source.
E.g. if the users are located in an LDAP or Active Directory the user needs to authenticate with his LDAP/AD password.
Note
The user my either login with his password from the userstore or with any of his tokens.
Note
The administrator may change this behaviour by creating an according policy, which then requires the user to authenticate against privacyIDEA itself. I.e. this way the user needs to authenticate with a second factor/token to access the self service portal. (see the policy section login_mode)
Administrators can authenticate at this login form to access the management UI.
Administrators are stored in the database table Admin and can be managed with the tool:
pi-manage.py admin ...
The administrator just logs in with his username.
Note
You can configure privacyIDEA to authenticate administrators against privacyIDEA itself, so that administrators need to login with a second factor. See SUPERUSER_REALM in inifile_superusers how to do this.
The configuration menu can be used to define useridresolvers and realms, set the system config and the token config.
It also contains a shortcut to the policy tab (see Policies).
Each organisation or company usually has its users managed at a central location. This is why privacyIDEA does not provide its own user management but rather connects to existing user stores.
UserIdResolvers are connectors to those user stores, the locations, where the users are managed. Nowadays this can be LDAP directories or especially Active Directory, some times FreeIPA or the Redhat 389 service. But classically users are also located in files like /etc/passwd on standalone unix systems. Web services often use SQL databases as user store.
Today with many more online cloud services SCIM is also an uprising protocol to access userstores.
privacyIDEA already comes with UserIdResolvers to talk to all these user stores:
- Flatfile resolver,
- LDAP resolver,
- SQL resolver,
- SCIM resolver. (TODO): Not yet migrated.
Note
New resolver types (python modules) can be added easily. See the module section for this (UserIdResolvers).
You can create as many UserIdResolvers as you wish and edit existing resolvers. When you have added all configuration data, most UIs of the UserIdResolvers have a button “Test resolver”, so that you can test your configuration before saving it.
Note
Using the policy authentication:otppin=userstore users can authenticate with the password from their user store, being the LDAP password, SQL password or password from flat file.
Flatfile resolvers read files like /etc/passwd.
Note
The file /etc/passwd does not contain the unix password. Thus, if you create a flatfile resolver from this file the functionality with otppin=userstore is not available. You can create a flatfile with passwords using the tool privacyidea-create-pwidresolver-user.
Create a flat file like this:
privacyidea-create-pwidresolver-user -u user2 -i 1002 >> /your/flat/file
The LDAP resolver can be used to access any kind of LDAP service like OpenLDAP, Active Directory, FreeIPA, Penrose, Novell eDirectory.
In case of Active Directory connections you might need to check the box No anonymous referral chasing. The underlying LDAP library is only able to do anonymous referral chasing. Active Directory will produce an error in this case [1].
The LoginName attribute is the attribute that holds the loginname. It can be changed to your needs.
The searchfilter and the userfilter are used for forward and backward search the object in LDAP.
The searchfilter is used to list all possible users, that can be used in this resolver.
The userfilter is used to find the LDAP object for a given loginname. This is why the userfilter contains the python string replacement parameter %s, which will be filled with the given loginname to find the LDAP object.
The attribute mapping maps LDAP object attributes to user attributes in privacyIDEA. privacyIDEA knows the following attributes:
- username,
- phone,
- mobile,
- email,
- surname,
- givenname.
The UID Type is the unique identifier for the LDAP object. If it is left blank, the distinguished name will be used. In case of OpenLDAP this can be entryUUID and in case of Active Directory objectGUID.
The SQL resolver can be used to retrieve users from any kind of SQL database like MySQL, PostgreSQL, Oracle, DB2 or sqlite.
In the upper frame you need to configure the SQL connection. The SQL resolver uses SQLAlchemy internally. In the field Driver you need to set a driver name as defined by the SQLAlchemy dialects like “mysql” or “postgres”.
In the SQL attributes frame you can specify how the users are identified.
The Database table contains the users.
Note
At the moment only one table is supported, i.e. if some of the user data like email address or telephone number is located in a second table, those data can not be retrieved.
The Limit is the SQL limit for a userlist request. This can be important if you have several thousand user entries in the table.
The Attribute mapping defines which table column should be mapped to which privayIDEA attribute. The known attributes are:
- userid,
- username,
- phone,
- mobile,
- email,
- givenname,
- surname.
You can add an additional Where statement if you do not want to use all users from the table.
Note
The Additional connection parameters refer to the SQLAlchemy connection but are not used at the moment.
SCIM is a “System for Cross-domain Identity Management”. SCIM is a REST-based protocol that can be used to ease identity management in the cloud.
The SCIM resolver is tested in basic functions with OSIAM [2], the “Open Source Idenitty & Access Management”.
To connect to a SCIM service you need to provide a URL to an authentication server and a URL to the resource server. The authentication server is used to authenticate the privacyIDEA server. The authentication is based on a client name and the Secret for this client.
Userinformation is then retrieved from the resource server.
The available attributes for the Attribute mapping are:
- username,
- givenname,
- surname,
- phone,
- mobile,
- email.
Footnotes
[1] | http://blogs.technet.com/b/ad/archive/2009/07/06/referral-chasing.aspx |
[2] | http://www.osiam.org |
Users need to be in realms to have tokens assigned. A user, who is not member of a realm can not have a token assigned and can not authenticate.
You can combine several different UserIdResolvers (see UserIdResolvers) into a realm. The system knows one default realm. Users within this default realm can authenticate with their username.
Users in realms, that are not the default realm, need to be additionally identified. Therefor the users need to authenticate with their username and the realm like this:
user@realm
The realms dialog gives you a list of the already defined realms.
It shows the name of the realms, whether it is the default realm and the names of the resolvers, that are combined to this realm.
You can delete or edit an existing realm or create a new realm.
Each realm has to have a unique name. The name of the realm is case insensitive. If you create a new realm with the same name like an existing realm, the existing realm gets overwritten.
If you click Edit Realm you can select which userresolver should be contained in this realm. A realm can contain several resolvers.
If you have a fresh installation, no resolver and no realm is defined. To get you up and running faster, the system will ask you, if it should create the first realm for you.
If you answer “yes”, it will create a resolver named “deflocal” that contains all users from /etc/passwd and a realm named “defrealm” with this very resolver.
Thus you can immediately start assigning and enrolling tokens.
If you check “Do not ask again” this will be stored in a cookie in your browser.
Note
The realm “defrealm” will be the default realm. So if you create a new realm manually and want this new realm to be the default realm, you need to set this new realm to be default manually.
The system configuration has three logical topics: Settings, token default settings and GUI settings.
splitAtSign defines if the username like user@company given during authentication should be split into the loginname user and the realm name company. In most cases this is the wanted behaviour.
But given your users log in with email addresses like user@gmail.com and otheruser@outlook.com you probably do not want to split.
Return SAML attributes defines if during an SAML authentication request additional SAML attributes should be returned. Usuall an authentication response only returns true or false.
If FailCounterIncOnFalsePin is set the failcounter of all tokens of a user will be increased. When given a false PIN the system can not identify the token the user wants to login with. So it either increases the failcounter of all tokens of this user or none.
PrependPin defines if the OTP PIN should be given in front (“pin123456”) or in the back (“12345pin”) of the OTP value.
Auto resync defines if the system should try to resync a token if a user provides a wrong OTP value. If checked, the system remembers the OTP value and if during auto resync timeout the user tries to authenticate again with the next, successive OTP value, the system tries to resync this token with the two given OTP values.
Pass on user not found let the system return a successful authentication response if the authenticating user does not exist in the system.
Warning
Use with care and only if you know what you are doing!
Pass on user no token let the system return a successful authentication response if the authenticating user exists in the system but has no token assigned.
Warning
Use with care and only if you know what you are doing! Since the user could remove all his tokens in selfservice and then have free rides forever.
Override Authentication client is important with client specific policies (see Policies) and RADIUS servers. In case of RADIUS the authenticating client for the privacyIDEA system will always be the RADIUS serve, which issues the authentication request. But you can allow the RADIUS server IP to send another client information (in this case the RADIUS client) so that the policy is evaluated for the RADIUS client. This field takes a comma seperated list of IP addresses.
maximum concurrent OCRA challenges defines how many OCRA requests for a single OCRA token are allowed to be active simultaniously. (TODO): Not migrated, yet.
OCRA challenge timeout defines how many seconds an OCRA challenge is kept active. The response must be sent within this timeout. (TODO): Not migrated, yet.
DefaultResetFailCount will reset the failcounter of a token if this token was used for a successful authentication. If not checked, the failcounter will not be resetted and must be resetted manually.
Note
The following settings are token specific value which are set during enrollment. If you want to change this value of a token lateron, you need to change this at the tokeninfo dialog.
DefaultMaxFailCount is the maximum failcounter a token way get. If the failcounter exceeds this number the token can not be used unless the failcounter is resetted.
Note
In fact the failcounter will only increas till this maxfailcount. Even if more failed authentication request occur, the failcounter will not increase anymore.
DefaultSyncWindow is the window how many OTP values will be caluculated during resync of the token.
DefaultOtpLen is the length of the OTP value. If no OTP lenght is specified during enrollment, this value will be used.
DefaultCountWindow defines how many OTP values will be calculated during an authentication request.
DefaultChallengeValidityTime is the timeout for a challenge response authentication.
default OCRA suite is the OCRA suite that is set for an OCRA token during enrollment if no OCRA suite is specified. (TODO): Not migrated, yet.
default QR suite is the OCRA suite that is set for a QR token during enrollment if no OCRA suite is specified. (TODO): Not migrated, yet.
The login window of the WebUI may display a dropdown box with all realms. You might hide this dropdown box, if you do not want to tell the world which realms are defined on your system. If you check display realm select box the list of all realms including the special realm admin for the administrators from the superuser file will be displayed in the login form.
Each token type can provide its own configuration dialog.
The SMS OTP token creates a OTP value and sends this OTP value to the mobile phone of the user. The SMS can be triggered by an API call or by authenticating with only the OTP PIN:
In the first step the user will enter his OTP PIN an the sending of the SMS is triggered. The user is denied the access.
In the second step the user authenticates with the OTP PIN and the OTP value he received via SMS. The user is granted access.
A python SMS provider module defines how the SMS is sent. This can be done using an HTTP SMS Gateway. Most services like Clickatel or sendsms.de provide such a simple HTTP gateway. Another possibility is to send SMS via sipgate, which provides an XMLRPC API. The third possibility is to send the SMS via an SMTP gateway. The proovider receives a specially designed email and sends the SMS accordingly. The last possibility to send SMS is to use an attached GSM modem.
In the field SMS provider you can enter the SMS provider module, you wish to use. In the empty field hit the arrow-down key and you will get a list of the ready made modules.
In the SMS configuration text area you can enter the configuration, which contents is very much dependant on the selected provider module.
The HTTP and the Sipgate module provide a preset-button, which give you an idea of the configuration.
The HTTP provider can be used for any SMS gateway that provides a simple HTTP POST or GET request.
The following parameters can be used:
URL
This is the URL for the gateway.
HTTP_Method
Can be GET or POST.
USERNAME and PASSWORD
These are the username and the password if the HTTP request requires basic authentication.
SMS_PHONENUMBER_KEY
This is the name of the HTTP parameter that holds the mobile phone number of the recipient.
SMS_TEXT_KEY
This is the name of the HTTP parameter that holds the SMS text.
RETURN_SUCCESS
You can either use RETURN_SUCCESS or RETURN_FAIL. If the text of RETURN_SUCCESS is found in the HTTP response of the gateway privacyIDEA assumes that the SMS was sent successfully.
RETURN_FAIL
If the text of RETURN_FAIL is found in the HTTP response of the gateway privacyIDEA assumes that the SMS could not be sent and an error occurred.
PROXY
You can specify a proxy to connect to the HTTP gateway.
PARAMETER
This can contain a dictionary of arbitrary fixed additional parameters. Usually this would also contain an ID or a password to identify you as a sender.
In case of the Clicaktell provider the configuration will look like this:
{ "URL" : "http://api.clickatell.com/http/sendmsg",
"PARAMETER" : {
"user":"YOU",
"password":"YOUR PASSWORD",
"api_id":"YOUR API ID"
},
"SMS_TEXT_KEY":"text",
"SMS_PHONENUMBER_KEY":"to",
"HTTP_Method":"GET",
"RETURN_SUCCESS" : "ID"
}
This will consturct an HTTP GET request like this:
http://api.clickatell.com/http/sendmsg?user=YOU&password=YOU&\
api_id=YOUR API ID&text=....&to=....
where text and to will contain the OTP value and the mobile phone number. privacyIDEA will assume a successful sent SMS if the response contains the text “ID”.
The sipgate provider connects to https://samurai.sipgate.net/RPC2 and takes only two arguments USERNAME and PASSWORD. The arguments have to be passed in a dictionary like this:
{ "USERNAME" : "youruser",
"PASSWORD" : "yourpassword" }
Note
You need to use double quotes around the values.
If you activate debug log level you will see the submitted SMS and the response content from the Sipgate gateway.
(TODO): Not yet migrated.
Starting with 1.3.3 privacyIDEA comes with a graphical setup tool to manage your token administrators and RADIUS clients. Thus you will get a kind of appliance experience. To install all necessary components read appliance.
To configure the system, login as the user root on your machine and run the command:
privacyidea-setup-tui
This will bring you to this start screen.
You can configure privacyidea settings, the log level, administrators, encryption key and much more. You can configure the webserver settings and RADIUS clients.
All changes done in this setup tool are directly read from and written to the corresponding configuration files. The setup tool parses the original nginx and freeradius configuration files. So there is no additional place where this data is kept.
Note
You can also edit the clients.conf and other configuration files manually. The setup tool will also read those manual changes!
Starting with version 1.5 the setup tool also supports backup and restore. Backups are written to the directory /var/lib/privacyidea/backup.
The backup contains all privacyIDEA configuration, the contents of the directory /etc/privacyidea, the encryption key, the configured administrators, the complete token database (MySQL) and Audit log. Furthermore if you are running FreeRADIUS the backup also contains the /etc/freeradius/clients.conf file.
At the configuration point Configure Backup you can define times when a scheduled backup should be performed. This information is written to the file /etc/crontab.
You can enter minutes, hours, day of month, month and day of week. If the entry should be valid for each e.g. month or hour, you need to enter a ‘*’.
In this example the 10 17 * * * (minute=10, hour=17) means to perform a backup each day and each month at 17:10 (5:10pm).
The example 1 10 1 * * (minute=1, hour=10, day of month=1) means to perform a backup on the first day of each month at 10:01 am.
Thus you could also perform backups only once a week at the weekend.
If you want to run a backup right now you can choose the entry Backup now.
The entry View Backups will list all the backups available.
You can select a backup and you are asked if you want to restore the data.
Warning
Existing data is overwritten and will be lost.
The administrator can see all the tokens of all realms he is allowed to manage in the tokenview. Each token can be located in several realms and be assigned to one user. The administrator can see all the details of the token.
The administrator can click on one token, to show more details of this token and to perform actions on this token.
The Token Details give you more information about the token and also let the administrator perform specific tasks for this token.
At the bottom you see the assigned user. You can click on the username and change to the User Details.
(TODO): Not migrated, yet.
When a user has lost a token, the administrator can create a temporary password token for the user to login.
The administrator has to select the token that was lost and click the button Lost token. A new token of type PW is generated. The OTP PIN of the old token is automatically copied to the new token. Thus the administrator does not know the OTP PIN, while the user can use his old PIN.
A long password is displayed to the administrator and the administrator can read this password to the user. The user now can authenticate with his old OTP PIN and the long password.
The lost token is deactivated.
You can change the following token settings.
MaxFail and FailCount
If the login fail counter reaches the MaxFail the user can not login with this token anymore. The Failcounter FailCount has to be reset to zero.
TokenDesc
The token description is also displayed in the tokenview. You can set a description to make it easier to identify a token.
CountWindow
The CountWindow is the look ahead window of event based tokens. If the user pressed the button on an event based token the counter in the token is increased. If the user does not use this otp value to authenticate, the server does not know, that the counter in the token was increased. This way the counter in the token can get out of sync with the server.
SyncWindow
If a token was out of sync (see CountWindow), then it needs to be synchronized. This is done by entering two consecutive OTP values. The server searches these two values within the next CountWindow (default 1000) values.
OtpLen
This is the length of the OTP value that is generated by the token. The password that is entered by the user is splitted according to this length. 6 or 8 characters are splitted as OTP value and the rest is used as static password (OTP PIN).
Hashlib
The HOTP algorith can be used with SHA1 or SHA256.
Auth max
The administrator can set a value how often this token may be used for authentication. If the number of authentication try exceed this value, the token can not be used, until this Auth max value is increased.
Note
This way you could create tokens, that can be used only once.
Auth max success
The administrator can set a value how often this token may be used to successfully authenticate.
Valid start
A timestamp can be set. The token will only be usable for authentication after this start time.
Valid end
A timestamp can be set. The token can only be used before this end time.
Note
This way you can create temporary tokens for guests or short time or season employees.
The administrator can select one token and then enter two consecutive OTP values to resynchonize the token if it was out of sync.
A token can be assigned to several realms. This is important if you have administrators for different realms. A realm administrator is only allowed to see tokens within his realms. He will not see tokens, that are not in his realm. So you can assign a token to realm A and realm B, thus the administrator A and the administrator B will be able to see the token.
If the corresponding getOTP policy (Policies) is set, the administrator can get the OTP values of a token from the server without having the token with him.
Note
Of course this is a potential backdoor, since the administrator could login as the user/owner of this very token.
For this function you do not need to select any token. But you can select a user on the userview, then the token to be enrolled will be directly assigend to this user.
When enrolling the token, you can choose, which token type you want to enroll. After enrolling the token, you can set a OTP PIN for this token.
This function is used to assign a token to a user. Select a realm and start typing a username to find the user, to whom the token should be assigned.
In the token details view you can unassign the token. After that, the token can be assigned to a new user.
If a token is disabled, it can be enabled again.
Tokens can be disabled. Disabled tokens still belong to the assigned user but those tokens can not be used to authenticate. Disabled tokens can be enabled again.
You can set the OTP PIN or the mOTP PIN for tokens.
If a used locked his token, since he entered wrong OTP values or wrong OTP PINs, the fail counter has reached the mail failcount. The administrator or help desk user can select those tokens and click the button reset failcounter to reset the fail counter to zero. The tokens can be used for authentication again.
Selecting tokens and clicking the button delete will remove the tokens from the database. The token information can not be recovered. But all events that occurred with this token still remain in the audit log.
The administrator can see all users in realms he is allowed to manage.
Note
Users are only visible, if the useridresolver is located within a realm. If you only define a useridresolver but no realm, you will not be able to see the users!
You can select one of the realms in the left drop down box. The administrator will only see the realms in the drop down box, that he is allowed to manage. (TODO) No migrated, yet.
The list shows the users from the select realm. The username, surname, given name, email and phone are filled according to the definition of the useridresolver.
Even if a realm contains several useridresolvers all users from all resolvers within this realm are displayed.
As privacyIDEA only reads users from user sources the actions you can perform on the users are limited.
Note
You can not create or modify users in privacyIDEA!
When clicking on a username, you can see the users details and perform several actions on the user.
You see a list of the users tokens and change to the Token Details.
In the users details view you can enroll additional tokens to the user. In the enrollment dialog the user will be selected and you only need to choose what tokentype you wish to enroll for this user.
You can assign a new, already existing token to the user. Just start typing the token serial number. The system will search for tokens, that are not assigned yet and present you a list to choose from.
Policies can be used to define the reaction and behaviour of the system.
Each policy defines the behaviour in a certain area, called scope. privacyIDEA knows the scopes:
Admin policies are used to regulate the actions that administrators are allowed to do. Technically admin policies control the use of the REST API Token endpoint, System endpoints, Realm endpoints and Resolver endpoints.
Admin policies are implemented as decorators in Policy Module and Policy Decorators.
The user in the admin policies refers to the administrator.
All administrative actions also refer to the defined realm. Meaning an administrator may have many rights in one realm and only a few rights in another realm.
Creating a policy with scope:admin, user:frank, action:enable and realm:sales means that the administrator Frank is allowed to enable tokens in the realm sales.
Note
As long as no admin policy is defined all administrators are allowed to do everything.
The following actions are available in the scope admin:
type: bool
There are init actions per token type. Thus you can create policy that allow an administrator to enroll SMS tokens but not to enroll HMAC tokens.
type: bool
Tokens can be enabled and disabled. Disabled tokens can not be used to authenticate. The disable action allows the administrator to disable tokens.
type: bool
Tokens can have additional token information, which can be viewed in the Token Details.
If the set action is defined, the administrator allowed to set those token information.
type: bool
If the setOTPPIN action is defined, the administrator is allowed to set the OTP PIN of a token.
type: bool
If the setMOTPPIN action is defined, the administrator is allowed to set the mOTP PIN of an mOTP token.
type: bool
If the resync action is defined, the administrator is allowed to resynchronize a token.
type: bool
If the assign action is defined, the administrator is allowed to assign a token to a user. This is used for assigning an existing token to a user but also to enroll a new token to a user.
Without this action, the administrator can not create a connection (assignment) between a user and a token.
type: bool
If the unassign action is defined, the administrator is allowed to unassign tokens from a user. I.e. the administrator can remove the link between the token and the user. The token still continues to exist in the system.
type: bool
If the import action is defined, the administrator is allowed to import token seeds from a token file, thus creating many new token objects in the systems database.
type: bool
If the remove action is defined, the administrator is allowed to delete a token from the system.
Note
If a token is removed, it can not be recovered.
Note
All audit entries of this token still exist in the audit log.
type: bool
If the userlist action is defined, the administrator is allowed to view the user list in a realm. An administrator might not be allowed to list the users, if he should only work with tokens, but not see all users at once.
Note
If an administrator has any right in a realm, the administrator is also allowed to view the token list.
type: bool
If the checkstatus action is defined, the administrator is allowed to check the status of open challenge requests.
type: bool
If the manageToken action is defined, the administrator is allowed to manage the realms of a token.
A token may be located in multiple realms. This can be interesting if you have a pool of spare tokens and several realms but want to make the spare tokens available to several realm administrators. (Administrators, who have only rights in one realm)
Then all administrators can see these tokens and assign the tokens. But as soon as the token is assigned to a user in one realm, the administrator of another realm can not manage the token anymore.
(TODO) Not yet migrated.
type: bool
If the getserial action is defined, the administrator is allowed to calculate the token serial number for a given OTP value.
(TODO) Not yet migrated.
type: bool
If the copytokenuser action is defined, the administrator is allowed to copy the user assignment of one token to another.
This functionality is also used during the lost token process. But you only need to define this action, if the administrator should be able to perform this task manually.
(TODO) Not yet migrated.
type: bool
If the copytokenpin action is defined, the administrator is allowed to copy the OTP PIN from one token to another without knowing the PIN.
This functionality is also used during the lost token process. But you only need to define this action, if the administrator should be able to perform this task manually.
(TODO) Not yet migrated.
type: bool
If the losttoken action is defined, the adminstrator is allowed to perform the lost token process.
To only perform the lost token process the actions copytokenuser and copytokenpin are not necessary!
(TODO) Not yet migrated.
type: bool
If the getserial action is defined, the administrator is allowed to retrieve OTP values for a given token.
In the Web UI users can manage their own tokens. User can login to the Web UI with the username of their useridresolver. I.e. if a user is found in an LDAP resolver pointing to Active Directory the user needs to login with his domain password.
User policies are used to define, which actions users are allowed to perform.
The user policies also respect the client input, where you can enter a list of IP addresses and subnets (like 10.2.0.0/16).
Using the client parameter you can allow different actions in if the user either logs in from the internal network or remotely from the internet via the firewall.
Technically user policies control the use of the REST API Token endpoint and are checked using Policy Module and Policy Decorators.
Note
If no user policy is defined, the user has all actions available to him, to manage his tokens.
The following actions are available in the scope user:
type: bool
There are enroll actions per token type. Thus you can create policies that allow the user to enroll SMS tokens but not to enroll HMAC tokens.
type: bool
The user is allowed to assgin an existing token, that is located in his realm and that does not belong to any other user, by entering the serial number.
type: bool
The user is allowed to disable his own tokens. Disabled tokens can not be used to authenticate.
type: bool
The user is allowed to delete his own tokens from the database. Those tokens can not be recovered. Anyway, the audit log concerning these tokens remains.
type: bool
The user is allowed to drop his ownership of the token. The token does not belong to any user anymore and can be reassigned.
type: bool
The user is allowed to resynchronize the token if it has got out of synchronization.
(TODO): not yet migrated.
type: bool
The user is allowed to retrieve OTP values from a token.
(TODO): not yet migrated.
type: integer
range: 0 - 100
This is the maximum allowed PIN length the user is allowed to use when setting the OTP PIN.
(TODO): not yet migrated.
type: integer
range: o - 100
This is the minimum required PIN the user must use when setting the OTP PIN.
(TODO): not yet migrated.
type: string
contents: cnso+-
This defines what characters an OTP PIN should contain when the user sets it.
c are letters matching [a-zA-Z].
n are digits mathcing [0-].
s are special characters matching [.:,;-_<>+*!/()=?$§%&#~^]
o are other characters.
Note
You can change these character definitions in the privacyidea.ini file using privacyideaPolicy.pin_c, privacyideaPolicy.pin_n and privacyideaPolicy.pin_s.
Example: The policy action otp_pin_contents=cn, otp_pin_minlength=8 would require the user to choose OTP PINs that consist of letters and digits which have a minimum length of 8.
The logic of the otp_pin_contents can be enhanced and reversed using the characters + and -.
-cn would still mean, that the OTP PIN needs to contain letters and digits and it must not contain any other characters.
cn
test1234 and test12$$ would be valid OTP PINs. testABCD would not be a valid OTP PIN.
-cn
test1234 would be a valid OTP PIN, but test12$$ and testABCS would not be valid OTP PINs. The later since it does not contain digits, the first (test12$$) since it does contain a special character ($), which it should not.
+cn combines the two required groups. I.e. the OTP PIN should contain characters from the sum of the two groups.
test1234, test12$$, test and 1234 would all be valid OTP PINs.
(TODO): not yet migrated.
type: integer
This works together with the getotp action. This is the maximum number of OTP values the user may retrieve from DPW tokens.
(TODO): not yet migrated.
type: integer
This works together with the getotp action. This is the maximum number of OTP values the user may retrieve from HOTP tokens.
(TODO): not yet migrated.
type: integer
This works together with the getotp action. This is the maximum number of OTP values the user may retrieve from TOTP tokens.
type: bool
This action allows the user to view and search the audit log for actions with his own tokens.
(TODO): not yet migrated.
type: bool
This action allows the user to search for the serial number of an unassigned token by entering an OTP value.
The scope authentication gives you more detailed possibilities to authenticate the user or to define what happens during authentication.
Technically the authentication policies apply to the REST API Validate endpoints and are checked using Policy Module and Policy Decorators.
The following actions are available in the scope authentication:
type: string
This action defines how the fixed password part during authentication should be validated. Each token has its own OTP PIN, but you can choose how the authentication should be processed:
otppin=tokenpin
This is the default behaviour. The user needs to pass the OTP PIN concatenated with the OTP value.
otppin=userstore
The user needs to pass the user store password concatenated with the OTP value. It does not matter if the OTP PIN is set or not. If the user is located in an Active Directory the user needs to pass his domain password together with the OTP value.
Note
The domain password is checked with an LDAP bind right at the moment of authentication. So if the user is locked or the password was changed authentication will fail.
otppin=none
The user does not have to pass any fixed password. Authentication is only done via the OTP value.
type: bool
If the user has no token assigned, he will be authenticated against the UserIdResolver, i.e. he needs to provide the LDAP- or SQL-password.
Note
This is a good way to do a smooth enrollment. Users having a token enrolled will have to use the token, users not having a token, yet, will be able to authenticate with their domain password.
Warning
If the user has the right to delete his tokens in selfservice portal, the user could delete all his tokens and then authenticate with his static password again.
type: bool
If the user has no token assigned an authentication request for this user will always be true.
Warning
Only use this if you know exactly what you are doing.
type: bool
If the user does not exist, the authentication request is successful.
Warning
Only use this if you know exactly what you are doing.
type: string
This is the text that is sent via SMS to the user trying to authenticate with an SMS token. You can use the tags <otp> and <serial>.
Note
You need to put the text into quotes, otherwise it will be split
and interpreted as list of words.
type: bool
A new OTP value will be sent via SMS if the user authenticated successfully with his SMS token. Thus the user does not have to trigger a new SMS when he wants to login again.
(TODO): not yet migrated.
type: string
This is the URL for the half automatic mode of the QR token. To this URL the TAN/OTP value will be pushed.
(TODO): not yet migrated.
type: string
This is a list of token types for which challenge response can be used during authentication.
The scope authorization provides means to define what should happen if a user proved his identity and authenticated successfully.
Authorization policies take the realm, the user and the client into account.
Technically the authorization policies apply to the Validate endpoints and are checked using Policy Module and Policy Decorators.
The following actions are available in the scope authorization:
type: string
Users will only be authorized with this very tokentype. The string can hold a comma separated list of case insensitive tokentypes.
This is checked after the authentication request, so that a valid OTP value is wasted, so that it can not be used, even if the user was not authorized at this request
Note
Combining this with the client IP you can use this to allow remote access to sensitive areas only with one special token type while allowing access to less sensitive areas with other token types.
type: string
Users will only be authorized with the serial number. The string can hold a regular expression as serial number.
This is checked after the authentication request, so that a valid OTP value is wasted, so that it can not be used, even if the user was not authorized at this request
Note
Combining this with the client IP you can use this to allow remote access to sensitive areas only with hardware tokens like the Yubikey, while allowing access to less secure areas also with a Google Authenticator.
type: string
This policy is checked before the user authenticates. The realm of the user matching this policy will be set to the realm in this action.
Note
This can be used if the user can not pass his realm when authenticating at a certain client, but the realm needs to be available during authentication since the user is not located in the default realm.
type: bool
Usually an authentication response returns additional information like the serial number of the token that was used to authenticate or the reason why the authentication request failed.
If this action is set and the user authenticated successfully this additional information will not be returned.
type: bool
Usually an authentication response returns additional information like the serial number of the token that was used to authenticate or the reason why the authentication request failed.
If this action is set and the user fails to authenticate this additional information will not be returned.
The scope enrollment defines what happens during enrollment either by an administrator or during the user self enrollment.
Enrollment policies take the realms, the client (see Policies) and the user settings into account.
Technically enrollment policies control the use of the REST API Token endpoint and specially the init and assign-methods.
Technically the decorators in API Policies are used.
The following actions are available in the scope enrollment:
type: int
This is the maximum allowed number of tokens in the specified realm.
Note
If you have several realms with realm admins and you imported a pool of hardware tokens you can thus limit the consumed hardware tokens per realm.
type: int
Limit the maximum number of tokens per user in this realm.
Note
If you do not set this action, a user may have unlimited tokens assigned.
type: string
This sets the label for a newly enrolled Google Authenticator. Possible tags to be replaces are <u> for user, <r> for realm an <s> for the serial number.
The default behaviour is to use the serial number.
Note
This is useful to identify the token in the Authenticator App.
type: bool
Users can assign a token just by using this token. The user can take a token from a pool of unassigned tokens. When this policy is set, and the user has no token assigned, autoassignment will be done: The user authenticates with a new PIN and an OTP value from the token. If the OTP value is correct the token gets assigned to the user and the given PIN is set as the OTP PIN.
Note
Requirements are:
Warning
In this case assigning the token is only a
one-factor-authentication: the possession of the token.
(TODO) Not yet migrated.
type: int
Generates a random OTP PIN during self enrollment. Thus the user is forced to set a certain OTP PIN.
Note
At the moment this randomly generated PIN is not used. It could be used to be sent via a PIN letter in the future.
(TODO) Not yet migrated.
type: int
values: 0 or 1
If set to 1 the OTP PIN of a token will be encrypted. The default behaviour is to hash the OTP PIN, which is safer.
(TODO) Not yet migrated.
type: int
This is the length of the generated password for the lost token process.
(TODO) Not yet migrated.
type: string
This is the contents that a generated password for the lost token process should have. You can use
- c: for lowercase letters
- n: for digits
- s: for special characters (!#$%&()*+,-./:;<=>?@[]^_)
- C: for uppercase letters
Example:
The action lostTokenPWLen=10, lostTokenPWContents=Cns could generate a password like AC#!49MK)).
(TODO) Not yet migrated.
type: int
This is how many days the replacement token for the lost token should be valid. After this many days the replacement can not be used anymore.
type: string
allowed values: “userstore”, “privacyIDEA”
If this action is set to login_mode=privacyIDEA, the users and administrators need to authenticate against privacyIDEA when logging into the WebUI. I.e. they can not login with their domain password anymore but need to authenticate with one of their tokens.
Warning
If you set this action and the user deletes or disables all his tokens, he will not be able to login anymore.
Note
Administrators defined in the database using the pi-manage.py command can still login with their normal passwords.
Note
A sensible way to use this, is to combine this action in a policy with the client parameter: requiring the users to login to the Web UI remotely from the internet with OTP but still login from within the LAN with the domain password.
You can define as many policies as you wish to. The logic of the policies in the scopes is additive.
Each policy can contain the following attributes:
policy name
A unique name of the policy. The name is the identifier of the policy. If you create a new policy with the same name, the policy is overwritten.
scope
The scope of the policy as described above.
action
This is the important part of the policy. Each scope provides its own set of actions. An action describes that something is allowed or that some behaviour is configured. A policy can contain several actions. Actions can be of type boolean, string or integer. Boolean actions are enabled by just adding this action - like scope=user:action=disable, which allows the user to disable his own tokens. string and integer actions require an additional value - like scope=authentication:action='otppin=userstore'.
user
This is the user, for whom this policy is valid. Depending on the scope the user is either an administrator or a normal authenticating user.
If this field is left blank, this policy is valid for all users.
resolver
This policy will be valid for all users in this resolver.
If this field is left blank, this policy is valid for all resolvers.
realm
This is the realm, for which this policy is valid.
If this field is left blank, this policy is valid for all realms.
client
This is the requesting client, for which this action is valid. I.e. you can define different policies if the user access is allowed to manage his tokens from different IP addresses like the internal network or remotely via the firewall.
You can enter several IP addresses or subnets divided by comma (like 10.2.0.0/16, 192.168.0.1).
time
Not used, yet.
Note
Policies can be active or inactive. So be sure to activate a policy to get the desired effect.
The systems provides a sophisticated audit log, that can be viewed in the WebUI.
privacyIDEA comes with an SQL audit module. (see Audit log)
The sqlaudit module writes audit entries to an SQL database. For performance reasons the audit module does no log rotation during the logging process.
But you can set up a cron job to clean up old audit entries.
You can specify a highwatermark and a lowwatermark. To clean up the audit log table, you can call the sqlaudit module at the command line:
python privacyidea/lib/auditmodules/sqlaudit.py \
-f config/privacyidea.ini.example \
--low=5000
--high=10000
This will, if there are more than 10.000 log entries, clean all old log entries, so that only 5000 log entries remain.
privacyIDEA lets you define Machine Resolvers to connect to existing machine stores. The idea is for users to be able to authenticate on those client machines. Not in all cases an online authentication request is possible, so that authentication items can be passed to those client machines.
In addition you need to define, which application on the client machine the user should authenticate to. Different application require different authentication items.
Therefore privacyIDEA can define application types. At the moment privacyIDEA knows the application luks` and ``ssh. You can write your own application class, which is defined in :ref:code_application_class.
You need to assign an application and a token to a client machine. Each application type can work with certain token types and each application type can use additional parameters.
Note
Not all tokens work well with all applications!
Currently working token types: SSH
Parameters:
user (optional, default=root)
When the SSH token type is assigned to a client, the user specified in the user parameter can login with the private key of the SSH token.
In the sshd_config file you need to configure the AuthorizedKeysCommand. Set it to:
privacyidea-authorizedkeys
This will fetch the SSH public keys for the requesting machine.
The command expects a configuration file /etc/privacyidea/authorizedkeyscommand which looks like this:
[Default]
url=https://localhost
admin=admin
password=test
Note
To disable a SSH key for all servers, you simple can disable the
SSH token in privacyIDEA.
Currently working token types: Yubikey Challenge Response
Parameters:
slot The slot to which the authentication information should be written
partition The encrypted partition (usually /dev/sda3 or /dev/sda5)
These authentication items need to be pulled on the client machine from the privacyIDEA server.
Thus, the following script need to be executed with root rights (able to write to LUKS) on the client machine:
privacyidea-luks-assign @secrets.txt --cleanslot --name salt-minion
For more information please see the man page of this tool.
[1] | http://www.saltstack.com/ |
privacyIDEA comes with application plugins. These are plugins for applications like OTRS, FreeRADIUS or simpleSAMLphp which enable these application to authenticate users against privacyIDEA.
If you want to install the FreeRADIUS Plugin on Ubuntu 14.04 LTS this can be easily done, since there is a ready made package (see FreeRADIUS).
If you want to run your FreeRADIUS server on another distribution, you may download the module at [1].
Then you need to configure your FreeRADIUS site and the perl module. The latest FreeRADIUS plugin uses the /validate/check REST API of privacyIDEA.
You need to configure the perl module in FreeRADIUS modules/perl to look something like this:
perl {
module = /usr/share/privacyidea/freeradius/privacyidea_radius.pm
}
Your freeradius enabled site config should contain something like this:
authenticate {
Auth-Type Perl {
perl
}
digest
unix
}
While you define the default authenticate type to be Perl in the users file:
DEFAULT Auth-Type := Perl
Note
The perl module is not thread safe, so you need to start FreeRADIUS with the -t switch.
You can install the plugin for simpleSAMLphp on Ubuntu 14.04 LTS (see SimpleSAMLphp) or on any other distribution using the source files from [2].
Follow the simpleSAMLphp instructions to configure your authsources.php. A usual configuration will look like this:
'example-privacyidea' => array(
'privacyidea:privacyidea',
/*
* The name of the privacyidea server and the protocol
* A port can be added by a colon
* Required.
*/
'privacyideaserver' => 'https://your.server.com',
/*
* Check if the hostname matches the name in the certificate
* Optional.
*/
'sslverifyhost' => False,
/*
* Check if the certificate is valid, signed by a trusted CA
* Optional.
*/
'sslverifypeer' => False,
/*
* The realm where the user is located in.
* Optional.
*/
'realm' => '',
/*
* This is the translation from privacyIDEA attribute names to
* SAML attribute names.
*/
'attributemap' => array('username' => 'samlLoginName',
'surname' => 'surName',
'givenname' => 'givenName',
'email' => 'emailAddress',
'phone' => 'telePhone',
'mobile' => 'mobilePhone',
),
),
[1] | https://github.com/privacyidea/privacyidea/tree/master/authmodules/FreeRADIUS |
[2] | https://github.com/privacyidea/privacyidea/tree/master/authmodules/simpleSAMLphp |
(TODO): Not yet migrated.
The menu tools contains some helpful tools to manage your tokens.
Here you can enter an OTP value and have the system identify the token. This can be useful if the printed serial number on the token can not be read anymore or if the hardware token has not serial number printed on it at all.
You can choose the
- tokentype
- whether the token is an assigned token or not
- and the realm of the token
to set limits to the token search.
Warning
The system needs to got to all tokens and calculate the next (default) 10 OTP values. Depending on the number of tokens you have in the system this can be very time consuming!
Here you can enter a token serial number of the token from which you want to copy the OTP PIN and the serial number of the token to which you want to copy it.
This function is also used in the lost token scenario.
But the help desk can also use it if the administrator enrolls a new token to the user and
Then the administrator can create a second token for the user and copy the OTP PIN (which only the user knows) of the old token to the new, second token.
If you have complicated policy settings you can use this dialog to determine if the policies behave as expected. You can enter the scope, the real, action user and client to “simulate” e.g. an authentication request.
The system will tell you if any policy is triggered.
Here you can export the list of the tokens to a CSV file.
Note
In the resolver you can define additional fields, that are usually not used by privacyIDEA. But you can add those fields to the export. Thus you can e.g. add special LDAP attributes in the list of the exported tokens.
Here you can export the audit information.
Warning
You should limit the export to a number of audit entries. As the audit log can grow very big, the export of 20.000 audit lines could result in blocking the system.
Seed files that contain the secret keys of hardware tokens can be imported to the system via the menu Import.
The default import options are to import SafeNet XML file, OATH CSV files or Yubikey CSV files.
SafeNet XML (former Aladdin) files are usually shipped with SafeNet eToken PASS.
Each token is contained in a <Token> tag and the seed is containted in a <seed> tag. These files were originally used to import HOTP-SHA1 tokens.
Note
Today usually SafeNet tokens are shipped with a .dat file. If you get such a file, please contact the mailing list.
This is a very simple CSV file to import HOTP, TOTP or OATH tokens. You can also convert your seed easily to this file format, to import the tokens.
The file format looks like this:
<serial>, <seed>, <type>, <otp length>, <time step>
For OCRA tokens it looks like this:
<serial>, <seed>, OCRA, <ocra suite>
serial is the serial number of the token that will also be used to identify the token in the database. Importing the same serial number twice will overwrite the token data.
seed is the secret key, that is used to calculate the OTP value. The seed is provided in a hexadecimal notation. Depending on the length either the SHA1 or SHA256 hash algorithm is identified.
type is either HOTP, TOTP or OCRA.
otp length is the length of the OTP value generated by the token. This is usually 6 or 8.
time step is the time step of TOTP tokens. This is usually 30 or 60.
ocra suite is the ocra suite of the OCRA token according to [1].
Here you can import the CSV file that is written by the Yubikey personalization tool [2]. privacyIDEA can import all Yubikey modes, either Yubico mode or HOTP mode.
Note
There is an annoying drawback of the personalization tool: If you a initializing several HOTP yubikeys it will not write the serial number to the file.
[1] | http://tools.ietf.org/html/rfc6287#section-6 |
[2] | http://www.yubico.com/products/services-software/personalization-tools/use/ |
The code roughly has three levels.
The API level is used to access the system. For some calls you need to be authenticated as administrator, for some calls you can be authenticated as normal user. These are the token and the audit endpoint. For calls to the validate API you do not need to be authenticated at all.
At this level Authentication is performed. In the lower levels there is no authentication anymore.
The object g.logged_in_user is used to pass the authenticated user. The client gets a JSON Web Token to authenticate every request.
API functions are decorated with the decorators admin_required and user_required to define access rules.
At the LIB level all library functions are defined. There is no authentication on this level. Also there is no flask/Web/request code on this level.
Request information and the logged_in_user need to be passed to the functions as parameters, if they are needed.
If possible, policies are checked with policy decorators.
On the DB level you can simply modify all objects.
The administrators for managing the system. To manage the administrators use the command pi-manage.py.
In addition certain realms can be defined to be administrative realms.
Table for handling of the generic challenges.
return a dictionary of all vars in the challenge class
Parameters: | timestamp (bool) – if true, the timestamp will given in a readable format 2014-11-29 21:56:43.057293 |
---|---|
Returns: | dict of vars |
This returns how many OTPs were already received for this challenge. and if a valid OTP was received.
Returns: | tuple of count and True/False |
---|---|
Return type: | tuple |
The config table holds all the system configuration in key value pairs.
Additional configuration for realms, resolvers and machine resolvers is stored in specific tables.
This model holds the definition to the machinestore. Machines could be located in flat files, LDAP directory or in puppet services or other...
The usual MachineResolver just holds a name and a type and a reference to its config
Each Machine Resolver can have multiple configuration entries. The config entries are referenced by the id of the machine resolver
The MachineToken assigns a Token and an application type to a machine. The Machine is represented as the tuple of machineresolver.id and the machine_id. The machine_id is defined by the machineresolver.
This can be an n:m mapping.
This class holds an Option for the token assigned to a certain client machine. Each Token-Clientmachine-Combination can have several options.
This class mixes in some common Class table functions like delete and save
The policy table contains policy definitions which control the behaviour during
- enrollment
- authentication
- authorization
- administration
- user actions
The realm table contains the defined realms. User Resolvers can be grouped to realms. This very table contains just contains the names of the realms. The linking to resolvers is stored in the table “resolverrealm”.
The table “resolver” contains the names and types of the defined User Resolvers. As each Resolver can have different required config values the configuration of the resolvers is stored in the table “resolverconfig”.
Each Resolver can have multiple configuration entries. Each Resolver type can have different required config values. Therefor the configuratinon is stored in simple key/value pairs. If the type of a config entry is set to “password” the value of this config entry is stored encrypted.
The config entries are referenced by the id of the resolver.
This table stores which Resolver is located in which realm This is a N:M relation
while the table “tokeninfo” contains additional information that is specific to the tokentype.
Deletes tokeninfo for a given token. If the key is omitted, all Tokeninfo is deleted.
Parameters: | key – searches for the given key to delete the entry |
---|---|
Returns: |
simulate the dict behaviour to make challenge processing easier, as this will have to deal as well with ‘dict only challenges’
Parameters: |
|
---|
calculate a hash from a pin Fix for working with MS SQL servers MS SQL servers sometimes return a ‘<space>’ when the column is empty: ‘’
Set the additional token info for this token
Entries that end with ”.type” are used as type for the keys. I.e. two entries sshkey=”XYZ” and sshkey.type=”password” will store the key sshkey as type “password”.
Parameters: | info (dict) – The key-values to set for this token |
---|
Set the list of the realms. This is done by filling the tokenrealm table. :param realms: realms :type realms: list
For smartcards this sets the security officer pin of the token
:rtype : None
The password is split into the PIN and the OTP component. THe token knows its length, so it can split accordingly.
Parameters: |
|
---|---|
Returns: | tuple of (res, pin, otpval) |
The table “tokeninfo” is used to store additional, long information that is specific to the tokentype. E.g. the tokentype “TOTP” has additional entries in the tokeninfo table for “timeStep” and “timeWindow”, which are stored in the column “Key” and “Value”.
The tokeninfo is reference by the foraign key to the “token” table.
This table stored to wich realms a token is assigned. A token is in the realm of the user it is assigned to. But a token can also be put into many additional realms.
Produce a conjunction of expressions joined by AND.
E.g.:
from sqlalchemy import and_
stmt = select([users_table]).where(
and_(
users_table.c.name == 'wendy',
users_table.c.enrolled == True
)
)
The and_() conjunction is also available using the Python & operator (though note that compound expressions need to be parenthesized in order to function with Python operator precedence behavior):
stmt = select([users_table]).where(
(users_table.c.name == 'wendy') &
(users_table.c.enrolled == True)
)
The and_() operation is also implicit in some cases; the Select.where() method for example can be invoked multiple times against a statement, which will have the effect of each clause being combined using and_():
stmt = select([users_table]).\
where(users_table.c.name == 'wendy').\
where(users_table.c.enrolled == True)
See also
or_()
Delete all challenges, that have expired.
Returns: | None |
---|
Return the database ID of the machine resolver :param resolvername: :return:
Returns the ID in the machinetoken table
Parameters: |
|
---|---|
Returns: | The ID of the machinetoken entry |
Return type: | int |
Based on the database models, which are tested in tests/test_db_model.py, there are different modules.
resolver.py contains functions to simply deal with resolver definitions. On this level users and realms are not know, yet.
realm.py contains functions to deal with realm. Realms are a list of several resolvers. So prior to bother the realm.py, the resolver.py should be understood and working. On this level, users are not known, yet.
user.py contains functions to deal with users. A user object is an entity in a realm. And of course the user object itself can be found in a resolver. But you need to have working resolver.py and realm.py to be able to work with user.py
For further details see the following modules:
There are the library functions for user functions. It depends on the lib.resolver and lib.realm.
There are and must be no dependencies to the token functions (lib.token) or to webservices!
Usually a user can be found via “login@realm”.
A user object with an empty login and realm should not exist, whereas a user object could have an empty resolver.
Find the parameters user, realm and resolver and create a user object from these parameters.
An exception is raised, if a user in a realm is found in more than one resolvers.
Parameters: | param (dict) – The dictionary of request parameters |
---|---|
Returns: | User as found in the parameters |
Return type: | User object |
return the detailed information for a user in a resolver
Parameters: |
|
---|---|
Returns: | a dict with all the userinformation |
Return type: | dict |
Determine the username for a given id and a resolvername.
Parameters: |
|
---|---|
Returns: | the username or “” if it does not exist |
Return type: | string |
Split the username of the form user@realm into the username and the realm splitting myemail@emailprovider.com@realm is also possible and will return (myemail@emailprovider, realm).
We can also split realmuser to (user, realm)
Parameters: | username (string) – the username to split |
---|---|
Returns: | username and realm |
Return type: | tuple |
Add a key and a value to the DB tokeninfo :param key: :param value: :return:
High level interface which covers the check_pin and check_otp This is the method that verifies single shot authentication like they are done with push button tokens.
It is a high level interface to support other tokens as well, which do not have a pin and otp separation - they could overwrite this method
If the authentication succeeds an OTP counter needs to be increased, i.e. the OTP value that was used for this authentication is invalidated!
Parameters: |
|
---|---|
Returns: | returns tuple of 1. true or false for the pin match, 2. the otpcounter (int) and the 3. reply (dict) that will be added as
|
Return type: | tuple |
Just clean up all challenges, for which the expiration has expired.
Returns: | None |
---|
This function checks the count_auth and the count_auth_success. If the count_auth is less than count_auth_max and count_auth_success is less than count_auth_success_max it returns True. Otherwise False.
Returns: | success if the counter is less than max |
---|---|
Return type: | bool |
This method verifies if there is a matching challenge for the given passw and also verifies if the response is correct.
It then returns the new otp_counter of the token.
In case of success the otp_counter will be >= 0.
Parameters: |
|
---|---|
Returns: | return otp_counter. If -1, challenge does not match |
Return type: | int |
This checks the OTP value, AFTER the upper level did the checkPIN
In the base class we do not know, how to calculate the OTP value. So we return -1. In case of success, we should return >=0, the counter
Parameters: |
|
---|---|
Returns: | counter of the matching OTP value. |
Return type: | int |
checks if the given OTP value is/are values of this very token. This is used to autoassign and to determine the serial number of a token.
Parameters: |
|
---|---|
Returns: | True or a value > 0 in case of success |
Check the PIN of the given Password. Usually this is only dependent on the token itself, but the user object can cause certain policies.
Each token could implement its own PIN checking behaviour.
Parameters: |
|
---|---|
Returns: | If the PIN is correct, return True |
Return type: | bool |
This checks if the datetime.datetime.now() is within the validity period of the token.
Returns: | success |
---|---|
Return type: | bool |
This method creates a challenge, which is submitted to the user. The submitted challenge will be preserved in the challenge database.
If no transaction id is given, the system will create a transaction id and return it, so that the response can refer to this transaction.
Parameters: |
|
---|---|
Returns: | tuple of (bool, message, transactionid, attributes) |
Return type: | tuple |
The return tuple builds up like this: bool if submit was successful; message which is displayed in the JSON response; additional attributes, which are displayed in the JSON response.
This returns the token data as a dictionary. It is used to display the token list at /token/list.
Returns: | The token data as dict |
---|---|
Return type: | dict |
Returns a hashlib function for a given string :param hLibStr: the hashlib :type hLibStr: string :return: the hashlib :rtype: function
to complete the token normalisation, the response of the initialiastion should be build by this token specific method.
get_init_detail returns additional information after an admin/init like the QR code of an HOTP/TOTP token. Can be anything else.
Parameters: |
|
---|---|
Returns: | additional descriptions |
Return type: | dict |
return the status of the token rollout
Returns: | return the status dict. |
---|---|
Return type: | dict |
This returns a dictionary of multiple future OTP values of a token.
Parameters: |
|
---|---|
Returns: | True/False, error text, OTP dictionary |
Return type: | Tuple |
The default token does not support getting the otp value will return a tuple of four values a negative value is a failure.
Returns: | something like: (1, pin, otpval, combined) |
---|
Return a list of realms the token is assigned to :return: realms :rtype:l list
return the complete token info or a single key of the tokeninfo. When returning the complete token info dictionary encrypted entries are not decrypted. If you want to receive a decrypted value, you need to call it directly with the key.
Parameters: |
|
---|---|
Returns: | the value for the key |
Return type: | int or string |
return the user (owner) of a token If the token has no owner assigned, we return None
Returns: | The owner of the token |
---|---|
Return type: | User object |
returns the end of validity period (if set) if not set, “” is returned. :return: the end of the validity period :rtype: string
returns the start of validity period (if set) if not set, “” is returned. :return: the start of the validity period :rtype: string
return the token state as dicts with keys like type, token, mode... :return: token as dict
Increase the counter, that counts authentications - successful and unsuccessful
Increase the otp counter and store the token in the database :param counter: the new counter value. If counter is given, than
the counter is increased to counter+1 If the counter is not given, the counter is increased +1
Parameters: | reset (bool) – reset the failcounter if set to True |
---|---|
Returns: | the new counter value |
This method checks, if this is a request, that triggers a challenge.
The default behaviour to trigger a challenge is, if the passw parameter only contains the correct token pin and the request contains a data or a challenge key i.e. if the options parameter contains a key data or challenge.
Each token type can decide on its own under which condition a challenge is triggered by overwriting this method.
please note: in case of pin policy == 2 (no pin is required) the check_pin would always return true! Thus each request containing a data or challenge would trigger a challenge!
Parameters: |
|
---|---|
Returns: | true or false |
Return type: | bool |
This method checks, if this is a request, that is the response to a previously sent challenge.
The default behaviour to check if this is the response to a previous challenge is simply by checking if the request contains a parameter state or transactionid i.e. checking if the options parameter contains a key state or transactionid.
This method does not try to verify the response itself! It only determines, if this is a response for a challenge or not. The response is verified in check_challenge_response.
Parameters: |
|
---|---|
Returns: | true or false |
Return type: | bool |
Sets the counter for the occurred login attepms as key “count_auth” in token info :param count: a number :type count: int
Sets the counter for the maximum allowed login attemps as key “count_auth_max” in token info :param count: a number :type count: int
Sets the counter for the occurred successful logins as key “count_auth_success” in token info :param count: a number :type count: int
Sets the counter for the maximum allowed successful logins as key “count_auth_success_max” in token info :param count: a number :type count: int
Set the description on the database level
Parameters: | description (string) – description of the token |
---|
set the PIN of a token. Usually the pin is stored in a hashed way. :param pin: the pin to be set for the token :type pin: basestring :param encrypt: If set to True, the pin is stored encrypted and
can be retrieved from the database again
Set the list of the realms of a token. :param realms: realms the token should be assigned to :type realms: list
Set the tokeninfo field in the DB. Old values will be deleted. :param info: dictionary with key and value :type info: dict :return:
Set the tokentype in this object and also in the underlying database-Token-object.
Parameters: | tokentype (string) – The type of the token like HOTP or TOTP |
---|
Set the user attributes (uid, resolvername, resolvertype) of a token.
Parameters: |
|
---|---|
Returns: | None |
(was setUid) Set the user attributes of a token :param uid: The user id in the user source :param resolvername: The name of the resolver :param resolvertype: The type of the resolver :return: None
sets the end date of the validity period for a token :param end_date: the end date in the format “%d/%m/%y %H:%M”
if the format is wrong, the method will throw an exception
sets the start date of the validity period for a token :param start_date: the start date in the format “%d/%m/%y %H:%M”
if the format is wrong, the method will throw an exception
Split the password into the token PIN and the OTP value
take the given password and split it into the PIN and the OTP value. The splitting can be dependent of certain policies. The policies may depend on the user.
Each token type may define its own way to slit the PIN and the OTP value.
Parameters: |
|
---|---|
Returns: | tuple of pin and otp value |
Returns: | tuple of (split status, pin, otp value) |
Return type: | tuple |
This module contains all top level token functions. It depends on the models, lib.user and lib.tokenclass (which depends on the tokenclass implementations like lib.tokens.hotptoken)
This is the middleware/glue between the HTTP API and the database
Sets a token info field in the database. The info is a dict for each token of key/value pairs.
Parameters: |
|
---|
is stored encrypted :type value_type: basestring :param user: The owner of the tokens, that should be modified :type user: User object :return: the number of modified tokens :rtype: int
Produce a conjunction of expressions joined by AND.
E.g.:
from sqlalchemy import and_
stmt = select([users_table]).where(
and_(
users_table.c.name == 'wendy',
users_table.c.enrolled == True
)
)
The and_() conjunction is also available using the Python & operator (though note that compound expressions need to be parenthesized in order to function with Python operator precedence behavior):
stmt = select([users_table]).where(
(users_table.c.name == 'wendy') &
(users_table.c.enrolled == True)
)
The and_() operation is also implicit in some cases; the Select.where() method for example can be invoked multiple times against a statement, which will have the effect of each clause being combined using and_():
stmt = select([users_table]).\
where(users_table.c.name == 'wendy').\
where(users_table.c.enrolled == True)
See also
or_()
Assign token to a user. If the PIN is given, the PIN is reset.
Parameters: |
|
---|---|
Returns: | True if the token was assigned, in case of an error an exception |
is thrown :rtype: bool
This function is called to auto_assign a token to the user.
If the user does not have a token, yet, the not assigned tokens in his realm are searched if they match the given passw.
Parameters: |
|
---|---|
Returns: | True or False and detailed reply information |
Return type: | bool, dict |
This checks, if the given serial number can be used for a new token. it returns a tuple (result, new_serial) result being True if the serial does not exist, yet. new_serial is a suggestion for a new serial number, that does not exist, yet.
Parameters: | serial – Seral number that is to be checked, if it can be used for |
---|
a new token. :type serial: string :result: bool and serial number :rtype: tuple
This function checks the otp for a given serial
If the OTP matches, True is returned and the otp counter is increased.
The function tries to determine the user (token owner), to derive possible additional policies from the user.
Parameters: |
|
---|---|
Returns: | tuple of result (True, False) and additional dict |
Return type: | tuple |
this takes a list of token objects and tries to find the matching token for the given passw. In also tests, * if the token is active or * the max fail count is reached, * if the validity period is ok...
This function is called by check_serial_pass, check_user_pass and check_yubikey_pass.
Parameters: |
|
---|---|
Returns: | tuple of success and optional response |
Return type: | (bool, dict) |
This function checks the otp for a given user. It is called by the API /validate/check and simplecheck
If the OTP matches, True is returned and the otp counter is increased.
Parameters: |
|
---|---|
Returns: | tuple of result (True, False) and additional dict |
Return type: | tuple |
This function copies the token PIN from one token to the other token. This can be used for workflows like lost token.
In fact the PinHash and the PinSeed are transferred
Parameters: |
|
---|---|
Returns: | True. In case of an error raise an exception |
Return type: | bool |
Copy the realms of one token to the other token
Parameters: |
|
---|---|
Returns: | None |
This function copies the user from one token to the other token. In fact the user_id, resolver and resolver type are transferred.
Parameters: |
|
---|---|
Returns: | True. In case of an error raise an exception |
Return type: | bool |
(was createTokenClassObject) create a token class object from a given type If a tokenclass for this type does not exist, the function returns None.
Parameters: | db_token (database token object) – the database referenced token |
---|---|
Returns: | instance of the token class object |
Return type: | tokenclass object |
Enable or disable a token. This can be checked with is_token_active
Enabling an already active token will return 0.
Parameters: |
|
---|---|
Returns: | Number of tokens that were enabled/disabled |
Return type: |
generate a serial for a given tokentype
Parameters: |
|
---|---|
Returns: | serial number |
Return type: | string |
return a dictionary with all tokens, that are assigned to users. This returns a dictionary with the key being the serial number of the token and the user information as dict.
Returns: | dictionary of serial numbers |
---|---|
Return type: | dict |
This returns the dynamic policy definitions that come with the new loaded token classes.
Parameters: | scope – an optional scope parameter. Only return the policies of |
---|
this scope. :return: The policy definition for the token or only for the scope.
This function returns a list of OTP values for the given Token. Please note, that the tokentype needs to support this function.
Parameters: |
|
---|
time based tokens) :param epoch_start: unix time start date (used with time based tokens) :param epoch_end: unix time end date (used with time based tokens) :param curTime: Simulate the servertime :type curTime: datetime :param timestamp: Simulate the servertime (unix time in seconds) :type timestamp: int
Returns: | dictionary of otp values |
---|---|
Return type: | dictionary |
This returns the number of tokens in one realm. :param realm: The name of the realm :type realm: basestring :param active: If only active tokens should be taken into account :type active: bool :return: The number of tokens in the realm :rtype: int
This function returns the current OTP value for a given Token. The tokentype needs to support this function. if the token does not support getting the OTP value, a -2 is returned.
Parameters: |
|
---|---|
Returns: | tuple with (result, pin, otpval, passw) |
Return type: | tuple |
This function returns a list of the realms of a token
Parameters: | serial (basestring) – the serial number of the token |
---|---|
Returns: | list of the realm names |
Return type: | list |
Returns the serial for a given OTP value The tokenobject_list would be created by get_tokens()
Parameters: |
|
---|---|
Returns: | the serial for a given OTP value and the user |
Return type: | basestring |
search the token in the token_list, that creates the given OTP value. The tokenobject_list would be created by get_tokens()
Parameters: |
|
---|---|
Returns: | The token, that creates this OTP value |
Return type: | Tokenobject |
returns the user object, to which the token is assigned. the token is identified and retrieved by it’s serial number
If the token has no owner, None is returned
Parameters: | serial (basestring) – serial number of the token |
---|---|
Returns: | The owner of the token |
Return type: | User object or None |
Returns the tokentype of a given serial number
Parameters: | serial (string) – the serial number of the to be searched token |
---|---|
Returns: | tokentype |
Return type: | string |
return the config definition of a dynamic token
Parameters: |
|
---|---|
Returns: | dict - if nothing found an empty dict |
Return type: | dict |
(was getTokensOfType) This function returns a list of token objects of a * given type, * of a realm * or tokens with assignment or not * for a certain serial number or * for a User
E.g. thus you can get all assigned tokens of type totp.
Parameters: |
|
---|
If None get all tokens. :type assigned: bool :param user: Filter for the Owner of the token :type user: User Object :param serial: The serial number of the token :type serial: basestring :param active: Whether only active (True) or inactive (False) tokens should be returned :type active: bool :param resolver: filter for the given resolver name :type resolver: basestring :param rollout_state: returns a list of the tokens in the certain rollout state. Some tokens are not enrolled in a single step but in multiple steps. These tokens are then identified by the DB-column rollout_state. :param count: If set to True, only the number of the result and not the list is returned. :type count: bool
Returns: | A list of tokenclasses (lib.tokenclass) |
---|---|
Return type: | list |
Return a list of the token ojects, that contain this very resolver
Parameters: | resolver (basestring) – The resolver, the tokens should be in |
---|---|
Returns: | list of tokens with this resolver |
Return type: | list of token objects |
This function is used to retrieve a token list, that can be displayed in the Web UI. It supports pagination. Each retrieved page will also contain a “next” and a “prev”, indicating the next or previous page. If either does not exist, it is None.
Parameters: |
|
---|
Token.serial. If a string like “serial” is provided, we try to convert it to the DB column. :type sortby: A Token column or a string. :param sortdir: Can be “asc” (default) or “desc” :type sortdir: basestring :param psize: The size of the page :type psize: int :param page: The number of the page to view. Starts with 1 ;-) :type page: int :return: dict with tokens, prev, next and count :rtype: dict
get the serial number of a token from a challenge state / transaction
Parameters: | transaction_id (basestring) – the state / transaction id |
---|---|
Returns: | the serial number or None |
Return type: | basestring |
create a new token or update an existing token
Parameters: | param – initialization parameters like: serial (optional) type (optionl, default=hotp) otpkey |
---|
type param: dict :param user: the token owner :type user: User Object :param tokenrealms: the realms, to which the token should belong :type tokenrealms: list
Returns: | token object or None |
---|---|
Return type: | TokenClass object |
Return True if the token is active, otherwise false Returns None, if the token does not exist.
Parameters: | serial (basestring) – The serial number of the token |
---|---|
Returns: | True or False |
Return type: | bool |
Check if the given user is the owner of the token with the given serial number :param serial: The serial number of the token :type serial: str :param user: The user that needs to be checked :type user: User object :return: Return True or False :rtype: bool
This is the workflow to handle a lost token. The token <serial> is lost and will be disabled. A new token of type password token will be created and assigned to the user. The PIN of the lost token will be copied to the new token. The new token will have a certain validity period.
Parameters: |
|
---|---|
Returns: | result dictionary |
remove the token that matches the serial number or all tokens of the given user and also remove the realm associations and all its challenges
Parameters: |
|
---|---|
Returns: | The number of deleted token |
Return type: | int |
Reset the :param serial: :param user: :return: The number of tokens, that were resetted :rtype: int
Resyncronize the token of the given serial number by searching the otp1 and otp2 in the future otp values.
Parameters: |
|
---|---|
Returns: |
The auth counters are stored in the token info database field. There are different counters, that can be set
count_auth -> max=False, success=False count_auth_max -> max=True, success=False count_auth_success -> max=False, success=True count_auth_success_max -> max=True, success=True
Parameters: |
|
---|
to be modified :type max: bool :param success: True, if either count_auth_success or count_auth_success_max are to be modified :type success: bool :return: number of modified tokens :rtype: int
The count window is used during authentication to find the matching OTP value. This sets the count window per token.
Parameters: |
|
---|---|
Returns: | number of modified tokens |
Return type: | int |
Set the default values for the token with the given serial number :param serial: token serial :type serial: basestring :return: None
Set the description of a token
Parameters: |
|
---|---|
Returns: | number of modified tokens |
Return type: | int |
Set the hashlib in the tokeninfo. Can be something like sha1, sha256...
Parameters: |
|
---|---|
Returns: | the number of token infos set |
Return type: | int |
Set the maximum fail counts of tokens. This is the maximum number a failed authentication is allowed.
Parameters: |
|
---|---|
Returns: | number of modified tokens |
Return type: | int |
Set the otp length of the token defined by serial or for all tokens of the user. The OTP length is usually 6 or 8.
Parameters: |
|
---|---|
Returns: | number of modified tokens |
Return type: | int |
Set the token PIN of the token. This is the static part that can be used to authenticate.
Parameters: |
|
---|
user will be set :type used: User object :param serial: If the serial is specified, the PIN for this very token will be set. :param param: Optional parameters, token specific :return: The number of PINs set (usually 1) :rtype: int
Set the SO PIN of a smartcard. The SO Pin can be used to reset the PIN of a smartcard. The SO PIN is stored in the database, so that it could be used for automatic processes for User PIN resetting.
Parameters: |
|
---|---|
Returns: | The number of SO PINs set. (usually 1) |
Return type: | int |
This sets the user pin of a token. This just stores the information of the user pin for (e.g. an eTokenNG, Smartcard) in the database
Parameters: |
|
---|---|
Returns: | The number of PINs set (usually 1) |
Return type: | int |
Set all realms of a token. This sets the realms new. I.e. it does not add realms. So realms that are not contained in the list will not be assigned to the token anymore.
Thus, setting realms=[] clears all realms assignments.
Parameters: |
|
---|---|
Returns: | the number of tokens, to which realms where added. As a serial |
number should be unique, this is either 1 or 0. :rtype: int
The sync window is the window that is used during resync of a token. Such many OTP values are calculated ahead, to find the matching otp value and counter.
Parameters: |
|
---|---|
Returns: | number of modified tokens |
Return type: | int |
returns true if the token with the given serial number exists
Parameters: | serial – the serial number of the token |
---|
Base function to handle the policy entries in the database. This module only depends on the db/models.py
The functions of this module are tested in tests/test_lib_policy.py
A policy has the attributes
- name
- scope
- action
- realm
- resolver
- user
- client
- active
name is the unique identifier of a policy. scope is the area, where this policy is meant for. This can be values like admin, selfservice, authentication... scope takes only one value.
active is bool and indicates, whether a policy is active or not.
action, realm, resolver, user and client can take a comma separated list of values.
If these are empty ‘*’, this policy matches each requested realm.
If the user is empty or ‘*’, this policy matches each user. You can exclude users from matching this policy, by prepending a ‘-‘ or a ‘!’. *, -admin will match for all users except the admin.
The client is identified by its IP address. A policy can contain a list of IP addresses or subnets. You can exclude clients from subnets by prepending the client with a ‘-‘ or a ‘!’. 172.16.0.0/24, -172.16.0.17 will match each client in the subnet except the 172.16.0.17.
This is the list of usual actions.
This is a list of usual action values for e.g. policy action-values like otppin.
This is the list of possible values for the login mode.
The Policy_Object will contain all database policy entries for easy filtering and mangling. It will be created at the beginning of the request and is supposed to stay alive unchanged during the request.
would return a list of the tokentypes
scope: authorization action: serial
would return a list of allowed serials
Returns: | A list of the allowed tokentypes |
---|---|
Return type: | list |
This is the list of the allowed scopes that can be used in policy definitions.
Function to delete one named policy
Parameters: | name – the name of the policy to be deleted |
---|---|
Returns: | the count of the deleted policies. |
Return type: | int |
Enable or disable the policy with the given name :param name: :return: ID of the policy
This function takes a policy list and creates an export file from it
Parameters: | policies (list of policy dictionaries) – a policy definition |
---|---|
Returns: | the contents of the file |
Return type: | string |
These are the static hard coded policy definitions. They can be enhanced by token based policy definitions, that can be found int lib.token.get_dynamic_policy_definitions.
Parameters: | scope (basestring) – Optional the scope of the policies |
---|---|
Returns: | allowed scopes with allowed actions, the type of action and a |
description. :rtype: dict
This function imports policies from a file. The file has a config_object format, i.e. the text file has a header
[<policy_name>] key = value
and key value pairs.
Parameters: | file_contents (basestring) – The contents of the file |
---|---|
Returns: | number of imported policies |
Return type: | int |
Function to set a policy. If the policy with this name already exists, it updates the policy. It expects a dict of with the following keys: :param name: The name of the policy :param scope: The scope of the policy. Something like “admin”, “system”, “authentication” :param action: A scope specific action or a comma seperated list of actions :type active: basestring :param realm: A realm, for which this policy is valid :param resolver: A resolver, for which this policy is valid :param user: A username or a list of usernames :param time: N/A if type() :param client: A client IP with optionally a subnet like 172.16.0.0/16 :param active: If the policy is active or not :type active: bool :return: The database ID od the the policy :rtype: int
These are the policy decorators as PRE conditions for the API calls. I.e. these conditions are executed before the wrapped API call. This module uses the policy base functions from privacyidea.lib.policy but also components from flask like g.
Wrapping the functions in a decorator class enables easy modular testing.
The functions of this module are tested in tests/test_api_lib_policy.py
This decorator function takes the request and verifies the given action for the SCOPE ADMIN or USER. :param req: :param action: :return: True otherwise raises an Exception
Pre Policy This checks the maximum token per realm. Check ACTION.MAXTOKENREALM
Parameters: |
|
---|---|
Returns: | True otherwise raises an Exception |
Pre Policy This checks the maximum token per user policy. Check ACTION.MAXTOKENUSER
Parameters: |
|
---|---|
Returns: | True otherwise raises an Exception |
This decorator function takes the request and verifies if the requested tokentype is allowed to be enrolled in the SCOPE ADMIN or the SCOPE USER. :param request: :param action: :return: True or an Exception is raised
This decorator function takes the request and verifies the given action for scope ADMIN :param req: :param filename: :return:
This policy function is to be used as a decorator in the API init function. If the policy is set accordingly it adds a random PIN to the request.all_data like.
It uses the policy SCOPE.ENROLL, ACTION.OTPPINRANDOM to set a random OTP PIN during Token enrollment
This policy function is to be used as a decorator in the API init function. It adds the tokenlabel definition to the params like this: params : { “tokenlabel”: “<u>@<r>” }
It uses the policy SCOPE.ENROLL, ACTION.TOKENLABEL to set the tokenlabel of Smartphone tokens during enrollment and this fill the details of the response.
This is the decorator wrapper to call a specific function before an API call. The prepolicy decorator is to be used in the API calls.
Pre Policy This pre condition gets the current realm and verifies if the realm should be rewritten due to the policy definition. I takes the realm from the request and - if a policy matches - replaces this realm with the realm defined in the policy
Check ACTION.SETREALM
Parameters: |
|
---|---|
Returns: | Always true. Modified the parameter request |
These are the policy decorators as POST conditions for the API calls. I.e. these conditions are executed after the wrapped API call. This module uses the policy base functions from privacyidea.lib.policy but also components from flask like g.
Wrapping the functions in a decorator class enables easy modular testing.
The functions of this module are tested in tests/test_api_lib_policy.py
This decorator decorates the function /validate/check. Depending on ACTION.AUTOASSIGN it checks if the user has no token and if the given OTP-value matches a token in the users realm, that is not yet assigned to any user.
If a token can be found, it assigns the token to the user also taking into account ACTION.MAXTOKENUSER and ACTION.MAXTOKENREALM. :return:
This policy function is to be used in a decorator of an API function. It checks, if the token, that was used in the API call has a serial number that is allowed to be used.
If not, a PolicyException is raised.
Parameters: | response (Response object) – The response of the decorated function |
---|---|
Returns: | A new (maybe modified) response |
This policy function is to be used in a decorator of an API function. It checks, if the token, that was used in the API call is of a type that is allowed to be used.
If not, a PolicyException is raised.
Parameters: | response (Response object) – The response of the decorated function |
---|---|
Returns: | A new (maybe modified) response |
This policy function is used with the AUTHZ scope. If the boolean value no_detail_on_fail is set, the details will be stripped if the authentication request failed.
Parameters: |
|
---|---|
Returns: |
These are the policy decorator functions for internal (lib) policy decorators. policy decorators for the API (pre/post) are defined in api/lib/policy
The functions of this module are tested in tests/test_lib_policy_decorator.py
Decorator to decorate the tokenclass.check_pin function. Depending on the ACTION.OTPPIN it * either simply accepts an empty pin * checks the pin against the userstore * or passes the request to the wrapped_function
Parameters: | wrapped_function – In this case the wrapped function should be |
---|
tokenclass.check_ping :param *args: args[1] is the pin :param **kwds: kwds[“options”] contains the flask g :return: True or False
This decorator checks, if the user does exist at all. If the user does exist, the wrapped function is called.
The wrapped function is usually token.check_user_pass, which takes the arguments (user, passw, options={})
Parameters: |
|
---|---|
Returns: | Tuple of True/False and reply-dictionary |
This decorator checks if the user has a token at all. If the user has a token, the wrapped function is called.
The wrapped function is usually token.check_user_pass, which takes the arguments (user, passw, options={})
Parameters: |
|
---|---|
Returns: | Tuple of True/False and reply-dictionary |
This decorator checks the policy settings of ACTION.PASSTHRU. If the authentication against the userstore is not successful, the wrapped function is called.
The wrapped function is usually token.check_user_pass, which takes the arguments (user, passw, options={})
Parameters: |
|
---|---|
Returns: | Tuple of True/False and reply-dictionary |
This is the decorator wrapper to call a specific function before a library call in contrast to prepolicy and postpolicy, which are to be called in API Calls.
The decorator expects a named parameter “options”. In this options dict it will look for the flask global “g”.
Decorator to decorate the lib.auth.check_webui_user function. Depending on ACTION.LOGINMODE it sets the check_otp paramter, to signal that the authentication should be performed against privacyIDEA.
Parameters: |
|
---|
kwds[“options”] contains the flask g :return: calls the original function with the modified “check_otp” argument
The useridresolver is responsible for getting userids for loginnames and vice versa.
This base module contains the base class UserIdResolver.UserIdResolver and also the community class PasswdIdResolver.IdResolver, that is inherited from the base class.
This function checks the password for a given uid. returns true in case of success false if password does not match
Parameters: |
|
---|---|
Returns: | True or False |
Return type: | bool |
return the descriptor of the resolver, which is - the class name and - the config description
Returns: | resolver description dict |
---|---|
Return type: | dict |
return the descriptor of the resolver, which is - the class name and - the config description
Returns: | resolver description dict |
---|---|
Return type: | dict |
get resolver specific information :return: the resolver identifier string - empty string if not exist
getResolverType - return the type of the resolver
Returns: | returns the string ‘ldapresolver’ |
---|---|
Return type: | string |
The loginname is resolved to a user_id. Depending on the resolver type the user_id can be an ID (like in /etc/passwd) or a string (like the DN in LDAP)
It needs to return an emptry string, if the user does not exist.
Parameters: | loginName (sting) – The login name of the user |
---|---|
Returns: | The ID of the user |
Return type: | string or int |
This function returns all user information for a given user object identified by UserID. :param userid: ID of the user in the resolver :type userid: int or string :return: dictionary, if no object is found, the dictionary is empty :rtype: dict
This function finds the user objects, that have the term ‘value’ in the user object field ‘key’
Parameters: | searchDict (dict) – dict with key values of user attributes - the key may be something like ‘loginname’ or ‘email’ the value is a regular expression. |
---|---|
Returns: | list of dictionaries (each dictionary contains a user object) or an empty string if no object is found. |
Return type: | list of dicts |
Returns the username/loginname for a given userid :param userid: The userid in this resolver :type userid: string :return: username :rtype: string
Load the configuration from the dict into the Resolver object. If attributes are missing, need to set default values. If required attributes are missing, this should raise an Exception.
Parameters: | config (dict) – The configuration values of the resolver |
---|
This function lets you test if the parameters can be used to create a working resolver. The implemenation should try to connect to the user store and verify if users can be retrieved. In case of success it should return a text like “Resolver config seems OK. 123 Users found.”
param param: The parameters that should be saved as the resolver type param: dict return: returns True in case of success and a descriptive text rtype: tuple
This function checks the password for a given uid. returns true in case of success false if password does not match
We do not support shadow passwords. so the seconds column of the passwd file needs to contain the crypted password
Parameters: |
|
---|---|
Returns: | True or False |
Return type: | bool |
Check if a userid matches a pattern. A pattern can be “=1000”, “>=1000”, “<2000” or “between 1000,2000”.
Parameters: |
|
---|---|
Returns: | True or False |
Return type: | bool |
return the descriptor of the resolver, which is - the class name and - the config description
Returns: | resolver description dict |
---|---|
Return type: | dict |
return the resolver identifier string, which in fact is filename, where it points to.
show, which search fields this userIdResolver supports
TODO: implementation is not completed
Parameters: | searchDict (dict) – fields, which can be queried |
---|---|
Returns: | dict of all searchFields |
Return type: | dict |
search the user id from the login name
Parameters: | LoginName – the login of the user |
---|---|
Returns: | the userId |
get some info about the user as we only have the loginId, we have to traverse the dict for the value
Parameters: |
|
---|---|
Returns: | dict of user info |
get a list of all users matching the search criteria of the searchdict
Parameters: | searchDict – dict of search expressions |
---|
Returns the username/loginname for a given userid :param userid: The userid in this resolver :type userid: string :return: username :rtype: string
The UserIdResolver could be configured from the pylons app config - here this could be the passwd file , whether it is /etc/passwd or /etc/shadow
This function checks the password for a given uid. - returns true in case of success - false if password does not match
return the descriptor of the resolver, which is - the class name and - the config description
Returns: | resolver description dict |
---|---|
Return type: | dict |
Returns the resolver Id This should be an Identifier of the resolver, preferable the type and the name of the resolver.
resolve the loginname to the userid.
Parameters: | LoginName (string) – The login name from the credentials |
---|---|
Returns: | UserId as found for the LoginName |
This function returns all user info for a given userid/object.
Parameters: | userId (string) – The userid of the object |
---|---|
Returns: | A dictionary with the keys defined in self.userinfo |
Return type: | dict |
Parameters: | searchDict (dict) – A dictionary with search parameters |
---|---|
Returns: | list of users, where each user is a dictionary |
Returns the username/loginname for a given user_id :param user_id: The user_id in this resolver :type user_id: string :return: username :rtype: string
Load the config from conf.
Parameters: | config (dict) – The configuration from the Config Table |
---|
manage.js: function save_ldap_config
‘#ldap_uri’: ‘LDAPURI’, ‘#ldap_basedn’: ‘LDAPBASE’, ‘#ldap_binddn’: ‘BINDDN’, ‘#ldap_password’: ‘BINDPW’, ‘#ldap_timeout’: ‘TIMEOUT’, ‘#ldap_sizelimit’: ‘SIZELIMIT’, ‘#ldap_loginattr’: ‘LOGINNAMEATTRIBUTE’, ‘#ldap_searchfilter’: ‘LDAPSEARCHFILTER’, ‘#ldap_userfilter’: ‘LDAPFILTER’, ‘#ldap_mapping’: ‘USERINFO’, ‘#ldap_uidtype’: ‘UIDTYPE’, ‘#ldap_noreferrals’ : ‘NOREFERRALS’, ‘#ldap_certificate’: ‘CACERTIFICATE’,
this setup hook is triggered, when the server starts to serve the first request
Parameters: | config (the privacyidea config dict) – the privacyidea config |
---|
This function lets you test the to be saved LDAP connection.
This is taken from controllers/admin.py
Parameters: | param (dict) – A dictionary with all necessary parameter to test the connection. |
---|---|
Returns: | Tuple of success and a description |
Return type: | (bool, string) |
This is the REST API for privacyidea. It lets you create the system configuration, which is denoted in the system endpoints.
Special system configuration is the configuration of
- the resolvers
- the realms
- the defaultrealm
- the policies.
Resolvers are dynamic links to existing user sources. You can find users in LDAP directories, SQL databases, flat files or SCIM services. A resolver translates a loginname to a user object in the user source and back again. It is also responsible for fetching all additional needed information from the user source.
Realms are collections of resolvers that can be managed by administrators and where policies can be applied.
Defaultrealm is a special endpoint to define the default realm. The default realm is used if no user realm is specified. If a user from realm1 tries to authenticate or is addressed, the notation user@realm1 is used. If the @realm1 is ommitted, the user is searched in the default realm.
Policies are rules how privacyidea behaves and which user and administrator is allowed to do what.
Start to read about authentication to the API at Authentication endpoints.
Now you can take a look at the several REST endpoints.
return a paginated list of audit entries.
Params can be passed as key-value-pairs.
Example request:
GET /audit?realm=realm1 HTTP/1.1
Host: example.com
Accept: application/json
Example response:
HTTP/1.1 200 OK
Content-Type: application/json
{
"id": 1,
"jsonrpc": "2.0",
"result": {
"status": true,
"value": [
{
"serial": "....",
"missing_line": "..."
}
]
},
"version": "privacyIDEA unknown"
}
Download the audit entry as CSV file.
Params can be passed as key-value-pairs.
Example request:
GET /audit/audit.csv?realm=realm1 HTTP/1.1
Host: example.com
Accept: text/csv
Example response:
HTTP/1.1 200 OK
Content-Type: text/csv
{
"id": 1,
"jsonrpc": "2.0",
"result": {
"status": true,
"value": [
{
"serial": "....",
"missing_line": "..."
}
]
},
"version": "privacyIDEA unknown"
}
You need to authenticate for all administrative tasks. If you are not authenticated, the API returns a 401 response.
To authenticate you need to send a POST request to /auth containing username and password.
This call verifies the credentials of the user and issues an authentication token, that is used for the later API calls. The authentication token has a validity, that is usually 1 hour.
JSON Parameters: | |
---|---|
|
|
Return: | A json response with an authentication token, that needs to be used in any further request. |
Status Codes: |
|
Example Authentication Request:
POST /auth HTTP/1.1
Host: example.com
Accept: application/json
username=admin
password=topsecret
Example Authentication Response:
HTTP/1.0 200 OK
Content-Length: 354
Content-Type: application/json
{
"id": 1,
"jsonrpc": "2.0",
"result": {
"status": true,
"value": {
"token": "eyJhbGciOiJIUz....jdpn9kIjuGRnGejmbFbM"
}
},
"version": "privacyIDEA unknown"
}
Response for failed authentication:
HTTP/1.1 401 UNAUTHORIZED
Content-Type: application/json
Content-Length: 203
{
"id": 1,
"jsonrpc": "2.0",
"result": {
"error": {
"code": -401,
"message": "missing Authorization header"
},
"status": false
},
"version": "privacyIDEA unknown"
}
Example Request:
Requests to privacyidea then should use this security token in the Authorization field in the header.
GET /users/ HTTP/1.1
Host: example.com
Accept: application/json
Authorization: eyJhbGciOiJIUz....jdpn9kIjuGRnGejmbFbM
check the authentication for a user or a serial number. Either a serial or a user is required to authenticate. The PIN and OTP value is sent in the parameter pass.
TODO: This API call does not honour the policies AUTHZ:tokentype and AUTHZ:serial!
Parameters: |
|
---|---|
Return: | a json result with a boolean “result”: true |
Example response for a succesful authentication:
HTTP/1.1 200 OK Content-Type: application/text :-)
check the authentication for a user or a serial number. Either a serial or a user is required to authenticate. The PIN and OTP value is sent in the parameter pass.
TODO: This API call does not honour the policies AUTHZ:tokentype and AUTHZ:serial!
Parameters: |
|
---|---|
Return: | a json result with a boolean “result”: true |
Example response for a succesful authentication:
HTTP/1.1 200 OK Content-Type: application/text :-)
check the authentication for a user or a serial number. Either a serial or a user is required to authenticate. The PIN and OTP value is sent in the parameter pass.
Parameters: |
|
---|---|
Return: | a json result with a boolean “result”: true |
Example response for a successful authentication:
HTTP/1.1 200 OK Content-Type: application/json { "detail": { "message": "matching 1 tokens", "serial": "PISP0000AB00", "type": "spass" }, "id": 1, "jsonrpc": "2.0", "result": { "status": true, "value": true }, "version": "privacyIDEA unknown" }
check the authentication for a user or a serial number. Either a serial or a user is required to authenticate. The PIN and OTP value is sent in the parameter pass.
Parameters: |
|
---|---|
Return: | a json result with a boolean “result”: true |
Example response for a successful authentication:
HTTP/1.1 200 OK Content-Type: application/json { "detail": { "message": "matching 1 tokens", "serial": "PISP0000AB00", "type": "spass" }, "id": 1, "jsonrpc": "2.0", "result": { "status": true, "value": true }, "version": "privacyIDEA unknown" }
DefaultCountWindow - Default value for the coutner window DefaultOtpLen - Default value for the OTP value length –
usually 6 or 8
set a configuration key or a set of configuration entries
parameter are generic keyname=value pairs.
Parameters: |
|
---|
Parameters: |
|
---|---|
Return: | a json result with a boolean “result”: true |
This endpoint either returns all config entries or only the value of the one config key.
Parameters: |
|
---|---|
Return: | A json response or a single value, when queried with a key. |
Rtype: | json or scalar |
This endpoint either returns all config entries or only the value of the one config key.
Parameters: |
|
---|---|
Return: | A json response or a single value, when queried with a key. |
Rtype: | json or scalar |
delete a configuration key * if an error occurs an exception is serializedsetConfig and returned
Parameters: |
|
---|---|
Returns: | a json result with the deleted value |
The resolver endpoints is a subset of the system endpoint.
Return: | a json result with True, if the given values can create a |
---|
working resolver and a description.
returns a json list of all resolver.
Parameters: |
|
---|
This creates a new resolver or updates an existing one. A resolver is uniquely identified by its name.
If you update a resolver, you do not need to provide all parameters. Parameters you do not provide are left untouched. When updating a resolver you must not change the type! You do not need to specify the type, but if you specify a wrong type, it will produce an error.
Parameters: |
|
---|
ldapresolver, sqlresolver, scimresolver :type type: string :return: a json result with the value being the database id (>0)
Additional parameters depend on the resolver type.
- LDAP:
- LDAPURI LDAPBASE BINDDN BINDPW TIMEOUT SIZELIMIT LOGINNAMEATTRIBUTE LDAPSEARCHFILTER LDAPFILTER USERINFO NOREFERRALS - True|False
- SQL:
- Database Driver Server Port User Password Table Map
- Passwd
- Filename
this function deletes an existing resolver A resolver can not be deleted, if it is contained in a realm
Parameters: |
|
---|---|
Return: | json with success or fail |
This function retrieves the definition of a single resolver. If can be called via /system/getResolver?resolver= or via /resolver/<resolver>
Parameters: |
|
---|---|
Return: | a json result with the configuration of a specified resolver |
The realm endpoints are used to define realms. A realm groups together many users. Administrators can manage the tokens of the users in such a realm. Policies and tokens can be assigned to realms.
A realm consists of several resolvers. Thus you can create a realm and gather users from LDAP and flat file source into one realm or you can pick resolvers that collect users from different points from your vast LDAP directory and group these users into a realm.
You will only be able to see and use user object, that are contained in a realm.
The code of this module is tested in tests/test_api_system.py
This call returns the list of all defined realms. It take no arguments.
Return: | a json result with a list of realms |
---|
Example request:
GET / HTTP/1.1
Host: example.com
Accept: application/json
Example response:
HTTP/1.1 200 OK
Content-Type: application/json
{
"id": 1,
"jsonrpc": "2.0",
"result": {
"status": true,
"value": {
"realm1_with_resolver": {
"default": true,
"resolver": [
{
"name": "reso1_with_realm",
"type": "passwdresolver"
}
]
}
}
},
"version": "privacyIDEA unknown"
}
This call creates a new realm or reconfigures a realm. The realm contains a list of resolvers.
Parameters: |
|
---|
list object :type resolvers: string or list :return: a json result with a list of Realms
In the result it returns a list of added resolvers and a list of resolvers, that could not be added.
Example request:
To create a new realm “newrealm”, that consists of the resolvers “reso1_with_realm” and “reso2_with_realm” call:
POST /realm/newrealm HTTP/1.1
Host: example.com
Accept: application/json
Content-Length: 26
Content-Type: application/x-www-form-urlencoded
resolvers=resol_with_realm, reso2_with_realm
Example response:
HTTP/1.1 200 OK
Content-Type: application/json
{
"id": 1,
"jsonrpc": "2.0",
"result": {
"status": true,
"value": {
"added": ["reso1_with_realm", "reso2_with_realm"],
"failed": []
}
}
"version": "privacyIDEA unknown"
}
This call deletes the given realm.
Parameters: |
|
---|---|
Return: | a json result with value=1 if deleting the realm was successful |
Example request:
DELETE /realm/realm_to_delete HTTP/1.1
Host: example.com
Accept: application/json
Example response:
HTTP/1.1 200 OK
Content-Type: application/json
{
"id": 1,
"jsonrpc": "2.0",
"result": {
"status": true,
"value": 1
},
"version": "privacyIDEA unknown"
}
These endpoints are used to define the default realm, retrieve it and delete it.
This call deletes the default realm.
Return: | a json result with either 1 (success) or 0 (fail) |
---|
Example response:
{
"id": 1,
"jsonrpc": "2.0",
"result": {
"status": true,
"value": 1
},
"version": "privacyIDEA unknown"
}
This call returns the default realm
Return: | a json description of the default realm with the resolvers |
---|
Example response:
{
"id": 1,
"jsonrpc": "2.0",
"result": {
"status": true,
"value": {
"defrealm": {
"default": true,
"resolver": [
{
"name": "defresolver",
"type": "passwdresolver"
}
]
}
}
},
"version": "privacyIDEA unknown"
}
This call sets the default realm.
Parameters: |
|
---|---|
Return: | a json result with either 1 (success) or 0 (fail) |
Unssign a token from a user. You can either provide “serial” as an argument to unassign this very token or you can provide user and realm, to unassign all tokens of a user.
Return: | In case of success it returns “value”: True. |
---|---|
Rtype: | json object |
Copy the token user from one token to the other.
Parameters: |
|
---|
copy the pin. :type from: basestring :param to: the serial number of the single, from where you want to copy the pin. :type to: basestring :return: returns value=True in case of success :rtype: bool
Disable a single token or all the tokens of a user. Disabled tokens can not be used to authenticate but can be enabled again.
Parameters: |
|
---|---|
Return: | In case of success it returns the number of disabled |
tokens in “value”. :rtype: json object
Copy the token PIN from one token to the other.
Parameters: |
|
---|
copy the pin. :type from: basestring :param to: the serial number of the single, from where you want to copy the pin. :type to: basestring :return: returns value=True in case of success :rtype: bool
Assign a token to a user. The required arguments are serial, user and realm.
Return: | In case of success it returns “value”: True. |
---|---|
Rtype: | json object |
Enable a single token or all the tokens of a user.
Parameters: |
|
---|---|
Return: | In case of success it returns the number of enabled |
tokens in “value”. :rtype: json object
Resync the OTP token by providing two consecutive OTP values.
Parameters: |
|
---|---|
Return: | In case of success it returns “value”=True |
Rtype: | json object |
Set the the user pin or the SO PIN of the specific token. Usually these are smartcard or token specific PINs. E.g. the userpin is used with mOTP tokens to store the mOTP PIN.
The token is identified by the unique serial number.
Parameters: |
|
---|---|
Return: | In “value” returns the number of PINs set. |
Rtype: | json object |
Reset the failcounter of a single token or of all tokens of a user.
Parameters: |
|
---|---|
Return: | In case of success it returns “value”=True |
Rtype: | json object |
create a new token.
Parameters: |
|
---|
ocra arguments: for generating OCRA Tokens type=ocra you can specify the following parameters: :param ocrasuite: if you do not want to use the default
ocra suite OCRA-1:HOTP-SHA256-8:QA64
Parameters: |
|
---|---|
Return: | a json result with a boolean “result”: true |
Example response:
HTTP/1.1 200 OK Content-Type: application/json { "detail": { "googleurl": { "description": "URL for google Authenticator", "img": "<img width=250 src=""/>", "value": "otpauth://hotp/mylabel?secret=GEZDGNBVGY3TQOJQGEZDGNBVGY3TQOJQ&counter=0" }, "oathurl": { "description": "URL for OATH token", "img": "<img width=250 src=""/>", "value": "oathtoken:///addToken?name=mylabel&lockdown=true&key=3132333435363738393031323334353637383930" }, "otpkey": { "description": "OTP seed", "img": "<img width=200 src=""/>", "value": "seed://3132333435363738393031323334353637383930" }, "serial": "OATH00096020" }, "id": 1, "jsonrpc": "2.0", "result": { "status": true, "value": true }, "version": "privacyIDEA unknown" }
This API is only to be used by the admin! This can be used to set token specific attributes like
- description
- count_window
- sync_window
- count_auth_max
- count_auth_success_max
- hashlib,
- max_failcount
The token is identified by the unique serial number or by the token owner. In the later case all tokens of the owner will be modified.
Parameters: |
|
---|---|
Return: | returns the number of attributes set in “value” |
Rtype: | json object |
The method is called at /token Display the list of available tokens.
Parameters: |
|
---|
not strict matching by specifying a serial like “OATH”. :param type: Display only token of type. You ca do a non strict matching by specifying a tokentype like “otp”, to file hotp and totp tokens. :param user: display tokens of this user :param viewrealm: takes a realm, only the tokens in this realm will be displayed :param description: Display token with this kind of description :type description: basestring :param sortby: sort the output by column :param sortdir: asc/desc :param page: request a certain page :param assigned: Only return assigned (True) or not assigned (False) tokens :param pagesize: limit the number of returned tokens :param user_fields: additional user fields from the userid resolver of the owner (user) :param outform: if set to “csv”, than the token list will be given in CSV
Return: | a json result with the data being a list of token dictionaries { “data”: [ { <token1> }, { <token2> }. ] } |
---|---|
Rtype: | json |
Disable a single token or all the tokens of a user. Disabled tokens can not be used to authenticate but can be enabled again.
Parameters: |
|
---|---|
Return: | In case of success it returns the number of disabled |
tokens in “value”. :rtype: json object
Enable a single token or all the tokens of a user.
Parameters: |
|
---|---|
Return: | In case of success it returns the number of enabled |
tokens in “value”. :rtype: json object
Resync the OTP token by providing two consecutive OTP values.
Parameters: |
|
---|---|
Return: | In case of success it returns “value”=True |
Rtype: | json object |
Set the the user pin or the SO PIN of the specific token. Usually these are smartcard or token specific PINs. E.g. the userpin is used with mOTP tokens to store the mOTP PIN.
The token is identified by the unique serial number.
Parameters: |
|
---|---|
Return: | In “value” returns the number of PINs set. |
Rtype: | json object |
Reset the failcounter of a single token or of all tokens of a user.
Parameters: |
|
---|---|
Return: | In case of success it returns “value”=True |
Rtype: | json object |
Set the realms of a token. The token is identified by the unique serial number
Parameters: |
|
---|---|
Return: | returns value=True in case of success |
Rtype: | bool |
The call imports the given file containing token definitions. The file can be an OATH CSV file, an aladdin XML file or a Yubikey CSV file exported from the yubikey initialization tool.
The function is called as a POST request with the file upload.
Parameters: |
|
---|---|
Return: | The number of the imported tokens |
Rtype: | int |
Mark the specified token as lost and create a new temporary token. This new token gets the new serial number “lost<old-serial>” and a certain validity period and the PIN of the lost token.
Parameters: |
|
---|---|
Return: | returns value=True in case of success |
Rtype: | bool |
This API is only to be used by the admin! This can be used to set token specific attributes like
- description
- count_window
- sync_window
- count_auth_max
- count_auth_success_max
- hashlib,
- max_failcount
The token is identified by the unique serial number or by the token owner. In the later case all tokens of the owner will be modified.
Parameters: |
|
---|---|
Return: | returns the number of attributes set in “value” |
Rtype: | json object |
Delete a token by its serial number. There are three different ways to delete a token. You can call one of these URLs:
POST /token/delete?serial=<serial> DELETE /token/<serial> POST /token/delete/<serial>
Return: | In case of success it return the number of deleted tokens in |
---|
“value” :rtype: json object
The user endpoints is a subset of the system endpoint.
list the users in a realm
Parameters: |
|
---|---|
Return: | json result with “result”: true and the userlist in “value”. |
Example request:
GET /user?realm=realm1 HTTP/1.1
Host: example.com
Accept: application/json
Example response:
HTTP/1.1 200 OK
Content-Type: application/json
{
"id": 1,
"jsonrpc": "2.0",
"result": {
"status": true,
"value": [
{
"description": "Cornelius K\u00f6lbel,,+49 151 2960 1417,+49 561 3166797,cornelius.koelbel@netknights.it",
"email": "cornelius.koelbel@netknights.it",
"givenname": "Cornelius",
"mobile": "+49 151 2960 1417",
"phone": "+49 561 3166797",
"surname": "K\u00f6lbel",
"userid": "1009",
"username": "cornelius"
}
]
},
"version": "privacyIDEA unknown"
}
The policy endpoints are a subset of the system endpoint.
This function checks, if the given parameters would match a defined policy or not.
Query Parameters: | |
---|---|
|
|
Return: | a json result with the keys allowed and policy in the value key |
Rtype: | json |
Status Codes: |
|
Example request:
GET /policy/check?user=admin&realm=r1&client=172.16.1.1 HTTP/1.1
Host: example.com
Accept: application/json
Example response:
HTTP/1.0 200 OK
Content-Type: application/json
{
"id": 1,
"jsonrpc": "2.0",
"result": {
"status": true,
"value": {
"pol_update_del": {
"action": "enroll",
"active": true,
"client": "172.16.0.0/16",
"name": "pol_update_del",
"realm": "r1",
"resolver": "test",
"scope": "selfservice",
"time": "",
"user": "admin"
}
}
},
"version": "privacyIDEA unknown"
}
This is a helper function that returns the POSSIBLE policy definitions, that can be used to define your policies.
Parameters: |
|
---|---|
Return: | The policy definitions of the allowed scope with the actions and |
action types. The top level key is the scope. :rtype: dict
this function is used to retrieve the policies that you defined. It can also be used to export the policy to a file.
Parameters: |
|
---|---|
JSON Parameters: | |
|
|
Return: | a json result with the configuration of the specified policies |
Rtype: | json |
Status Codes: |
|
Example request:
In this example a policy “pol1” is created.
GET /policy/pol1 HTTP/1.1
Host: example.com
Accept: application/json
Example response:
HTTP/1.0 200 OK
Content-Type: application/json
{
"id": 1,
"jsonrpc": "2.0",
"result": {
"status": true,
"value": {
"pol_update_del": {
"action": "enroll",
"active": true,
"client": "1.1.1.1",
"name": "pol_update_del",
"realm": "r1",
"resolver": "test",
"scope": "selfservice",
"time": "",
"user": "admin"
}
}
},
"version": "privacyIDEA unknown"
}
Disable a given policy by its name. :param name: The name of the policy :return: ID in the database
Enable a given policy by its name. :param name: Name of the policy :return: ID in the database
this function is used to retrieve the policies that you defined. It can also be used to export the policy to a file.
Parameters: |
|
---|---|
JSON Parameters: | |
|
|
Return: | a json result with the configuration of the specified policies |
Rtype: | json |
Status Codes: |
|
Example request:
In this example a policy “pol1” is created.
GET /policy/pol1 HTTP/1.1
Host: example.com
Accept: application/json
Example response:
HTTP/1.0 200 OK
Content-Type: application/json
{
"id": 1,
"jsonrpc": "2.0",
"result": {
"status": true,
"value": {
"pol_update_del": {
"action": "enroll",
"active": true,
"client": "1.1.1.1",
"name": "pol_update_del",
"realm": "r1",
"resolver": "test",
"scope": "selfservice",
"time": "",
"user": "admin"
}
}
},
"version": "privacyIDEA unknown"
}
This function is used to import policies from a file.
Parameters: |
|
---|---|
Form Parameters: | |
|
|
Return: | A json response with the number of imported policies. |
Status Codes: |
|
Example request:
POST /policy/import/backup-policy.cfg HTTP/1.1
Host: example.com
Accept: application/json
Example response:
HTTP/1.0 200 OK
Content-Type: application/json
{
"id": 1,
"jsonrpc": "2.0",
"result": {
"status": true,
"value": 2
},
"version": "privacyIDEA unknown"
}
This is a helper function that returns the POSSIBLE policy definitions, that can be used to define your policies.
Parameters: |
|
---|---|
Return: | The policy definitions of the allowed scope with the actions and |
action types. The top level key is the scope. :rtype: dict
Creates a new policy that defines access or behaviour of different actions in privacyIDEA
Parameters: |
|
---|---|
JSON Parameters: | |
|
|
Return: | a json result with success or error |
Status Codes: |
|
Example request:
In this example a policy “pol1” is created.
POST /policy/pol1 HTTP/1.1
Host: example.com
Accept: application/json
scope=admin
realm=realm1
action=enroll, disable
Example response:
HTTP/1.0 200 OK
Content-Length: 354
Content-Type: application/json
{
"id": 1,
"jsonrpc": "2.0",
"result": {
"status": true,
"value": {
"setPolicy pol1": 1
}
},
"version": "privacyIDEA unknown"
}
this function is used to retrieve the policies that you defined. It can also be used to export the policy to a file.
Parameters: |
|
---|---|
JSON Parameters: | |
|
|
Return: | a json result with the configuration of the specified policies |
Rtype: | json |
Status Codes: |
|
Example request:
In this example a policy “pol1” is created.
GET /policy/pol1 HTTP/1.1
Host: example.com
Accept: application/json
Example response:
HTTP/1.0 200 OK
Content-Type: application/json
{
"id": 1,
"jsonrpc": "2.0",
"result": {
"status": true,
"value": {
"pol_update_del": {
"action": "enroll",
"active": true,
"client": "1.1.1.1",
"name": "pol_update_del",
"realm": "r1",
"resolver": "test",
"scope": "selfservice",
"time": "",
"user": "admin"
}
}
},
"version": "privacyIDEA unknown"
}
This deletes the policy of the given name.
Parameters: |
|
---|---|
Return: | a json result about the delete success. In case of success value > 0 |
Status Codes: |
|
Example request:
In this example a policy “pol1” is created.
DELETE /policy/pol1 HTTP/1.1
Host: example.com
Accept: application/json
Example response:
HTTP/1.0 200 OK
Content-Type: application/json
{
"id": 1,
"jsonrpc": "2.0",
"result": {
"status": true,
"value": 1
},
"version": "privacyIDEA unknown"
}
This function tests, if the given parameter will create a working machine resolver. The Machine Resolver Class itself verifies the functionality. This can also be network connectivity to a Machine Store.
Return: | a json result with bool |
---|
returns a json list of all machine resolver.
Parameters: |
|
---|
This creates a new machine resolver or updates an existing one. A resolver is uniquely identified by its name.
If you update a resolver, you do not need to provide all parameters. Parameters you do not provide are left untouched. When updating a resolver you must not change the type! You do not need to specify the type, but if you specify a wrong type, it will produce an error.
Parameters: |
|
---|---|
Return: | a json result with the value being the database id (>0) |
Additional parameters depend on the resolver type.
this function deletes an existing machine resolver
Parameters: |
|
---|---|
Return: | json with success or fail |
This function retrieves the definition of a single machine resolver.
Parameters: |
|
---|---|
Return: | a json result with the configuration of a specified resolver |
This sets a Machine Token option or deletes it, if the value is empty.
Parameters: |
|
---|
name :param resolver: identify the machine by the machine ID and the resolver name :param serial: identify the token by the serial number :param application: the name of the application like “luks” or “ssh”.
Parameters not listed will be treated as additional options.
Return: |
---|
This fetches the authentication items for a given application and the given client machine.
Parameters: |
|
---|
calculated. In case of the Yubikey this can be a challenge that produces a response. The authentication item is the combination of the challenge and the response.
Parameters: |
|
---|---|
Return: | dictionary with lists of authentication items |
Example response:
HTTP/1.1 200 OK
Content-Type: application/json
{
"id": 1,
"jsonrpc": "2.0",
"result": {
"status": true,
"value": { "ssh": [ { "username": "....",
"sshkey": "...."
}
],
"luks": [ { "slot": ".....",
"challenge": "...",
"response": "...",
"partition": "..."
]
}
},
"version": "privacyIDEA unknown"
}
Attach an existing token to a machine with a certain application.
Parameters: |
|
---|
name :param resolver: identify the machine by the machine ID and the resolver name :param serial: identify the token by the serial number :param application: the name of the application like “luks” or “ssh”.
Parameters not listed will be treated as additional options.
Return: | json result with “result”: true and the machine list in “value”. |
---|
Example request:
POST /token HTTP/1.1
Host: example.com
Accept: application/json
{ "hostname": "puckel.example.com",
"machienid": "12313098",
"resolver": "machineresolver1",
"serial": "tok123",
"application": "luks" }
Return a list of MachineTokens either for a given machine or for a given token.
Parameters: |
|
---|
name :param resolver: Identify the machine by the machine ID and the resolver name :return:
List all machines that can be found in the machine resolvers.
Parameters: |
|
---|
or “id”
Return: | json result with “result”: true and the machine list in “value”. |
---|
Example request:
GET /hostname?hostname=on HTTP/1.1
Host: example.com
Accept: application/json
Example response:
HTTP/1.1 200 OK
Content-Type: application/json
{
"id": 1,
"jsonrpc": "2.0",
"result": {
"status": true,
"value": [
{
"id": "908asljdas90ad0",
"hostname": [ "flavon.example.com", "test.example.com" ],
"ip": "1.2.3.4",
"resolver_name": "machineresolver1"
},
{
"id": "1908209x48x2183",
"hostname": [ "london.example.com" ],
"ip": "2.4.5.6",
"resolver_name": "machineresolver1"
}
]
},
"version": "privacyIDEA unknown"
}
Detach a token from a machine with a certain application.
Parameters: |
|
---|
name :param resolver: identify the machine by the machine ID and the resolver name :param serial: identify the token by the serial number :param application: the name of the application like “luks” or “ssh”.
Return: | json result with “result”: true and the machine list in “value”. |
---|
Example request:
DELETE /token HTTP/1.1
Host: example.com
Accept: application/json
{ "hostname": "puckel.example.com",
"resolver": "machineresolver1",
"application": "luks" }
This fetches the authentication items for a given application and the given client machine.
Parameters: |
|
---|
calculated. In case of the Yubikey this can be a challenge that produces a response. The authentication item is the combination of the challenge and the response.
Parameters: |
|
---|---|
Return: | dictionary with lists of authentication items |
Example response:
HTTP/1.1 200 OK
Content-Type: application/json
{
"id": 1,
"jsonrpc": "2.0",
"result": {
"status": true,
"value": { "ssh": [ { "username": "....",
"sshkey": "...."
}
],
"luks": [ { "slot": ".....",
"challenge": "...",
"response": "...",
"partition": "..."
]
}
},
"version": "privacyIDEA unknown"
}
If the searchQuery returns an iteretor with elements that are not a dictionary, the audit module needs to provide this function, to convert the audit entry to a dictionary.
A generator that can be used to stream the audit log
Parameters: | param – |
---|---|
Returns: |
This method is called to finalize the audit_data. I.e. sign the data and write it to the database. It should hash the data and do a hash chain and sign the data
This method returns the total number of audit entries in the audit store
This method initialized the log state. The fact, that the log state was initialized, also needs to be logged. Therefor the same params are passed as i the log method.
This method is used to log the data. During a request this method can be called several times to fill the internal audit_data dictionary.
Log the number of the tokens. Can be passed like log_token_num(get_tokens(count=True))
Parameters: | count (int) – Number of tokens |
---|---|
Returns: |
Set the private and public key for the audit class. This is achieved by passing the entries.
#priv = config.get(“privacyideaAudit.key.private”) #pub = config.get(“privacyideaAudit.key.public”)
Parameters: |
|
---|---|
Returns: | None |
This function is used to search audit events.
param: Search parameters can be passed.
return: A pagination object
This function is deprecated.
This is the SQLAudit module, which writes the audit entries to an SQL database table. It requires the configuration parameters. PI_AUDIT_SQL_URI
Deletes all entries in the database table. This is only used for test cases! :return:
Returns the audit log as csv file. :param config: The current flask app configuration :type config: dict :param param: The request parameters :type param: dict :param user: The user, who issued the request :return: None. It yields results as a generator
This method is used to log the data. It should hash the data and do a hash chain and sign the data
This method returns the total number of audit entries in the audit store
Add new log details in param to the internal log data self.audit_data.
Parameters: | param (dict) – Log data that is to be added |
---|---|
Returns: | None |
Machine Resolvers are used to find machines in directories like LDAP, Active Directory, puppet, salt, or the /etc/hosts file.
Machines can then be used to assign applications and tokens to those machines.
Returns a description what config values are expected and allowed.
Returns: | dict |
---|
Returns the machine id for a given hostname or IP address.
If hostname and ip is given, the resolver should also check that the hostname matches the IP. If it can check this and hostname and IP do not match, then an Exception must be raised.
Parameters: |
|
---|---|
Returns: | The machine ID, which depends on the resolver |
Return type: | basestring |
Return a list of all machine objects in this resolver
Parameters: | substring – If set to true, it will also match search_hostnames, |
---|
that only are a subnet of the machines hostname. :type substring: bool :param any: a substring that matches EITHER hostname, machineid or ip :type any: basestring :return: list of machine objects
Returns the machine id for a given hostname or IP address.
If hostname and ip is given, the resolver should also check that the hostname matches the IP. If it can check this and hostname and IP do not match, then an Exception must be raised.
Parameters: |
|
---|---|
Returns: | The machine ID, which depends on the resolver |
Return type: | basestring |
Return matching machines.
Parameters: |
|
---|---|
Returns: | list of Machine Objects |