2012-07-26 11:02:28 -04:00
#!/usr/bin/python
2012-08-02 21:29:10 -04:00
# -*- coding: utf-8 -*-
2012-07-26 11:02:28 -04:00
# This file is part of Ansible
#
# Ansible is free software: you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# Ansible is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with Ansible. If not, see <http://www.gnu.org/licenses/>.
2012-09-29 16:46:30 +02:00
DOCUMENTATION = '''
- - -
module : postgresql_user
short_description : Adds or removes a users ( roles ) from a PostgreSQL database .
description :
2012-09-30 12:21:35 +02:00
- Add or remove PostgreSQL users ( roles ) from a remote host and , optionally ,
grant the users access to an existing database or tables .
- The fundamental function of the module is to create , or delete , roles from
a PostgreSQL cluster . Privilege assignment , or removal , is an optional
step , which works on one database at a time . This allows for the module to
be called several times in the same module to modify the permissions on
different databases , or to grant permissions to already existing users .
2012-11-21 18:49:30 +01:00
- A user cannot be removed until all the privileges have been stripped from
2012-09-30 12:21:35 +02:00
the user . In such situation , if the module tries to remove the user it
will fail . To avoid this from happening the fail_on_user option signals
the module to try to remove the user , but if not possible keep going ; the
module will report if changes happened and separately if the user was
removed or not .
2013-11-27 21:23:03 -05:00
version_added : " 0.6 "
2012-09-29 16:46:30 +02:00
options :
name :
description :
- name of the user ( role ) to add or remove
required : true
default : null
password :
description :
2013-11-14 11:57:01 -05:00
- set the user ' s password, before 1.4 this was required.
2014-05-14 07:49:11 -06:00
- " When passing an encrypted password, the encrypted parameter must also be true, and it must be generated with the format C( ' str[ \\ " md5 \\" ] + md5[ password + username ] ' ), resulting in a total of 35 characters. An easy way to do this is: C(echo \\ " md5 ` echo - n \\" verysecretpasswordJOE \\ " | md5 ` \\" ). "
2013-10-26 11:32:16 -04:00
required : false
2012-09-29 16:46:30 +02:00
default : null
db :
description :
- name of database where permissions will be granted
required : false
default : null
fail_on_user :
description :
2012-11-21 18:49:30 +01:00
- if C ( yes ) , fail when user can ' t be removed. Otherwise just log and continue
2012-09-29 16:46:30 +02:00
required : false
2013-06-01 12:11:26 -04:00
default : ' yes '
2012-09-29 16:46:30 +02:00
choices : [ " yes " , " no " ]
2014-02-07 18:25:26 +02:00
port :
description :
- Database port to connect to .
required : false
default : 5432
2012-09-29 16:46:30 +02:00
login_user :
description :
- User ( role ) used to authenticate with PostgreSQL
required : false
default : postgres
login_password :
description :
- Password used to authenticate with PostgreSQL
required : false
default : null
login_host :
description :
- Host running PostgreSQL .
required : false
default : localhost
2014-09-29 16:06:28 -07:00
login_unix_socket :
2014-12-03 08:16:59 -08:00
description :
2014-09-29 16:06:28 -07:00
- Path to a Unix domain socket for local connections
required : false
default : null
2012-09-29 16:46:30 +02:00
priv :
description :
2012-09-30 12:21:35 +02:00
- " PostgreSQL privileges string in the format: C(table:priv1,priv2) "
2012-09-29 16:46:30 +02:00
required : false
default : null
2012-10-15 14:42:06 -07:00
role_attr_flags :
description :
- " PostgreSQL role attributes string in the format: CREATEDB,CREATEROLE,SUPERUSER "
required : false
default : null
choices : [ " [NO]SUPERUSER " , " [NO]CREATEROLE " , " [NO]CREATEUSER " , " [NO]CREATEDB " ,
" [NO]INHERIT " , " [NO]LOGIN " , " [NO]REPLICATION " ]
2012-09-29 16:46:30 +02:00
state :
description :
2013-04-01 12:44:22 +02:00
- The user ( role ) state
2012-09-29 16:46:30 +02:00
required : false
default : present
choices : [ " present " , " absent " ]
2013-10-26 11:32:16 -04:00
encrypted :
description :
- denotes if the password is already encrypted . boolean .
required : false
default : false
2013-10-26 11:37:45 -04:00
version_added : ' 1.4 '
2013-10-26 11:32:16 -04:00
expires :
description :
- sets the user ' s password expiration.
required : false
default : null
2013-10-26 11:37:45 -04:00
version_added : ' 1.4 '
2012-09-29 16:46:30 +02:00
notes :
2012-09-30 12:21:35 +02:00
- The default authentication assumes that you are either logging in as or
sudo ' ing to the postgres account on the host.
- This module uses psycopg2 , a Python PostgreSQL database adapter . You must
ensure that psycopg2 is installed on the host before using this module . If
the remote host is the PostgreSQL server ( which is the default case ) , then
PostgreSQL must also be installed on the remote host . For Ubuntu - based
systems , install the postgresql , libpq - dev , and python - psycopg2 packages
on the remote host before using this module .
2013-02-18 22:59:51 -05:00
- If you specify PUBLIC as the user , then the privilege changes will apply
to all users . You may not specify password or role_attr_flags when the
PUBLIC user is specified .
2012-09-29 16:46:30 +02:00
requirements : [ psycopg2 ]
author : Lorin Hochstein
'''
2013-06-14 11:53:43 +02:00
EXAMPLES = '''
# Create django user and grant access to database and products table
2013-11-02 22:42:44 -07:00
- postgresql_user : db = acme name = django password = ceec4eif7ya priv = CONNECT / products : ALL
2013-06-14 11:53:43 +02:00
# Create rails user, grant privilege to create other databases and demote rails from super user status
2013-11-02 22:42:44 -07:00
- postgresql_user : name = rails password = secret role_attr_flags = CREATEDB , NOSUPERUSER
2013-06-14 11:53:43 +02:00
# Remove test user privileges from acme
2013-11-02 22:42:44 -07:00
- postgresql_user : db = acme name = test priv = ALL / products : ALL state = absent fail_on_user = no
2013-06-14 11:53:43 +02:00
# Remove test user from test database and the cluster
2013-11-02 22:42:44 -07:00
- postgresql_user : db = test name = test priv = ALL state = absent
2013-06-14 11:53:43 +02:00
# Example privileges string format
INSERT , UPDATE / table : SELECT / anothertable : ALL
2013-10-26 11:37:45 -04:00
# Remove an existing user's password
- postgresql_user : db = test user = test password = NULL
2013-06-14 11:53:43 +02:00
'''
2012-08-21 14:25:19 -05:00
import re
2014-11-24 22:30:10 -08:00
import itertools
2012-08-21 14:25:19 -05:00
2012-07-26 11:02:28 -04:00
try :
import psycopg2
except ImportError :
postgresqldb_found = False
else :
postgresqldb_found = True
2014-11-24 22:30:10 -08:00
_flags = ( ' SUPERUSER ' , ' CREATEROLE ' , ' CREATEUSER ' , ' CREATEDB ' , ' INHERIT ' , ' LOGIN ' , ' REPLICATION ' )
2014-11-25 00:44:18 -08:00
VALID_FLAGS = frozenset ( itertools . chain ( _flags , ( ' NO %s ' % f for f in _flags ) ) )
2014-11-24 22:30:10 -08:00
2014-12-03 14:43:20 -08:00
VALID_PRIVS = dict ( table = frozenset ( ( ' SELECT ' , ' INSERT ' , ' UPDATE ' , ' DELETE ' , ' TRUNCATE ' , ' REFERENCES ' , ' TRIGGER ' , ' ALL ' , ' USAGE ' ) ) ,
database = frozenset ( ( ' CREATE ' , ' CONNECT ' , ' TEMPORARY ' , ' TEMP ' , ' ALL ' , ' USAGE ' ) ) ,
2014-11-24 22:30:10 -08:00
)
class InvalidFlagsError ( Exception ) :
pass
class InvalidPrivsError ( Exception ) :
pass
2012-07-26 11:02:28 -04:00
# ===========================================
# PostgreSQL module specific support methods.
#
def user_exists ( cursor , user ) :
2013-02-18 21:33:36 -05:00
# The PUBLIC user is a special case that is always there
if user == ' PUBLIC ' :
return True
2012-07-26 11:02:28 -04:00
query = " SELECT rolname FROM pg_roles WHERE rolname= %(user)s "
cursor . execute ( query , { ' user ' : user } )
return cursor . rowcount > 0
2014-11-24 22:30:10 -08:00
def user_add ( cursor , user , password , role_attr_flags , encrypted , expires ) :
2013-04-01 12:44:22 +02:00
""" Create a new database user (role). """
2014-11-24 22:30:10 -08:00
# Note: role_attr_flags escaped by parse_role_attrs and encrypted is a literal
query_password_data = dict ( password = password , expires = expires )
query = [ ' CREATE USER %(user)s ' % { " user " : pg_quote_identifier ( user , ' role ' ) } ]
2013-10-26 11:32:16 -04:00
if password is not None :
2014-11-24 22:30:10 -08:00
query . append ( " WITH %(crypt)s " % { " crypt " : encrypted } )
query . append ( " PASSWORD %(password)s " )
2013-10-26 11:32:16 -04:00
if expires is not None :
2014-11-24 22:30:10 -08:00
query . append ( " VALID UNTIL %(expires)s " )
2014-11-25 11:46:41 -07:00
query . append ( role_attr_flags )
2014-11-24 22:30:10 -08:00
query = ' ' . join ( query )
2014-02-03 17:07:17 -05:00
cursor . execute ( query , query_password_data )
2012-07-26 11:02:28 -04:00
return True
2013-12-31 18:23:45 +00:00
def user_alter ( cursor , module , user , password , role_attr_flags , encrypted , expires ) :
2013-04-01 12:44:22 +02:00
""" Change user password and/or attributes. Return True if changed, False otherwise. """
2012-07-26 11:02:28 -04:00
changed = False
2014-11-24 22:30:10 -08:00
# Note: role_attr_flags escaped by parse_role_attrs and encrypted is a literal
2013-02-18 21:33:36 -05:00
if user == ' PUBLIC ' :
if password is not None :
module . fail_json ( msg = " cannot change the password for PUBLIC user " )
elif role_attr_flags != ' ' :
module . fail_json ( msg = " cannot change the role_attr_flags for PUBLIC user " )
else :
return False
2012-07-26 11:02:28 -04:00
# Handle passwords.
2012-10-15 14:42:06 -07:00
if password is not None or role_attr_flags is not None :
# Select password and all flag-like columns in order to verify changes.
2014-11-24 22:30:10 -08:00
query_password_data = dict ( password = password , expires = expires )
2012-12-05 09:51:15 +01:00
select = " SELECT * FROM pg_authid where rolname= %(user)s "
cursor . execute ( select , { " user " : user } )
2012-10-15 14:42:06 -07:00
# Grab current role attributes.
current_role_attrs = cursor . fetchone ( )
2014-11-25 12:04:47 -07:00
alter = [ ' ALTER USER %(user)s ' % { " user " : pg_quote_identifier ( user , ' role ' ) } ]
2012-10-15 14:42:06 -07:00
if password is not None :
2014-11-24 22:30:10 -08:00
alter . append ( " WITH %(crypt)s " % { " crypt " : encrypted } )
alter . append ( " PASSWORD %(password)s " )
alter . append ( role_attr_flags )
2013-10-26 11:32:16 -04:00
elif role_attr_flags :
2014-11-24 22:30:10 -08:00
alter . append ( ' WITH %s ' % role_attr_flags )
2013-10-26 11:32:16 -04:00
if expires is not None :
2014-11-24 22:30:10 -08:00
alter . append ( " VALID UNTIL %(expires)s " )
2013-10-26 11:32:16 -04:00
2013-12-31 18:23:45 +00:00
try :
2014-11-25 12:04:47 -07:00
cursor . execute ( ' ' . join ( alter ) , query_password_data )
2013-12-31 18:23:45 +00:00
except psycopg2 . InternalError , e :
if e . pgcode == ' 25006 ' :
# Handle errors due to read-only transactions indicated by pgcode 25006
# ERROR: cannot execute ALTER ROLE in a read-only transaction
changed = False
module . fail_json ( msg = e . pgerror )
return changed
else :
raise psycopg2 . InternalError , e
2013-10-26 11:32:16 -04:00
2012-10-15 14:42:06 -07:00
# Grab new role attributes.
2012-12-05 09:51:15 +01:00
cursor . execute ( select , { " user " : user } )
2012-10-15 14:42:06 -07:00
new_role_attrs = cursor . fetchone ( )
# Detect any differences between current_ and new_role_attrs.
for i in range ( len ( current_role_attrs ) ) :
if current_role_attrs [ i ] != new_role_attrs [ i ] :
changed = True
2012-07-26 11:02:28 -04:00
return changed
2012-08-21 11:20:16 -05:00
def user_delete ( cursor , user ) :
""" Try to remove a user. Returns True if successful otherwise False """
cursor . execute ( " SAVEPOINT ansible_pgsql_user_delete " )
try :
2014-11-24 22:30:10 -08:00
cursor . execute ( " DROP USER %s " % pg_quote_identifier ( user , ' role ' ) )
2012-08-21 11:20:16 -05:00
except :
cursor . execute ( " ROLLBACK TO SAVEPOINT ansible_pgsql_user_delete " )
cursor . execute ( " RELEASE SAVEPOINT ansible_pgsql_user_delete " )
return False
Fix postgresql_user bug
If I create a database from scratch and assign permissions by doing:
- name: ensure database is created
action: postgresql_db db=$dbname
- name: ensure django user has access
action: postgresql_user db=$dbname user=$dbuser priv=ALL password=$dbpassword
Then it fails with the error:
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 565, in <module>
main()
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 273, in main
changed = grant_privileges(cursor, user, privs) or changed
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 174, in grant_privileges
changed = grant_func(cursor, user, name, privilege)\
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 132, in grant_database_privilege
prev_priv = get_database_privileges(cursor, user, db)
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 118, in get_database_privileges
r = re.search('%s=(C?T?c?)/[a-z]+\,?' % user, datacl)
File "/usr/lib/python2.7/re.py", line 142, in search
return _compile(pattern, flags).search(string)
TypeError: expected string or buffer
This fix fixes the problem by not executing the regex if the
db query on pg_database returns None.
2012-09-07 16:24:00 -04:00
2012-08-21 11:20:16 -05:00
cursor . execute ( " RELEASE SAVEPOINT ansible_pgsql_user_delete " )
return True
def has_table_privilege ( cursor , user , table , priv ) :
2013-01-05 19:31:31 +01:00
query = ' SELECT has_table_privilege( %s , %s , %s ) '
2012-08-21 14:25:19 -05:00
cursor . execute ( query , ( user , table , priv ) )
2012-08-21 11:20:16 -05:00
return cursor . fetchone ( ) [ 0 ]
2012-08-21 14:25:19 -05:00
def get_table_privileges ( cursor , user , table ) :
if ' . ' in table :
schema , table = table . split ( ' . ' , 1 )
else :
schema = ' public '
query = ''' SELECT privilege_type FROM information_schema.role_table_grants
WHERE grantee = % s AND table_name = % s AND table_schema = % s '''
cursor . execute ( query , ( user , table , schema ) )
return set ( [ x [ 0 ] for x in cursor . fetchall ( ) ] )
Fix postgresql_user bug
If I create a database from scratch and assign permissions by doing:
- name: ensure database is created
action: postgresql_db db=$dbname
- name: ensure django user has access
action: postgresql_user db=$dbname user=$dbuser priv=ALL password=$dbpassword
Then it fails with the error:
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 565, in <module>
main()
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 273, in main
changed = grant_privileges(cursor, user, privs) or changed
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 174, in grant_privileges
changed = grant_func(cursor, user, name, privilege)\
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 132, in grant_database_privilege
prev_priv = get_database_privileges(cursor, user, db)
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 118, in get_database_privileges
r = re.search('%s=(C?T?c?)/[a-z]+\,?' % user, datacl)
File "/usr/lib/python2.7/re.py", line 142, in search
return _compile(pattern, flags).search(string)
TypeError: expected string or buffer
This fix fixes the problem by not executing the regex if the
db query on pg_database returns None.
2012-09-07 16:24:00 -04:00
2012-08-21 11:20:16 -05:00
def grant_table_privilege ( cursor , user , table , priv ) :
2014-11-24 22:30:10 -08:00
# Note: priv escaped by parse_privs
2012-08-21 14:25:19 -05:00
prev_priv = get_table_privileges ( cursor , user , table )
2014-01-04 15:34:05 -06:00
query = ' GRANT %s ON TABLE %s TO %s ' % (
2014-11-24 22:30:10 -08:00
priv , pg_quote_identifier ( table , ' table ' ) , pg_quote_identifier ( user , ' role ' ) )
2012-08-21 11:20:16 -05:00
cursor . execute ( query )
2012-08-21 14:25:19 -05:00
curr_priv = get_table_privileges ( cursor , user , table )
return len ( curr_priv ) > len ( prev_priv )
2012-08-21 11:20:16 -05:00
def revoke_table_privilege ( cursor , user , table , priv ) :
2014-11-24 22:30:10 -08:00
# Note: priv escaped by parse_privs
2012-08-21 14:25:19 -05:00
prev_priv = get_table_privileges ( cursor , user , table )
2014-01-04 15:34:05 -06:00
query = ' REVOKE %s ON TABLE %s FROM %s ' % (
2014-11-24 22:30:10 -08:00
priv , pg_quote_identifier ( table , ' table ' ) , pg_quote_identifier ( user , ' role ' ) )
2012-08-21 11:20:16 -05:00
cursor . execute ( query )
2012-08-21 14:25:19 -05:00
curr_priv = get_table_privileges ( cursor , user , table )
return len ( curr_priv ) < len ( prev_priv )
2012-08-21 11:20:16 -05:00
2012-08-21 14:25:19 -05:00
def get_database_privileges ( cursor , user , db ) :
priv_map = {
' C ' : ' CREATE ' ,
' T ' : ' TEMPORARY ' ,
' c ' : ' CONNECT ' ,
}
query = ' SELECT datacl FROM pg_database WHERE datname = %s '
cursor . execute ( query , ( db , ) )
datacl = cursor . fetchone ( ) [ 0 ]
Fix postgresql_user bug
If I create a database from scratch and assign permissions by doing:
- name: ensure database is created
action: postgresql_db db=$dbname
- name: ensure django user has access
action: postgresql_user db=$dbname user=$dbuser priv=ALL password=$dbpassword
Then it fails with the error:
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 565, in <module>
main()
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 273, in main
changed = grant_privileges(cursor, user, privs) or changed
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 174, in grant_privileges
changed = grant_func(cursor, user, name, privilege)\
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 132, in grant_database_privilege
prev_priv = get_database_privileges(cursor, user, db)
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 118, in get_database_privileges
r = re.search('%s=(C?T?c?)/[a-z]+\,?' % user, datacl)
File "/usr/lib/python2.7/re.py", line 142, in search
return _compile(pattern, flags).search(string)
TypeError: expected string or buffer
This fix fixes the problem by not executing the regex if the
db query on pg_database returns None.
2012-09-07 16:24:00 -04:00
if datacl is None :
return [ ]
2012-08-21 14:25:19 -05:00
r = re . search ( ' %s =(C?T?c?)/[a-z]+ \ ,? ' % user , datacl )
if r is None :
return [ ]
o = [ ]
for v in r . group ( 1 ) :
o . append ( priv_map [ v ] )
return o
2012-08-21 11:20:16 -05:00
def has_database_privilege ( cursor , user , db , priv ) :
2013-01-05 19:31:31 +01:00
query = ' SELECT has_database_privilege( %s , %s , %s ) '
2012-08-21 14:25:19 -05:00
cursor . execute ( query , ( user , db , priv ) )
2012-08-21 11:20:16 -05:00
return cursor . fetchone ( ) [ 0 ]
def grant_database_privilege ( cursor , user , db , priv ) :
2014-11-24 22:30:10 -08:00
# Note: priv escaped by parse_privs
2012-08-21 14:25:19 -05:00
prev_priv = get_database_privileges ( cursor , user , db )
2013-02-18 21:33:36 -05:00
if user == " PUBLIC " :
2014-11-24 22:30:10 -08:00
query = ' GRANT %s ON DATABASE %s TO PUBLIC ' % (
priv , pg_quote_identifier ( db , ' database ' ) )
2013-02-18 21:33:36 -05:00
else :
2014-11-24 22:30:10 -08:00
query = ' GRANT %s ON DATABASE %s TO %s ' % (
priv , pg_quote_identifier ( db , ' database ' ) ,
pg_quote_identifier ( user , ' role ' ) )
2012-08-21 11:20:16 -05:00
cursor . execute ( query )
2012-08-21 14:25:19 -05:00
curr_priv = get_database_privileges ( cursor , user , db )
return len ( curr_priv ) > len ( prev_priv )
2012-07-26 11:02:28 -04:00
2012-08-21 11:20:16 -05:00
def revoke_database_privilege ( cursor , user , db , priv ) :
2014-11-24 22:30:10 -08:00
# Note: priv escaped by parse_privs
2012-08-21 14:25:19 -05:00
prev_priv = get_database_privileges ( cursor , user , db )
2013-02-18 21:33:36 -05:00
if user == " PUBLIC " :
2014-11-24 22:30:10 -08:00
query = ' REVOKE %s ON DATABASE %s FROM PUBLIC ' % (
priv , pg_quote_identifier ( db , ' database ' ) )
2013-02-18 21:33:36 -05:00
else :
2014-11-24 22:30:10 -08:00
query = ' REVOKE %s ON DATABASE %s FROM %s ' % (
priv , pg_quote_identifier ( db , ' database ' ) ,
pg_quote_identifier ( user , ' role ' ) )
2012-08-21 11:20:16 -05:00
cursor . execute ( query )
2012-08-21 14:25:19 -05:00
curr_priv = get_database_privileges ( cursor , user , db )
return len ( curr_priv ) < len ( prev_priv )
2012-07-26 11:02:28 -04:00
2012-08-21 11:20:16 -05:00
def revoke_privileges ( cursor , user , privs ) :
if privs is None :
return False
2012-07-26 11:02:28 -04:00
2012-08-21 11:20:16 -05:00
changed = False
for type_ in privs :
revoke_func = {
' table ' : revoke_table_privilege ,
' database ' : revoke_database_privilege
} [ type_ ]
2012-08-21 14:25:19 -05:00
for name , privileges in privs [ type_ ] . iteritems ( ) :
2012-08-21 11:20:16 -05:00
for privilege in privileges :
changed = revoke_func ( cursor , user , name , privilege ) \
or changed
return changed
def grant_privileges ( cursor , user , privs ) :
if privs is None :
return False
changed = False
for type_ in privs :
grant_func = {
' table ' : grant_table_privilege ,
' database ' : grant_database_privilege
} [ type_ ]
2012-08-21 14:25:19 -05:00
for name , privileges in privs [ type_ ] . iteritems ( ) :
2012-08-21 11:20:16 -05:00
for privilege in privileges :
changed = grant_func ( cursor , user , name , privilege ) \
or changed
return changed
2012-10-15 14:42:06 -07:00
def parse_role_attrs ( role_attr_flags ) :
"""
Parse role attributes string for user creation .
Format :
attributes [ , attributes , . . . ]
Where :
attributes := CREATEDB , CREATEROLE , NOSUPERUSER , . . .
2014-11-24 22:30:10 -08:00
[ " [NO]SUPERUSER " , " [NO]CREATEROLE " , " [NO]CREATEUSER " , " [NO]CREATEDB " ,
" [NO]INHERIT " , " [NO]LOGIN " , " [NO]REPLICATION " ]
2012-10-15 14:42:06 -07:00
"""
2014-11-24 22:30:10 -08:00
if ' , ' in role_attr_flags :
2014-11-25 00:44:18 -08:00
flag_set = frozenset ( r . upper ( ) for r in role_attr_flags . split ( " , " ) )
2014-11-26 14:43:56 -08:00
elif role_attr_flags :
flag_set = frozenset ( ( role_attr_flags . upper ( ) , ) )
2014-11-24 22:30:10 -08:00
else :
2014-11-26 14:43:56 -08:00
flag_set = frozenset ( )
2014-11-25 10:44:04 -07:00
if not flag_set . issubset ( VALID_FLAGS ) :
2014-11-24 22:30:10 -08:00
raise InvalidFlagsError ( ' Invalid role_attr_flags specified: %s ' %
' ' . join ( flag_set . difference ( VALID_FLAGS ) ) )
o_flags = ' ' . join ( flag_set )
2012-10-15 14:42:06 -07:00
return o_flags
2012-08-21 11:20:16 -05:00
def parse_privs ( privs , db ) :
"""
Parse privilege string to determine permissions for database db .
Format :
privileges [ / privileges / . . . ]
Where :
privileges := DATABASE_PRIVILEGES [ , DATABASE_PRIVILEGES , . . . ] |
TABLE_NAME : TABLE_PRIVILEGES [ , TABLE_PRIVILEGES , . . . ]
"""
if privs is None :
return privs
2012-08-21 14:25:19 -05:00
o_privs = {
2012-08-21 11:20:16 -05:00
' database ' : { } ,
' table ' : { }
}
for token in privs . split ( ' / ' ) :
if ' : ' not in token :
type_ = ' database '
name = db
2014-11-26 14:43:56 -08:00
priv_set = frozenset ( x . strip ( ) . upper ( ) for x in token . split ( ' , ' ) if x . strip ( ) )
2012-08-21 11:20:16 -05:00
else :
type_ = ' table '
name , privileges = token . split ( ' : ' , 1 )
2014-11-26 14:43:56 -08:00
priv_set = frozenset ( x . strip ( ) . upper ( ) for x in privileges . split ( ' , ' ) if x . strip ( ) )
Fix postgresql_user bug
If I create a database from scratch and assign permissions by doing:
- name: ensure database is created
action: postgresql_db db=$dbname
- name: ensure django user has access
action: postgresql_user db=$dbname user=$dbuser priv=ALL password=$dbpassword
Then it fails with the error:
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 565, in <module>
main()
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 273, in main
changed = grant_privileges(cursor, user, privs) or changed
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 174, in grant_privileges
changed = grant_func(cursor, user, name, privilege)\
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 132, in grant_database_privilege
prev_priv = get_database_privileges(cursor, user, db)
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 118, in get_database_privileges
r = re.search('%s=(C?T?c?)/[a-z]+\,?' % user, datacl)
File "/usr/lib/python2.7/re.py", line 142, in search
return _compile(pattern, flags).search(string)
TypeError: expected string or buffer
This fix fixes the problem by not executing the regex if the
db query on pg_database returns None.
2012-09-07 16:24:00 -04:00
2014-11-24 22:30:10 -08:00
if not priv_set . issubset ( VALID_PRIVS [ type_ ] ) :
raise InvalidPrivsError ( ' Invalid privs specified for %s : %s ' %
( type_ , ' ' . join ( priv_set . difference ( VALID_PRIVS [ type_ ] ) ) ) )
2012-08-21 14:25:19 -05:00
o_privs [ type_ ] [ name ] = priv_set
Fix postgresql_user bug
If I create a database from scratch and assign permissions by doing:
- name: ensure database is created
action: postgresql_db db=$dbname
- name: ensure django user has access
action: postgresql_user db=$dbname user=$dbuser priv=ALL password=$dbpassword
Then it fails with the error:
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 565, in <module>
main()
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 273, in main
changed = grant_privileges(cursor, user, privs) or changed
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 174, in grant_privileges
changed = grant_func(cursor, user, name, privilege)\
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 132, in grant_database_privilege
prev_priv = get_database_privileges(cursor, user, db)
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 118, in get_database_privileges
r = re.search('%s=(C?T?c?)/[a-z]+\,?' % user, datacl)
File "/usr/lib/python2.7/re.py", line 142, in search
return _compile(pattern, flags).search(string)
TypeError: expected string or buffer
This fix fixes the problem by not executing the regex if the
db query on pg_database returns None.
2012-09-07 16:24:00 -04:00
2012-08-21 14:25:19 -05:00
return o_privs
2012-07-26 11:02:28 -04:00
# ===========================================
# Module execution.
#
def main ( ) :
module = AnsibleModule (
argument_spec = dict (
2012-07-29 12:47:44 -04:00
login_user = dict ( default = " postgres " ) ,
login_password = dict ( default = " " ) ,
login_host = dict ( default = " " ) ,
2014-09-29 16:06:28 -07:00
login_unix_socket = dict ( default = " " ) ,
2012-08-01 00:21:36 -04:00
user = dict ( required = True , aliases = [ ' name ' ] ) ,
2012-07-29 12:47:44 -04:00
password = dict ( default = None ) ,
2012-07-26 11:02:28 -04:00
state = dict ( default = " present " , choices = [ " absent " , " present " ] ) ,
2012-08-21 11:20:16 -05:00
priv = dict ( default = None ) ,
db = dict ( default = ' ' ) ,
2012-09-05 18:18:30 +02:00
port = dict ( default = ' 5432 ' ) ,
2014-03-11 22:56:51 -05:00
fail_on_user = dict ( type = ' bool ' , default = ' yes ' ) ,
2013-10-26 11:32:16 -04:00
role_attr_flags = dict ( default = ' ' ) ,
2014-03-11 22:56:51 -05:00
encrypted = dict ( type = ' bool ' , default = ' no ' ) ,
2013-10-26 11:32:16 -04:00
expires = dict ( default = None )
2013-02-20 22:12:25 +08:00
) ,
2013-02-26 20:30:33 -05:00
supports_check_mode = True
2012-07-26 11:02:28 -04:00
)
2013-02-20 22:12:25 +08:00
2012-07-26 11:02:28 -04:00
user = module . params [ " user " ]
2012-07-29 12:47:44 -04:00
password = module . params [ " password " ]
2012-07-26 11:02:28 -04:00
state = module . params [ " state " ]
2013-10-26 11:32:16 -04:00
fail_on_user = module . params [ " fail_on_user " ]
2012-07-26 11:02:28 -04:00
db = module . params [ " db " ]
2012-08-21 11:20:16 -05:00
if db == ' ' and module . params [ " priv " ] is not None :
module . fail_json ( msg = " privileges require a database to be specified " )
privs = parse_privs ( module . params [ " priv " ] , db )
2012-09-05 18:18:30 +02:00
port = module . params [ " port " ]
2014-11-24 22:30:10 -08:00
try :
role_attr_flags = parse_role_attrs ( module . params [ " role_attr_flags " ] )
except InvalidFlagsError , e :
module . fail_json ( msg = str ( e ) )
2013-10-26 11:32:16 -04:00
if module . params [ " encrypted " ] :
encrypted = " ENCRYPTED "
else :
encrypted = " UNENCRYPTED "
expires = module . params [ " expires " ]
2012-07-26 11:02:28 -04:00
if not postgresqldb_found :
module . fail_json ( msg = " the python psycopg2 module is required " )
Fix postgresql_user bug
If I create a database from scratch and assign permissions by doing:
- name: ensure database is created
action: postgresql_db db=$dbname
- name: ensure django user has access
action: postgresql_user db=$dbname user=$dbuser priv=ALL password=$dbpassword
Then it fails with the error:
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 565, in <module>
main()
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 273, in main
changed = grant_privileges(cursor, user, privs) or changed
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 174, in grant_privileges
changed = grant_func(cursor, user, name, privilege)\
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 132, in grant_database_privilege
prev_priv = get_database_privileges(cursor, user, db)
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 118, in get_database_privileges
r = re.search('%s=(C?T?c?)/[a-z]+\,?' % user, datacl)
File "/usr/lib/python2.7/re.py", line 142, in search
return _compile(pattern, flags).search(string)
TypeError: expected string or buffer
This fix fixes the problem by not executing the regex if the
db query on pg_database returns None.
2012-09-07 16:24:00 -04:00
# To use defaults values, keyword arguments must be absent, so
2012-08-17 10:37:02 -05:00
# check which values are empty and don't include in the **kw
# dictionary
2012-10-30 20:42:07 -04:00
params_map = {
2012-08-17 10:37:02 -05:00
" login_host " : " host " ,
" login_user " : " user " ,
2012-08-21 11:20:16 -05:00
" login_password " : " password " ,
2012-09-05 18:18:30 +02:00
" port " : " port " ,
Fix postgresql_user bug
If I create a database from scratch and assign permissions by doing:
- name: ensure database is created
action: postgresql_db db=$dbname
- name: ensure django user has access
action: postgresql_user db=$dbname user=$dbuser priv=ALL password=$dbpassword
Then it fails with the error:
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 565, in <module>
main()
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 273, in main
changed = grant_privileges(cursor, user, privs) or changed
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 174, in grant_privileges
changed = grant_func(cursor, user, name, privilege)\
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 132, in grant_database_privilege
prev_priv = get_database_privileges(cursor, user, db)
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 118, in get_database_privileges
r = re.search('%s=(C?T?c?)/[a-z]+\,?' % user, datacl)
File "/usr/lib/python2.7/re.py", line 142, in search
return _compile(pattern, flags).search(string)
TypeError: expected string or buffer
This fix fixes the problem by not executing the regex if the
db query on pg_database returns None.
2012-09-07 16:24:00 -04:00
" db " : " database "
2012-08-17 10:37:02 -05:00
}
Fix postgresql_user bug
If I create a database from scratch and assign permissions by doing:
- name: ensure database is created
action: postgresql_db db=$dbname
- name: ensure django user has access
action: postgresql_user db=$dbname user=$dbuser priv=ALL password=$dbpassword
Then it fails with the error:
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 565, in <module>
main()
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 273, in main
changed = grant_privileges(cursor, user, privs) or changed
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 174, in grant_privileges
changed = grant_func(cursor, user, name, privilege)\
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 132, in grant_database_privilege
prev_priv = get_database_privileges(cursor, user, db)
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 118, in get_database_privileges
r = re.search('%s=(C?T?c?)/[a-z]+\,?' % user, datacl)
File "/usr/lib/python2.7/re.py", line 142, in search
return _compile(pattern, flags).search(string)
TypeError: expected string or buffer
This fix fixes the problem by not executing the regex if the
db query on pg_database returns None.
2012-09-07 16:24:00 -04:00
kw = dict ( ( params_map [ k ] , v ) for ( k , v ) in module . params . iteritems ( )
2012-08-17 10:37:02 -05:00
if k in params_map and v != " " )
2014-09-29 16:06:28 -07:00
# If a login_unix_socket is specified, incorporate it here.
is_localhost = " host " not in kw or kw [ " host " ] == " " or kw [ " host " ] == " localhost "
if is_localhost and module . params [ " login_unix_socket " ] != " " :
kw [ " host " ] = module . params [ " login_unix_socket " ]
2012-07-26 11:02:28 -04:00
try :
2012-08-21 14:23:45 -05:00
db_connection = psycopg2 . connect ( * * kw )
2012-07-26 11:02:28 -04:00
cursor = db_connection . cursor ( )
2012-08-17 10:37:02 -05:00
except Exception , e :
2012-07-26 11:02:28 -04:00
module . fail_json ( msg = " unable to connect to database: %s " % e )
Fix postgresql_user bug
If I create a database from scratch and assign permissions by doing:
- name: ensure database is created
action: postgresql_db db=$dbname
- name: ensure django user has access
action: postgresql_user db=$dbname user=$dbuser priv=ALL password=$dbpassword
Then it fails with the error:
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 565, in <module>
main()
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 273, in main
changed = grant_privileges(cursor, user, privs) or changed
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 174, in grant_privileges
changed = grant_func(cursor, user, name, privilege)\
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 132, in grant_database_privilege
prev_priv = get_database_privileges(cursor, user, db)
File "/tmp/ansible-1347048449.32-29998829936529/postgresql_user", line 118, in get_database_privileges
r = re.search('%s=(C?T?c?)/[a-z]+\,?' % user, datacl)
File "/usr/lib/python2.7/re.py", line 142, in search
return _compile(pattern, flags).search(string)
TypeError: expected string or buffer
This fix fixes the problem by not executing the regex if the
db query on pg_database returns None.
2012-09-07 16:24:00 -04:00
2012-08-21 14:23:45 -05:00
kw = dict ( user = user )
2012-08-21 11:20:16 -05:00
changed = False
2012-08-21 14:23:45 -05:00
user_removed = False
2013-02-20 22:12:25 +08:00
2013-10-26 11:32:16 -04:00
if state == " present " :
2012-07-26 11:02:28 -04:00
if user_exists ( cursor , user ) :
2014-11-24 22:30:10 -08:00
try :
changed = user_alter ( cursor , module , user , password , role_attr_flags , encrypted , expires )
except SQLParseError , e :
module . fail_json ( msg = str ( e ) )
2012-07-26 11:02:28 -04:00
else :
2014-11-24 22:30:10 -08:00
try :
changed = user_add ( cursor , user , password , role_attr_flags , encrypted , expires )
except SQLParseError , e :
module . fail_json ( msg = str ( e ) )
try :
changed = grant_privileges ( cursor , user , privs ) or changed
except SQLParseError , e :
module . fail_json ( msg = str ( e ) )
2012-08-21 11:20:16 -05:00
else :
2012-07-26 11:02:28 -04:00
if user_exists ( cursor , user ) :
2013-10-26 11:32:16 -04:00
if module . check_mode :
changed = True
kw [ ' user_removed ' ] = True
else :
2014-11-24 22:30:10 -08:00
try :
changed = revoke_privileges ( cursor , user , privs )
user_removed = user_delete ( cursor , user )
except SQLParseError , e :
module . fail_json ( msg = str ( e ) )
2013-10-26 11:32:16 -04:00
changed = changed or user_removed
if fail_on_user and not user_removed :
msg = " unable to remove user "
module . fail_json ( msg = msg )
kw [ ' user_removed ' ] = user_removed
2012-08-21 11:20:16 -05:00
if changed :
2013-10-26 11:32:16 -04:00
if module . check_mode :
db_connection . rollback ( )
else :
db_connection . commit ( )
2012-08-21 14:23:45 -05:00
kw [ ' changed ' ] = changed
module . exit_json ( * * kw )
2012-07-26 11:02:28 -04:00
2013-12-02 15:13:49 -05:00
# import module snippets
2013-12-02 15:11:23 -05:00
from ansible . module_utils . basic import *
2014-11-24 22:30:10 -08:00
from ansible . module_utils . database import *
2012-07-26 11:02:28 -04:00
main ( )