postgresql_subscription: new module (#63661)
* postgresql_subscription: setup master-standby cluster into one container * postgresql_subscription: setup master-standby cluster into one container, fix * postgresql_subscription: setup master-standby cluster into one container, set up publication * postgresql_subscription: setup master-standby cluster into one container, add module template * postgresql_subscription: setup master-standby cluster into one container, fix tests * postgresql_subscription: setup master-standby cluster into one container, create subscr via shell * postgresql_subscription: setup replication, state stat * postgresql_subscription: add basic present mode * postgresql_subscription: add assertions * postgresql_subscription: add samples * postgresql_subscription: state absent, cascade * postgresql_subscription: add owner param * postgresql_subscription: update * postgresql_subscription: refresh * postgresql_subscription: doc, warns * postgresql_subscription: relinfo * postgresql_subscription: fixes * postgresql_subscription: fix CI tests
This commit is contained in:
parent
da25c2b07b
commit
22fe622589
14 changed files with 1761 additions and 0 deletions
|
@ -0,0 +1,731 @@
|
||||||
|
#!/usr/bin/python
|
||||||
|
# -*- coding: utf-8 -*-
|
||||||
|
|
||||||
|
# Copyright: (c) 2019, Andrew Klychkov (@Andersson007) <aaklychkov@mail.ru>
|
||||||
|
# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
|
||||||
|
|
||||||
|
from __future__ import absolute_import, division, print_function
|
||||||
|
__metaclass__ = type
|
||||||
|
|
||||||
|
|
||||||
|
ANSIBLE_METADATA = {'metadata_version': '1.1',
|
||||||
|
'status': ['preview'],
|
||||||
|
'supported_by': 'community'}
|
||||||
|
|
||||||
|
|
||||||
|
DOCUMENTATION = r'''
|
||||||
|
---
|
||||||
|
module: postgresql_subscription
|
||||||
|
short_description: Add, update, or remove PostgreSQL subscription
|
||||||
|
description:
|
||||||
|
- Add, update, or remove PostgreSQL subscription.
|
||||||
|
version_added: '2.10'
|
||||||
|
|
||||||
|
options:
|
||||||
|
name:
|
||||||
|
description:
|
||||||
|
- Name of the subscription to add, update, or remove.
|
||||||
|
type: str
|
||||||
|
required: yes
|
||||||
|
db:
|
||||||
|
description:
|
||||||
|
- Name of the database to connect to and where
|
||||||
|
the subscription state will be changed.
|
||||||
|
aliases: [ login_db ]
|
||||||
|
type: str
|
||||||
|
required: yes
|
||||||
|
state:
|
||||||
|
description:
|
||||||
|
- The subscription state.
|
||||||
|
- C(present) implies that if I(name) subscription doesn't exist, it will be created.
|
||||||
|
- C(absent) implies that if I(name) subscription exists, it will be removed.
|
||||||
|
- C(stat) implies that if I(name) subscription exists, returns current configuration.
|
||||||
|
- C(refresh) implies that if I(name) subscription exists, it will be refreshed.
|
||||||
|
Fetch missing table information from publisher. Always returns ``changed`` is ``True``.
|
||||||
|
This will start replication of tables that were added to the subscribed-to publications
|
||||||
|
since the last invocation of REFRESH PUBLICATION or since CREATE SUBSCRIPTION.
|
||||||
|
The existing data in the publications that are being subscribed to
|
||||||
|
should be copied once the replication starts.
|
||||||
|
- For more information about C(refresh) see U(https://www.postgresql.org/docs/current/sql-altersubscription.html).
|
||||||
|
type: str
|
||||||
|
choices: [ absent, present, refresh, stat ]
|
||||||
|
default: present
|
||||||
|
relinfo:
|
||||||
|
description:
|
||||||
|
- Get information of the state for each replicated relation in the subscription.
|
||||||
|
type: bool
|
||||||
|
default: false
|
||||||
|
owner:
|
||||||
|
description:
|
||||||
|
- Subscription owner.
|
||||||
|
- If I(owner) is not defined, the owner will be set as I(login_user) or I(session_role).
|
||||||
|
- Ignored when I(state) is not C(present).
|
||||||
|
type: str
|
||||||
|
publications:
|
||||||
|
description:
|
||||||
|
- The publication names on the publisher to use for the subscription.
|
||||||
|
- Ignored when I(state) is not C(present).
|
||||||
|
type: list
|
||||||
|
connparams:
|
||||||
|
description:
|
||||||
|
- The connection dict param-value to connect to the publisher.
|
||||||
|
- For more information see U(https://www.postgresql.org/docs/current/libpq-connect.html#LIBPQ-CONNSTRING).
|
||||||
|
- Ignored when I(state) is not C(present).
|
||||||
|
type: dict
|
||||||
|
cascade:
|
||||||
|
description:
|
||||||
|
- Drop subscription dependencies. Has effect with I(state=absent) only.
|
||||||
|
- Ignored when I(state) is not C(absent).
|
||||||
|
type: bool
|
||||||
|
default: false
|
||||||
|
subsparams:
|
||||||
|
description:
|
||||||
|
- Dictionary of optional parameters for a subscription, e.g. copy_data, enabled, create_slot, etc.
|
||||||
|
- For update the subscription allowed keys are C(enabled), C(slot_name), C(synchronous_commit), C(publication_name).
|
||||||
|
- See available parameters to create a new subscription
|
||||||
|
on U(https://www.postgresql.org/docs/current/sql-createsubscription.html).
|
||||||
|
- Ignored when I(state) is not C(present).
|
||||||
|
type: dict
|
||||||
|
|
||||||
|
notes:
|
||||||
|
- PostgreSQL version must be 10 or greater.
|
||||||
|
|
||||||
|
seealso:
|
||||||
|
- module: postgresql_publication
|
||||||
|
- name: CREATE SUBSCRIPTION reference
|
||||||
|
description: Complete reference of the CREATE SUBSCRIPTION command documentation.
|
||||||
|
link: https://www.postgresql.org/docs/current/sql-createsubscription.html
|
||||||
|
- name: ALTER SUBSCRIPTION reference
|
||||||
|
description: Complete reference of the ALTER SUBSCRIPTION command documentation.
|
||||||
|
link: https://www.postgresql.org/docs/current/sql-altersubscription.html
|
||||||
|
- name: DROP SUBSCRIPTION reference
|
||||||
|
description: Complete reference of the DROP SUBSCRIPTION command documentation.
|
||||||
|
link: https://www.postgresql.org/docs/current/sql-dropsubscription.html
|
||||||
|
|
||||||
|
author:
|
||||||
|
- Andrew Klychkov (@Andersson007) <aaklychkov@mail.ru>
|
||||||
|
|
||||||
|
extends_documentation_fragment:
|
||||||
|
- postgres
|
||||||
|
'''
|
||||||
|
|
||||||
|
EXAMPLES = r'''
|
||||||
|
- name: >
|
||||||
|
Create acme subscription in mydb database using acme_publication and
|
||||||
|
the following connection parameters to connect to the publisher.
|
||||||
|
Set the subscription owner as alice.
|
||||||
|
postgresql_subscription:
|
||||||
|
db: mydb
|
||||||
|
name: acme
|
||||||
|
state: present
|
||||||
|
publications: acme_publication
|
||||||
|
owner: alice
|
||||||
|
connparams:
|
||||||
|
host: 127.0.0.1
|
||||||
|
port: 5432
|
||||||
|
user: repl
|
||||||
|
password: replpass
|
||||||
|
dbname: mydb
|
||||||
|
|
||||||
|
- name: Assuming that acme subscription exists, try to change conn parameters
|
||||||
|
postgresql_subscription:
|
||||||
|
db: mydb
|
||||||
|
name: acme
|
||||||
|
connparams:
|
||||||
|
host: 127.0.0.1
|
||||||
|
port: 5432
|
||||||
|
user: repl
|
||||||
|
password: replpass
|
||||||
|
connect_timeout: 100
|
||||||
|
|
||||||
|
- name: Refresh acme publication
|
||||||
|
postgresql_subscription:
|
||||||
|
db: mydb
|
||||||
|
name: acme
|
||||||
|
state: refresh
|
||||||
|
|
||||||
|
- name: >
|
||||||
|
Return the configuration of subscription acme if exists in mydb database.
|
||||||
|
Also return state of replicated relations.
|
||||||
|
postgresql_subscription:
|
||||||
|
db: mydb
|
||||||
|
name: acme
|
||||||
|
state: stat
|
||||||
|
relinfo: yes
|
||||||
|
|
||||||
|
- name: Drop acme subscription from mydb with dependencies (cascade=yes)
|
||||||
|
postgresql_subscription:
|
||||||
|
db: mydb
|
||||||
|
name: acme
|
||||||
|
state: absent
|
||||||
|
cascade: yes
|
||||||
|
|
||||||
|
- name: Assuming that acme subscription exists and enabled, disable the subscription
|
||||||
|
postgresql_subscription:
|
||||||
|
db: mydb
|
||||||
|
name: acme
|
||||||
|
state: present
|
||||||
|
subsparams:
|
||||||
|
enabled: no
|
||||||
|
'''
|
||||||
|
|
||||||
|
RETURN = r'''
|
||||||
|
name:
|
||||||
|
description:
|
||||||
|
- Name of the subscription.
|
||||||
|
returned: always
|
||||||
|
type: str
|
||||||
|
sample: acme
|
||||||
|
exists:
|
||||||
|
description:
|
||||||
|
- Flag indicates the subscription exists or not at the end of runtime.
|
||||||
|
returned: always
|
||||||
|
type: bool
|
||||||
|
sample: true
|
||||||
|
queries:
|
||||||
|
description: List of executed queries.
|
||||||
|
returned: always
|
||||||
|
type: str
|
||||||
|
sample: [ 'DROP SUBSCRIPTION "mysubscription"' ]
|
||||||
|
initial_state:
|
||||||
|
description: Subscription configuration at the beginning of runtime.
|
||||||
|
returned: always
|
||||||
|
type: dict
|
||||||
|
sample: {"conninfo": {}, "enabled": true, "owner": "postgres", "slotname": "test", "synccommit": true}
|
||||||
|
final_state:
|
||||||
|
description: Subscription configuration at the end of runtime.
|
||||||
|
returned: always
|
||||||
|
type: dict
|
||||||
|
sample: {"conninfo": {}, "enabled": true, "owner": "postgres", "slotname": "test", "synccommit": true}
|
||||||
|
'''
|
||||||
|
|
||||||
|
from copy import deepcopy
|
||||||
|
|
||||||
|
try:
|
||||||
|
from psycopg2.extras import DictCursor
|
||||||
|
except ImportError:
|
||||||
|
# psycopg2 is checked by connect_to_db()
|
||||||
|
# from ansible.module_utils.postgres
|
||||||
|
pass
|
||||||
|
|
||||||
|
from ansible.module_utils.basic import AnsibleModule
|
||||||
|
from ansible.module_utils.postgres import (
|
||||||
|
connect_to_db,
|
||||||
|
exec_sql,
|
||||||
|
get_conn_params,
|
||||||
|
postgres_common_argument_spec,
|
||||||
|
)
|
||||||
|
from ansible.module_utils.six import iteritems
|
||||||
|
|
||||||
|
SUPPORTED_PG_VERSION = 10000
|
||||||
|
|
||||||
|
SUBSPARAMS_KEYS_FOR_UPDATE = ('enabled', 'synchronous_commit', 'slot_name')
|
||||||
|
|
||||||
|
|
||||||
|
################################
|
||||||
|
# Module functions and classes #
|
||||||
|
################################
|
||||||
|
|
||||||
|
def convert_conn_params(conn_dict):
|
||||||
|
"""Converts the passed connection dictionary to string.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
conn_dict (list): Dictionary which needs to be converted.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Connection string.
|
||||||
|
"""
|
||||||
|
conn_list = []
|
||||||
|
for (param, val) in iteritems(conn_dict):
|
||||||
|
conn_list.append('%s=%s' % (param, val))
|
||||||
|
|
||||||
|
return ' '.join(conn_list)
|
||||||
|
|
||||||
|
|
||||||
|
def convert_subscr_params(params_dict):
|
||||||
|
"""Converts the passed params dictionary to string.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
params_dict (list): Dictionary which needs to be converted.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Parameters string.
|
||||||
|
"""
|
||||||
|
params_list = []
|
||||||
|
for (param, val) in iteritems(params_dict):
|
||||||
|
if val is False:
|
||||||
|
val = 'false'
|
||||||
|
elif val is True:
|
||||||
|
val = 'true'
|
||||||
|
|
||||||
|
params_list.append('%s = %s' % (param, val))
|
||||||
|
|
||||||
|
return ', '.join(params_list)
|
||||||
|
|
||||||
|
|
||||||
|
class PgSubscription():
|
||||||
|
"""Class to work with PostgreSQL subscription.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
module (AnsibleModule): Object of AnsibleModule class.
|
||||||
|
cursor (cursor): Cursor object of psycopg2 library to work with PostgreSQL.
|
||||||
|
name (str): The name of the subscription.
|
||||||
|
db (str): The database name the subscription will be associated with.
|
||||||
|
|
||||||
|
Kwargs:
|
||||||
|
relinfo (bool): Flag indicates the relation information is needed.
|
||||||
|
|
||||||
|
Attributes:
|
||||||
|
module (AnsibleModule): Object of AnsibleModule class.
|
||||||
|
cursor (cursor): Cursor object of psycopg2 library to work with PostgreSQL.
|
||||||
|
name (str): Name of subscription.
|
||||||
|
executed_queries (list): List of executed queries.
|
||||||
|
attrs (dict): Dict with subscription attributes.
|
||||||
|
exists (bool): Flag indicates the subscription exists or not.
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __init__(self, module, cursor, name, db, relinfo):
|
||||||
|
self.module = module
|
||||||
|
self.cursor = cursor
|
||||||
|
self.name = name
|
||||||
|
self.db = db
|
||||||
|
self.relinfo = relinfo
|
||||||
|
self.executed_queries = []
|
||||||
|
self.attrs = {
|
||||||
|
'owner': None,
|
||||||
|
'enabled': None,
|
||||||
|
'synccommit': None,
|
||||||
|
'conninfo': {},
|
||||||
|
'slotname': None,
|
||||||
|
'publications': [],
|
||||||
|
'relinfo': None,
|
||||||
|
}
|
||||||
|
self.empty_attrs = deepcopy(self.attrs)
|
||||||
|
self.exists = self.check_subscr()
|
||||||
|
|
||||||
|
def get_info(self):
|
||||||
|
"""Refresh the subscription information.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
``self.attrs``.
|
||||||
|
"""
|
||||||
|
self.exists = self.check_subscr()
|
||||||
|
return self.attrs
|
||||||
|
|
||||||
|
def check_subscr(self):
|
||||||
|
"""Check the subscription and refresh ``self.attrs`` subscription attribute.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
True if the subscription with ``self.name`` exists, False otherwise.
|
||||||
|
"""
|
||||||
|
|
||||||
|
subscr_info = self.__get_general_subscr_info()
|
||||||
|
|
||||||
|
if not subscr_info:
|
||||||
|
# The subscription does not exist:
|
||||||
|
self.attrs = deepcopy(self.empty_attrs)
|
||||||
|
return False
|
||||||
|
|
||||||
|
self.attrs['owner'] = subscr_info.get('rolname')
|
||||||
|
self.attrs['enabled'] = subscr_info.get('subenabled')
|
||||||
|
self.attrs['synccommit'] = subscr_info.get('subenabled')
|
||||||
|
self.attrs['slotname'] = subscr_info.get('subslotname')
|
||||||
|
self.attrs['publications'] = subscr_info.get('subpublications')
|
||||||
|
if subscr_info.get('subconninfo'):
|
||||||
|
for param in subscr_info['subconninfo'].split(' '):
|
||||||
|
tmp = param.split('=')
|
||||||
|
try:
|
||||||
|
self.attrs['conninfo'][tmp[0]] = int(tmp[1])
|
||||||
|
except ValueError:
|
||||||
|
self.attrs['conninfo'][tmp[0]] = tmp[1]
|
||||||
|
|
||||||
|
if self.relinfo:
|
||||||
|
self.attrs['relinfo'] = self.__get_rel_info()
|
||||||
|
|
||||||
|
return True
|
||||||
|
|
||||||
|
def create(self, connparams, publications, subsparams, check_mode=True):
|
||||||
|
"""Create the subscription.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
connparams (str): Connection string in libpq style.
|
||||||
|
publications (list): Publications on the master to use.
|
||||||
|
subsparams (str): Parameters string in WITH () clause style.
|
||||||
|
|
||||||
|
Kwargs:
|
||||||
|
check_mode (bool): If True, don't actually change anything,
|
||||||
|
just make SQL, add it to ``self.executed_queries`` and return True.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
changed (bool): True if the subscription has been created, otherwise False.
|
||||||
|
"""
|
||||||
|
query_fragments = []
|
||||||
|
query_fragments.append("CREATE SUBSCRIPTION %s CONNECTION '%s' "
|
||||||
|
"PUBLICATION %s" % (self.name, connparams, ', '.join(publications)))
|
||||||
|
|
||||||
|
if subsparams:
|
||||||
|
query_fragments.append("WITH (%s)" % subsparams)
|
||||||
|
|
||||||
|
changed = self.__exec_sql(' '.join(query_fragments), check_mode=check_mode)
|
||||||
|
|
||||||
|
return changed
|
||||||
|
|
||||||
|
def update(self, connparams, publications, subsparams, check_mode=True):
|
||||||
|
"""Update the subscription.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
connparams (str): Connection string in libpq style.
|
||||||
|
publications (list): Publications on the master to use.
|
||||||
|
subsparams (dict): Dictionary of optional parameters.
|
||||||
|
|
||||||
|
Kwargs:
|
||||||
|
check_mode (bool): If True, don't actually change anything,
|
||||||
|
just make SQL, add it to ``self.executed_queries`` and return True.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
changed (bool): True if subscription has been updated, otherwise False.
|
||||||
|
"""
|
||||||
|
changed = False
|
||||||
|
|
||||||
|
if connparams:
|
||||||
|
if connparams != self.attrs['conninfo']:
|
||||||
|
changed = self.__set_conn_params(convert_conn_params(connparams),
|
||||||
|
check_mode=check_mode)
|
||||||
|
|
||||||
|
if publications:
|
||||||
|
if sorted(self.attrs['publications']) != sorted(publications):
|
||||||
|
changed = self.__set_publications(publications, check_mode=check_mode)
|
||||||
|
|
||||||
|
if subsparams:
|
||||||
|
params_to_update = []
|
||||||
|
|
||||||
|
for (param, value) in iteritems(subsparams):
|
||||||
|
if param == 'enabled':
|
||||||
|
if self.attrs['enabled'] and value is False:
|
||||||
|
changed = self.enable(enabled=False, check_mode=check_mode)
|
||||||
|
elif not self.attrs['enabled'] and value is True:
|
||||||
|
changed = self.enable(enabled=True, check_mode=check_mode)
|
||||||
|
|
||||||
|
elif param == 'synchronous_commit':
|
||||||
|
if self.attrs['synccommit'] is True and value is False:
|
||||||
|
params_to_update.append("%s = false" % param)
|
||||||
|
elif self.attrs['synccommit'] is False and value is True:
|
||||||
|
params_to_update.append("%s = true" % param)
|
||||||
|
|
||||||
|
elif param == 'slot_name':
|
||||||
|
if self.attrs['slotname'] and self.attrs['slotname'] != value:
|
||||||
|
params_to_update.append("%s = %s" % (param, value))
|
||||||
|
|
||||||
|
else:
|
||||||
|
self.module.warn("Parameter '%s' is not in params supported "
|
||||||
|
"for update '%s', ignored..." % (param, SUBSPARAMS_KEYS_FOR_UPDATE))
|
||||||
|
|
||||||
|
if params_to_update:
|
||||||
|
changed = self.__set_params(params_to_update, check_mode=check_mode)
|
||||||
|
|
||||||
|
return changed
|
||||||
|
|
||||||
|
def drop(self, cascade=False, check_mode=True):
|
||||||
|
"""Drop the subscription.
|
||||||
|
|
||||||
|
Kwargs:
|
||||||
|
cascade (bool): Flag indicates that the subscription needs to be deleted
|
||||||
|
with its dependencies.
|
||||||
|
check_mode (bool): If True, don't actually change anything,
|
||||||
|
just make SQL, add it to ``self.executed_queries`` and return True.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
changed (bool): True if the subscription has been removed, otherwise False.
|
||||||
|
"""
|
||||||
|
if self.exists:
|
||||||
|
query_fragments = ["DROP SUBSCRIPTION %s" % self.name]
|
||||||
|
if cascade:
|
||||||
|
query_fragments.append("CASCADE")
|
||||||
|
|
||||||
|
return self.__exec_sql(' '.join(query_fragments), check_mode=check_mode)
|
||||||
|
|
||||||
|
def set_owner(self, role, check_mode=True):
|
||||||
|
"""Set a subscription owner.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
role (str): Role (user) name that needs to be set as a subscription owner.
|
||||||
|
|
||||||
|
Kwargs:
|
||||||
|
check_mode (bool): If True, don't actually change anything,
|
||||||
|
just make SQL, add it to ``self.executed_queries`` and return True.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
True if successful, False otherwise.
|
||||||
|
"""
|
||||||
|
query = 'ALTER SUBSCRIPTION %s OWNER TO "%s"' % (self.name, role)
|
||||||
|
return self.__exec_sql(query, check_mode=check_mode)
|
||||||
|
|
||||||
|
def refresh(self, check_mode=True):
|
||||||
|
"""Refresh publication.
|
||||||
|
|
||||||
|
Fetches missing table info from publisher.
|
||||||
|
|
||||||
|
Kwargs:
|
||||||
|
check_mode (bool): If True, don't actually change anything,
|
||||||
|
just make SQL, add it to ``self.executed_queries`` and return True.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
True if successful, False otherwise.
|
||||||
|
"""
|
||||||
|
query = 'ALTER SUBSCRIPTION %s REFRESH PUBLICATION' % self.name
|
||||||
|
return self.__exec_sql(query, check_mode=check_mode)
|
||||||
|
|
||||||
|
def __set_params(self, params_to_update, check_mode=True):
|
||||||
|
"""Update optional subscription parameters.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
params_to_update (list): Parameters with values to update.
|
||||||
|
|
||||||
|
Kwargs:
|
||||||
|
check_mode (bool): If True, don't actually change anything,
|
||||||
|
just make SQL, add it to ``self.executed_queries`` and return True.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
True if successful, False otherwise.
|
||||||
|
"""
|
||||||
|
query = 'ALTER SUBSCRIPTION %s SET (%s)' % (self.name, ', '.join(params_to_update))
|
||||||
|
return self.__exec_sql(query, check_mode=check_mode)
|
||||||
|
|
||||||
|
def __set_conn_params(self, connparams, check_mode=True):
|
||||||
|
"""Update connection parameters.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
connparams (str): Connection string in libpq style.
|
||||||
|
|
||||||
|
Kwargs:
|
||||||
|
check_mode (bool): If True, don't actually change anything,
|
||||||
|
just make SQL, add it to ``self.executed_queries`` and return True.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
True if successful, False otherwise.
|
||||||
|
"""
|
||||||
|
query = "ALTER SUBSCRIPTION %s CONNECTION '%s'" % (self.name, connparams)
|
||||||
|
return self.__exec_sql(query, check_mode=check_mode)
|
||||||
|
|
||||||
|
def __set_publications(self, publications, check_mode=True):
|
||||||
|
"""Update publications.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
publications (list): Publications on the master to use.
|
||||||
|
|
||||||
|
Kwargs:
|
||||||
|
check_mode (bool): If True, don't actually change anything,
|
||||||
|
just make SQL, add it to ``self.executed_queries`` and return True.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
True if successful, False otherwise.
|
||||||
|
"""
|
||||||
|
query = 'ALTER SUBSCRIPTION %s SET PUBLICATION %s' % (self.name, ', '.join(publications))
|
||||||
|
return self.__exec_sql(query, check_mode=check_mode)
|
||||||
|
|
||||||
|
def enable(self, enabled=True, check_mode=True):
|
||||||
|
"""Enable or disable the subscription.
|
||||||
|
|
||||||
|
Kwargs:
|
||||||
|
enable (bool): Flag indicates that the subscription needs
|
||||||
|
to be enabled or disabled.
|
||||||
|
check_mode (bool): If True, don't actually change anything,
|
||||||
|
just make SQL, add it to ``self.executed_queries`` and return True.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
True if successful, False otherwise.
|
||||||
|
"""
|
||||||
|
if enabled:
|
||||||
|
query = 'ALTER SUBSCRIPTION %s ENABLE' % self.name
|
||||||
|
else:
|
||||||
|
query = 'ALTER SUBSCRIPTION %s DISABLE' % self.name
|
||||||
|
|
||||||
|
return self.__exec_sql(query, check_mode=check_mode)
|
||||||
|
|
||||||
|
def __get_general_subscr_info(self):
|
||||||
|
"""Get and return general subscription information.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Dict with subscription information if successful, False otherwise.
|
||||||
|
"""
|
||||||
|
query = ("SELECT d.datname, r.rolname, s.subenabled, "
|
||||||
|
"s.subconninfo, s.subslotname, s.subsynccommit, "
|
||||||
|
"s.subpublications FROM pg_catalog.pg_subscription s "
|
||||||
|
"JOIN pg_catalog.pg_database d "
|
||||||
|
"ON s.subdbid = d.oid "
|
||||||
|
"JOIN pg_catalog.pg_roles AS r "
|
||||||
|
"ON s.subowner = r.oid "
|
||||||
|
"WHERE s.subname = '%s' AND d.datname = '%s'" % (self.name, self.db))
|
||||||
|
|
||||||
|
result = exec_sql(self, query, add_to_executed=False)
|
||||||
|
if result:
|
||||||
|
return result[0]
|
||||||
|
else:
|
||||||
|
return False
|
||||||
|
|
||||||
|
def __get_rel_info(self):
|
||||||
|
"""Get and return state of relations replicated by the subscription.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
List of dicts containing relations state if successful, False otherwise.
|
||||||
|
"""
|
||||||
|
query = ("SELECT c.relname, r.srsubstate, r.srsublsn "
|
||||||
|
"FROM pg_catalog.pg_subscription_rel r "
|
||||||
|
"JOIN pg_catalog.pg_subscription s ON s.oid = r.srsubid "
|
||||||
|
"JOIN pg_catalog.pg_class c ON c.oid = r.srrelid "
|
||||||
|
"WHERE s.subname = '%s'" % self.name)
|
||||||
|
|
||||||
|
result = exec_sql(self, query, add_to_executed=False)
|
||||||
|
if result:
|
||||||
|
return [dict(row) for row in result]
|
||||||
|
else:
|
||||||
|
return False
|
||||||
|
|
||||||
|
def __exec_sql(self, query, check_mode=False):
|
||||||
|
"""Execute SQL query.
|
||||||
|
|
||||||
|
Note: If we need just to get information from the database,
|
||||||
|
we use ``exec_sql`` function directly.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
query (str): Query that needs to be executed.
|
||||||
|
|
||||||
|
Kwargs:
|
||||||
|
check_mode (bool): If True, don't actually change anything,
|
||||||
|
just add ``query`` to ``self.executed_queries`` and return True.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
True if successful, False otherwise.
|
||||||
|
"""
|
||||||
|
if check_mode:
|
||||||
|
self.executed_queries.append(query)
|
||||||
|
return True
|
||||||
|
else:
|
||||||
|
return exec_sql(self, query, ddl=True)
|
||||||
|
|
||||||
|
|
||||||
|
# ===========================================
|
||||||
|
# Module execution.
|
||||||
|
#
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
argument_spec = postgres_common_argument_spec()
|
||||||
|
argument_spec.update(
|
||||||
|
name=dict(required=True),
|
||||||
|
db=dict(type='str', aliases=['login_db']),
|
||||||
|
state=dict(type='str', default='present', choices=['absent', 'present', 'refresh', 'stat']),
|
||||||
|
publications=dict(type='list'),
|
||||||
|
connparams=dict(type='dict'),
|
||||||
|
cascade=dict(type='bool', default=False),
|
||||||
|
owner=dict(type='str'),
|
||||||
|
subsparams=dict(type='dict'),
|
||||||
|
relinfo=dict(type='bool', default=False),
|
||||||
|
)
|
||||||
|
module = AnsibleModule(
|
||||||
|
argument_spec=argument_spec,
|
||||||
|
supports_check_mode=True,
|
||||||
|
)
|
||||||
|
|
||||||
|
# Parameters handling:
|
||||||
|
db = module.params['db']
|
||||||
|
name = module.params['name']
|
||||||
|
state = module.params['state']
|
||||||
|
publications = module.params['publications']
|
||||||
|
cascade = module.params['cascade']
|
||||||
|
owner = module.params['owner']
|
||||||
|
subsparams = module.params['subsparams']
|
||||||
|
connparams = module.params['connparams']
|
||||||
|
relinfo = module.params['relinfo']
|
||||||
|
|
||||||
|
if state == 'present' and cascade:
|
||||||
|
module.warm('parameter "cascade" is ignored when state is not absent')
|
||||||
|
|
||||||
|
if state != 'present':
|
||||||
|
if owner:
|
||||||
|
module.warm("parameter 'owner' is ignored when state is not 'present'")
|
||||||
|
if publications:
|
||||||
|
module.warm("parameter 'publications' is ignored when state is not 'present'")
|
||||||
|
if connparams:
|
||||||
|
module.warm("parameter 'connparams' is ignored when state is not 'present'")
|
||||||
|
if subsparams:
|
||||||
|
module.warm("parameter 'subsparams' is ignored when state is not 'present'")
|
||||||
|
|
||||||
|
# Connect to DB and make cursor object:
|
||||||
|
pg_conn_params = get_conn_params(module, module.params)
|
||||||
|
# We check subscription state without DML queries execution, so set autocommit:
|
||||||
|
db_connection = connect_to_db(module, pg_conn_params, autocommit=True)
|
||||||
|
cursor = db_connection.cursor(cursor_factory=DictCursor)
|
||||||
|
|
||||||
|
# Check version:
|
||||||
|
if cursor.connection.server_version < SUPPORTED_PG_VERSION:
|
||||||
|
module.fail_json(msg="PostgreSQL server version should be 10.0 or greater")
|
||||||
|
|
||||||
|
# Set defaults:
|
||||||
|
changed = False
|
||||||
|
initial_state = {}
|
||||||
|
final_state = {}
|
||||||
|
|
||||||
|
###################################
|
||||||
|
# Create object and do rock'n'roll:
|
||||||
|
subscription = PgSubscription(module, cursor, name, db, relinfo)
|
||||||
|
|
||||||
|
if subscription.exists:
|
||||||
|
initial_state = deepcopy(subscription.attrs)
|
||||||
|
final_state = deepcopy(initial_state)
|
||||||
|
|
||||||
|
# If module.check_mode=True, nothing will be changed:
|
||||||
|
if state == 'stat':
|
||||||
|
# Information has been collected already, so nothing is needed:
|
||||||
|
pass
|
||||||
|
|
||||||
|
if state == 'present':
|
||||||
|
if not subscription.exists:
|
||||||
|
if subsparams:
|
||||||
|
subsparams = convert_subscr_params(subsparams)
|
||||||
|
|
||||||
|
if connparams:
|
||||||
|
connparams = convert_conn_params(connparams)
|
||||||
|
|
||||||
|
changed = subscription.create(connparams,
|
||||||
|
publications,
|
||||||
|
subsparams,
|
||||||
|
check_mode=module.check_mode)
|
||||||
|
|
||||||
|
else:
|
||||||
|
changed = subscription.update(connparams,
|
||||||
|
publications,
|
||||||
|
subsparams,
|
||||||
|
check_mode=module.check_mode)
|
||||||
|
|
||||||
|
if owner and subscription.attrs['owner'] != owner:
|
||||||
|
changed = subscription.set_owner(owner, check_mode=module.check_mode)
|
||||||
|
|
||||||
|
elif state == 'absent':
|
||||||
|
changed = subscription.drop(cascade, check_mode=module.check_mode)
|
||||||
|
|
||||||
|
elif state == 'refresh':
|
||||||
|
if not subscription.exists:
|
||||||
|
module.fail_json(msg="Refresh failed: subscription '%s' does not exist" % name)
|
||||||
|
|
||||||
|
# Always returns True:
|
||||||
|
changed = subscription.refresh(check_mode=module.check_mode)
|
||||||
|
|
||||||
|
# Get final subscription info if needed:
|
||||||
|
final_state = subscription.get_info()
|
||||||
|
|
||||||
|
# Connection is not needed any more:
|
||||||
|
cursor.close()
|
||||||
|
db_connection.close()
|
||||||
|
|
||||||
|
# Return ret values and exit:
|
||||||
|
module.exit_json(changed=changed,
|
||||||
|
name=name,
|
||||||
|
exists=subscription.exists,
|
||||||
|
queries=subscription.executed_queries,
|
||||||
|
initial_state=initial_state,
|
||||||
|
final_state=final_state)
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
main()
|
8
test/integration/targets/postgresql_subscription/aliases
Normal file
8
test/integration/targets/postgresql_subscription/aliases
Normal file
|
@ -0,0 +1,8 @@
|
||||||
|
destructive
|
||||||
|
shippable/posix/group1
|
||||||
|
skip/osx
|
||||||
|
skip/centos
|
||||||
|
skip/freebsd
|
||||||
|
skip/rhel
|
||||||
|
skip/opensuse
|
||||||
|
skip/fedora
|
|
@ -0,0 +1,15 @@
|
||||||
|
pg_user: postgres
|
||||||
|
db_default: postgres
|
||||||
|
master_port: 5433
|
||||||
|
replica_port: 5434
|
||||||
|
|
||||||
|
test_table1: acme1
|
||||||
|
test_pub: first_publication
|
||||||
|
test_pub2: second_publication
|
||||||
|
replication_role: logical_replication
|
||||||
|
replication_pass: alsdjfKJKDf1#
|
||||||
|
test_db: acme_db
|
||||||
|
test_subscription: test
|
||||||
|
test_role1: alice
|
||||||
|
test_role2: bob
|
||||||
|
conn_timeout: 100
|
|
@ -0,0 +1,2 @@
|
||||||
|
dependencies:
|
||||||
|
- setup_postgresql_replication
|
|
@ -0,0 +1,7 @@
|
||||||
|
# Initial tests of postgresql_subscription module:
|
||||||
|
|
||||||
|
- import_tasks: setup_publication.yml
|
||||||
|
when: ansible_distribution == 'Ubuntu' and ansible_distribution_major_version >= '18'
|
||||||
|
|
||||||
|
- import_tasks: postgresql_subscription_initial.yml
|
||||||
|
when: ansible_distribution == 'Ubuntu' and ansible_distribution_major_version >= '18'
|
|
@ -0,0 +1,697 @@
|
||||||
|
# Copyright: (c) 2019, Andrew Klychkov (@Andersson007) <aaklychkov@mail.ru>
|
||||||
|
# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
|
||||||
|
|
||||||
|
- vars:
|
||||||
|
task_parameters: &task_parameters
|
||||||
|
become_user: '{{ pg_user }}'
|
||||||
|
become: yes
|
||||||
|
register: result
|
||||||
|
pg_parameters: &pg_parameters
|
||||||
|
login_user: '{{ pg_user }}'
|
||||||
|
login_db: '{{ test_db }}'
|
||||||
|
|
||||||
|
block:
|
||||||
|
|
||||||
|
- name: Create roles to test owner parameter
|
||||||
|
<<: *task_parameters
|
||||||
|
postgresql_user:
|
||||||
|
<<: *pg_parameters
|
||||||
|
login_port: '{{ replica_port }}'
|
||||||
|
name: '{{ item }}'
|
||||||
|
role_attr_flags: SUPERUSER,LOGIN
|
||||||
|
loop:
|
||||||
|
- '{{ test_role1 }}'
|
||||||
|
- '{{ test_role2 }}'
|
||||||
|
|
||||||
|
####################
|
||||||
|
# Test mode: present
|
||||||
|
####################
|
||||||
|
- name: Create subscription
|
||||||
|
<<: *task_parameters
|
||||||
|
postgresql_subscription:
|
||||||
|
<<: *pg_parameters
|
||||||
|
login_port: '{{ replica_port }}'
|
||||||
|
name: '{{ test_subscription }}'
|
||||||
|
state: present
|
||||||
|
publications: '{{ test_pub }}'
|
||||||
|
connparams:
|
||||||
|
host: 127.0.0.1
|
||||||
|
port: '{{ master_port }}'
|
||||||
|
user: '{{ replication_role }}'
|
||||||
|
password: '{{ replication_pass }}'
|
||||||
|
dbname: '{{ test_db }}'
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- result is changed
|
||||||
|
- result.name == '{{ test_subscription }}'
|
||||||
|
- result.queries == ["CREATE SUBSCRIPTION test CONNECTION 'host=127.0.0.1 port={{ master_port }} user={{ replication_role }} password={{ replication_pass }} dbname={{ test_db }}' PUBLICATION {{ test_pub }}"]
|
||||||
|
- result.exists == true
|
||||||
|
- result.initial_state == {}
|
||||||
|
- result.final_state.owner == '{{ pg_user }}'
|
||||||
|
- result.final_state.enabled == true
|
||||||
|
- result.final_state.publications == ["{{ test_pub }}"]
|
||||||
|
- result.final_state.synccommit == true
|
||||||
|
- result.final_state.slotname == '{{ test_subscription }}'
|
||||||
|
- result.final_state.conninfo.dbname == '{{ test_db }}'
|
||||||
|
- result.final_state.conninfo.host == '127.0.0.1'
|
||||||
|
- result.final_state.conninfo.port == {{ master_port }}
|
||||||
|
- result.final_state.conninfo.user == '{{ replication_role }}'
|
||||||
|
- result.final_state.conninfo.password == '{{ replication_pass }}'
|
||||||
|
|
||||||
|
#################
|
||||||
|
# Test mode: stat
|
||||||
|
#################
|
||||||
|
|
||||||
|
- name: Stat mode in check mode
|
||||||
|
<<: *task_parameters
|
||||||
|
postgresql_subscription:
|
||||||
|
<<: *pg_parameters
|
||||||
|
login_port: '{{ replica_port }}'
|
||||||
|
name: '{{ test_subscription }}'
|
||||||
|
state: stat
|
||||||
|
check_mode: yes
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- result is not changed
|
||||||
|
- result.name == '{{ test_subscription }}'
|
||||||
|
- result.exists == true
|
||||||
|
- result.final_state == result.initial_state
|
||||||
|
- result.final_state.owner == '{{ pg_user }}'
|
||||||
|
- result.final_state.enabled == true
|
||||||
|
- result.final_state.publications == ["{{ test_pub }}"]
|
||||||
|
- result.final_state.synccommit == true
|
||||||
|
- result.final_state.slotname == '{{ test_subscription }}'
|
||||||
|
- result.final_state.conninfo.dbname == '{{ test_db }}'
|
||||||
|
- result.final_state.conninfo.host == '127.0.0.1'
|
||||||
|
- result.final_state.conninfo.port == {{ master_port }}
|
||||||
|
- result.final_state.conninfo.user == '{{ replication_role }}'
|
||||||
|
- result.final_state.conninfo.password == '{{ replication_pass }}'
|
||||||
|
|
||||||
|
- name: Stat mode
|
||||||
|
<<: *task_parameters
|
||||||
|
postgresql_subscription:
|
||||||
|
<<: *pg_parameters
|
||||||
|
login_port: '{{ replica_port }}'
|
||||||
|
name: '{{ test_subscription }}'
|
||||||
|
state: stat
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- result is not changed
|
||||||
|
- result.name == '{{ test_subscription }}'
|
||||||
|
- result.exists == true
|
||||||
|
- result.final_state == result.initial_state
|
||||||
|
- result.final_state.owner == '{{ pg_user }}'
|
||||||
|
- result.final_state.enabled == true
|
||||||
|
- result.final_state.publications == ["{{ test_pub }}"]
|
||||||
|
- result.final_state.synccommit == true
|
||||||
|
- result.final_state.slotname == '{{ test_subscription }}'
|
||||||
|
- result.final_state.conninfo.dbname == '{{ test_db }}'
|
||||||
|
- result.final_state.conninfo.host == '127.0.0.1'
|
||||||
|
- result.final_state.conninfo.port == {{ master_port }}
|
||||||
|
- result.final_state.conninfo.user == '{{ replication_role }}'
|
||||||
|
- result.final_state.conninfo.password == '{{ replication_pass }}'
|
||||||
|
|
||||||
|
###################
|
||||||
|
# Test mode: absent
|
||||||
|
###################
|
||||||
|
|
||||||
|
- name: Drop subscription in check mode
|
||||||
|
<<: *task_parameters
|
||||||
|
postgresql_subscription:
|
||||||
|
<<: *pg_parameters
|
||||||
|
login_port: '{{ replica_port }}'
|
||||||
|
name: '{{ test_subscription }}'
|
||||||
|
state: absent
|
||||||
|
check_mode: yes
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- result is changed
|
||||||
|
- result.queries == ["DROP SUBSCRIPTION {{ test_subscription }}"]
|
||||||
|
- result.final_state == result.initial_state
|
||||||
|
|
||||||
|
- name: Check the subscription exists
|
||||||
|
<<: *task_parameters
|
||||||
|
postgresql_subscription:
|
||||||
|
<<: *pg_parameters
|
||||||
|
login_port: '{{ replica_port }}'
|
||||||
|
name: '{{ test_subscription }}'
|
||||||
|
state: stat
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- result is not changed
|
||||||
|
- result.name == '{{ test_subscription }}'
|
||||||
|
- result.exists == true
|
||||||
|
|
||||||
|
- name: Drop subscription
|
||||||
|
<<: *task_parameters
|
||||||
|
postgresql_subscription:
|
||||||
|
<<: *pg_parameters
|
||||||
|
login_port: '{{ replica_port }}'
|
||||||
|
name: '{{ test_subscription }}'
|
||||||
|
state: absent
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- result is changed
|
||||||
|
- result.queries == ["DROP SUBSCRIPTION {{ test_subscription }}"]
|
||||||
|
- result.final_state != result.initial_state
|
||||||
|
|
||||||
|
- name: Check the subscription doesn't exist
|
||||||
|
<<: *task_parameters
|
||||||
|
postgresql_subscription:
|
||||||
|
<<: *pg_parameters
|
||||||
|
login_port: '{{ replica_port }}'
|
||||||
|
name: '{{ test_subscription }}'
|
||||||
|
state: stat
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- result is not changed
|
||||||
|
- result.name == '{{ test_subscription }}'
|
||||||
|
- result.exists == false
|
||||||
|
|
||||||
|
##################
|
||||||
|
# Test owner param
|
||||||
|
##################
|
||||||
|
|
||||||
|
- name: Create with owner
|
||||||
|
<<: *task_parameters
|
||||||
|
postgresql_subscription:
|
||||||
|
<<: *pg_parameters
|
||||||
|
login_port: '{{ replica_port }}'
|
||||||
|
name: '{{ test_subscription }}'
|
||||||
|
state: present
|
||||||
|
publications: '{{ test_pub }}'
|
||||||
|
owner: '{{ test_role1 }}'
|
||||||
|
connparams:
|
||||||
|
host: 127.0.0.1
|
||||||
|
port: '{{ master_port }}'
|
||||||
|
user: '{{ replication_role }}'
|
||||||
|
password: '{{ replication_pass }}'
|
||||||
|
dbname: '{{ test_db }}'
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- result.final_state.owner == '{{ test_role1 }}'
|
||||||
|
- result.queries[1] == 'ALTER SUBSCRIPTION {{ test_subscription }} OWNER TO "{{ test_role1 }}"'
|
||||||
|
|
||||||
|
- name: Try to set this owner again
|
||||||
|
<<: *task_parameters
|
||||||
|
postgresql_subscription:
|
||||||
|
<<: *pg_parameters
|
||||||
|
login_port: '{{ replica_port }}'
|
||||||
|
name: '{{ test_subscription }}'
|
||||||
|
state: present
|
||||||
|
publications: '{{ test_pub }}'
|
||||||
|
owner: '{{ test_role1 }}'
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- result is not changed
|
||||||
|
- result.initial_state == result.final_state
|
||||||
|
- result.final_state.owner == '{{ test_role1 }}'
|
||||||
|
|
||||||
|
- name: Check owner
|
||||||
|
<<: *task_parameters
|
||||||
|
postgresql_subscription:
|
||||||
|
<<: *pg_parameters
|
||||||
|
login_port: '{{ replica_port }}'
|
||||||
|
name: '{{ test_subscription }}'
|
||||||
|
state: stat
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- result is not changed
|
||||||
|
- result.name == '{{ test_subscription }}'
|
||||||
|
- result.exists == true
|
||||||
|
- result.initial_state.owner == '{{ test_role1 }}'
|
||||||
|
|
||||||
|
- name: Set another owner in check mode
|
||||||
|
<<: *task_parameters
|
||||||
|
postgresql_subscription:
|
||||||
|
<<: *pg_parameters
|
||||||
|
login_port: '{{ replica_port }}'
|
||||||
|
name: '{{ test_subscription }}'
|
||||||
|
state: present
|
||||||
|
publications: '{{ test_pub }}'
|
||||||
|
owner: '{{ test_role2 }}'
|
||||||
|
check_mode: yes
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- result is changed
|
||||||
|
- result.initial_state == result.final_state
|
||||||
|
- result.final_state.owner == '{{ test_role1 }}'
|
||||||
|
- result.queries == ['ALTER SUBSCRIPTION {{ test_subscription }} OWNER TO "{{ test_role2 }}"']
|
||||||
|
|
||||||
|
- name: Check owner
|
||||||
|
<<: *task_parameters
|
||||||
|
postgresql_subscription:
|
||||||
|
<<: *pg_parameters
|
||||||
|
login_port: '{{ replica_port }}'
|
||||||
|
name: '{{ test_subscription }}'
|
||||||
|
state: stat
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- result is not changed
|
||||||
|
- result.name == '{{ test_subscription }}'
|
||||||
|
- result.exists == true
|
||||||
|
- result.initial_state.owner == '{{ test_role1 }}'
|
||||||
|
|
||||||
|
- name: Set another owner
|
||||||
|
<<: *task_parameters
|
||||||
|
postgresql_subscription:
|
||||||
|
<<: *pg_parameters
|
||||||
|
login_port: '{{ replica_port }}'
|
||||||
|
name: '{{ test_subscription }}'
|
||||||
|
state: present
|
||||||
|
publications: '{{ test_pub }}'
|
||||||
|
owner: '{{ test_role2 }}'
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- result is changed
|
||||||
|
- result.initial_state != result.final_state
|
||||||
|
- result.final_state.owner == '{{ test_role2 }}'
|
||||||
|
- result.queries == ['ALTER SUBSCRIPTION {{ test_subscription }} OWNER TO "{{ test_role2 }}"']
|
||||||
|
|
||||||
|
- name: Check owner
|
||||||
|
<<: *task_parameters
|
||||||
|
postgresql_subscription:
|
||||||
|
<<: *pg_parameters
|
||||||
|
login_port: '{{ replica_port }}'
|
||||||
|
name: '{{ test_subscription }}'
|
||||||
|
state: stat
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- result is not changed
|
||||||
|
- result.name == '{{ test_subscription }}'
|
||||||
|
- result.exists == true
|
||||||
|
- result.initial_state.owner == '{{ test_role2 }}'
|
||||||
|
|
||||||
|
##############################
|
||||||
|
# Test cascade and owner param
|
||||||
|
##############################
|
||||||
|
|
||||||
|
- name: Drop subscription cascade in check mode
|
||||||
|
<<: *task_parameters
|
||||||
|
postgresql_subscription:
|
||||||
|
<<: *pg_parameters
|
||||||
|
login_port: '{{ replica_port }}'
|
||||||
|
name: '{{ test_subscription }}'
|
||||||
|
state: absent
|
||||||
|
cascade: yes
|
||||||
|
check_mode: yes
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- result is changed
|
||||||
|
- result.queries == ["DROP SUBSCRIPTION {{ test_subscription }} CASCADE"]
|
||||||
|
- result.final_state == result.initial_state
|
||||||
|
|
||||||
|
- name: Check the subscription exists
|
||||||
|
<<: *task_parameters
|
||||||
|
postgresql_subscription:
|
||||||
|
<<: *pg_parameters
|
||||||
|
login_port: '{{ replica_port }}'
|
||||||
|
name: '{{ test_subscription }}'
|
||||||
|
state: stat
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- result is not changed
|
||||||
|
- result.name == '{{ test_subscription }}'
|
||||||
|
- result.exists == true
|
||||||
|
|
||||||
|
- name: Drop subscription cascade
|
||||||
|
<<: *task_parameters
|
||||||
|
postgresql_subscription:
|
||||||
|
<<: *pg_parameters
|
||||||
|
login_port: '{{ replica_port }}'
|
||||||
|
name: '{{ test_subscription }}'
|
||||||
|
state: absent
|
||||||
|
cascade: yes
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- result is changed
|
||||||
|
- result.queries == ["DROP SUBSCRIPTION {{ test_subscription }} CASCADE"]
|
||||||
|
- result.final_state != result.initial_state
|
||||||
|
|
||||||
|
- name: Check the subscription doesn't exist
|
||||||
|
<<: *task_parameters
|
||||||
|
postgresql_subscription:
|
||||||
|
<<: *pg_parameters
|
||||||
|
login_port: '{{ replica_port }}'
|
||||||
|
name: '{{ test_subscription }}'
|
||||||
|
state: stat
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- result is not changed
|
||||||
|
- result.name == '{{ test_subscription }}'
|
||||||
|
- result.exists == false
|
||||||
|
|
||||||
|
###########################
|
||||||
|
# Test subsparams parameter
|
||||||
|
###########################
|
||||||
|
|
||||||
|
- name: Create subscription with subsparams
|
||||||
|
<<: *task_parameters
|
||||||
|
postgresql_subscription:
|
||||||
|
<<: *pg_parameters
|
||||||
|
login_port: '{{ replica_port }}'
|
||||||
|
name: '{{ test_subscription }}'
|
||||||
|
state: present
|
||||||
|
publications: '{{ test_pub }}'
|
||||||
|
connparams:
|
||||||
|
host: 127.0.0.1
|
||||||
|
port: '{{ master_port }}'
|
||||||
|
user: '{{ replication_role }}'
|
||||||
|
password: '{{ replication_pass }}'
|
||||||
|
dbname: '{{ test_db }}'
|
||||||
|
subsparams:
|
||||||
|
enabled: no
|
||||||
|
synchronous_commit: no
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- result is changed
|
||||||
|
- result.name == '{{ test_subscription }}'
|
||||||
|
- result.queries == ["CREATE SUBSCRIPTION test CONNECTION 'host=127.0.0.1 port={{ master_port }} user={{ replication_role }} password={{ replication_pass }} dbname={{ test_db }}' PUBLICATION {{ test_pub }} WITH (enabled = false, synchronous_commit = false)"]
|
||||||
|
- result.exists == true
|
||||||
|
- result.final_state.enabled == false
|
||||||
|
- result.final_state.synccommit == false
|
||||||
|
|
||||||
|
- name: Stat mode
|
||||||
|
<<: *task_parameters
|
||||||
|
postgresql_subscription:
|
||||||
|
<<: *pg_parameters
|
||||||
|
login_port: '{{ replica_port }}'
|
||||||
|
name: '{{ test_subscription }}'
|
||||||
|
state: stat
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- result.name == '{{ test_subscription }}'
|
||||||
|
- result.final_state.enabled == false
|
||||||
|
- result.final_state.synccommit == false
|
||||||
|
|
||||||
|
- name: Enable changed params
|
||||||
|
<<: *task_parameters
|
||||||
|
postgresql_subscription:
|
||||||
|
<<: *pg_parameters
|
||||||
|
login_port: '{{ replica_port }}'
|
||||||
|
name: '{{ test_subscription }}'
|
||||||
|
publications: '{{ test_pub }}'
|
||||||
|
subsparams:
|
||||||
|
enabled: yes
|
||||||
|
synchronous_commit: yes
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- result is changed
|
||||||
|
- result.name == '{{ test_subscription }}'
|
||||||
|
- result.queries == ["ALTER SUBSCRIPTION {{ test_subscription }} ENABLE", "ALTER SUBSCRIPTION {{ test_subscription }} SET (synchronous_commit = true)"]
|
||||||
|
- result.exists == true
|
||||||
|
- result.final_state.enabled == true
|
||||||
|
- result.final_state.synccommit == true
|
||||||
|
|
||||||
|
- name: Stat mode
|
||||||
|
<<: *task_parameters
|
||||||
|
postgresql_subscription:
|
||||||
|
<<: *pg_parameters
|
||||||
|
login_port: '{{ replica_port }}'
|
||||||
|
name: '{{ test_subscription }}'
|
||||||
|
state: stat
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- result.name == '{{ test_subscription }}'
|
||||||
|
- result.final_state.enabled == true
|
||||||
|
- result.final_state.synccommit == true
|
||||||
|
|
||||||
|
- name: Enable the same params again
|
||||||
|
<<: *task_parameters
|
||||||
|
postgresql_subscription:
|
||||||
|
<<: *pg_parameters
|
||||||
|
login_port: '{{ replica_port }}'
|
||||||
|
name: '{{ test_subscription }}'
|
||||||
|
publications: '{{ test_pub }}'
|
||||||
|
subsparams:
|
||||||
|
enabled: yes
|
||||||
|
synchronous_commit: yes
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- result is not changed
|
||||||
|
- result.name == '{{ test_subscription }}'
|
||||||
|
- result.queries == []
|
||||||
|
- result.exists == true
|
||||||
|
- result.final_state == result.initial_state
|
||||||
|
- result.final_state.enabled == true
|
||||||
|
- result.final_state.synccommit == true
|
||||||
|
|
||||||
|
##########################
|
||||||
|
# Test change publications
|
||||||
|
##########################
|
||||||
|
|
||||||
|
- name: Change publications in check mode
|
||||||
|
<<: *task_parameters
|
||||||
|
postgresql_subscription:
|
||||||
|
<<: *pg_parameters
|
||||||
|
login_port: '{{ replica_port }}'
|
||||||
|
name: '{{ test_subscription }}'
|
||||||
|
state: present
|
||||||
|
publications:
|
||||||
|
- '{{ test_pub }}'
|
||||||
|
- '{{ test_pub2 }}'
|
||||||
|
check_mode: yes
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- result is changed
|
||||||
|
- result.name == '{{ test_subscription }}'
|
||||||
|
- result.final_state.publications == result.initial_state.publications
|
||||||
|
- result.final_state.publications == ['{{ test_pub }}']
|
||||||
|
- result.queries == ['ALTER SUBSCRIPTION {{ test_subscription }} SET PUBLICATION {{ test_pub }}, {{ test_pub2 }}']
|
||||||
|
|
||||||
|
- name: Check publications
|
||||||
|
<<: *task_parameters
|
||||||
|
postgresql_subscription:
|
||||||
|
<<: *pg_parameters
|
||||||
|
login_port: '{{ replica_port }}'
|
||||||
|
name: '{{ test_subscription }}'
|
||||||
|
state: stat
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- result.name == '{{ test_subscription }}'
|
||||||
|
- result.final_state.publications == ['{{ test_pub }}']
|
||||||
|
|
||||||
|
- name: Change publications
|
||||||
|
<<: *task_parameters
|
||||||
|
postgresql_subscription:
|
||||||
|
<<: *pg_parameters
|
||||||
|
login_port: '{{ replica_port }}'
|
||||||
|
name: '{{ test_subscription }}'
|
||||||
|
state: present
|
||||||
|
publications:
|
||||||
|
- '{{ test_pub }}'
|
||||||
|
- '{{ test_pub2 }}'
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- result is changed
|
||||||
|
- result.name == '{{ test_subscription }}'
|
||||||
|
- result.final_state.publications != result.initial_state.publications
|
||||||
|
- result.final_state.publications == ['{{ test_pub }}', '{{ test_pub2 }}']
|
||||||
|
- result.queries == ['ALTER SUBSCRIPTION {{ test_subscription }} SET PUBLICATION {{ test_pub }}, {{ test_pub2 }}']
|
||||||
|
|
||||||
|
- name: Check publications
|
||||||
|
<<: *task_parameters
|
||||||
|
postgresql_subscription:
|
||||||
|
<<: *pg_parameters
|
||||||
|
login_port: '{{ replica_port }}'
|
||||||
|
name: '{{ test_subscription }}'
|
||||||
|
state: stat
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- result.name == '{{ test_subscription }}'
|
||||||
|
- result.final_state.publications == ['{{ test_pub }}', '{{ test_pub2 }}']
|
||||||
|
|
||||||
|
- name: Change publications with the same values again
|
||||||
|
<<: *task_parameters
|
||||||
|
postgresql_subscription:
|
||||||
|
<<: *pg_parameters
|
||||||
|
login_port: '{{ replica_port }}'
|
||||||
|
name: '{{ test_subscription }}'
|
||||||
|
state: present
|
||||||
|
publications:
|
||||||
|
- '{{ test_pub }}'
|
||||||
|
- '{{ test_pub2 }}'
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- result is not changed
|
||||||
|
- result.name == '{{ test_subscription }}'
|
||||||
|
- result.final_state.publications == result.initial_state.publications
|
||||||
|
- result.final_state.publications == ['{{ test_pub }}', '{{ test_pub2 }}']
|
||||||
|
- result.queries == []
|
||||||
|
|
||||||
|
######################
|
||||||
|
# Test update conninfo
|
||||||
|
######################
|
||||||
|
|
||||||
|
- name: Change conninfo in check mode
|
||||||
|
<<: *task_parameters
|
||||||
|
postgresql_subscription:
|
||||||
|
<<: *pg_parameters
|
||||||
|
login_port: '{{ replica_port }}'
|
||||||
|
name: '{{ test_subscription }}'
|
||||||
|
state: present
|
||||||
|
connparams:
|
||||||
|
host: 127.0.0.1
|
||||||
|
port: '{{ master_port }}'
|
||||||
|
user: '{{ replication_role }}'
|
||||||
|
password: '{{ replication_pass }}'
|
||||||
|
dbname: '{{ test_db }}'
|
||||||
|
connect_timeout: '{{ conn_timeout }}'
|
||||||
|
check_mode: yes
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- result is changed
|
||||||
|
- result.name == '{{ test_subscription }}'
|
||||||
|
- result.queries == ["ALTER SUBSCRIPTION {{ test_subscription }} CONNECTION 'host=127.0.0.1 port={{ master_port }} user={{ replication_role }} password={{ replication_pass }} dbname={{ test_db }} connect_timeout={{ conn_timeout }}'"]
|
||||||
|
- result.initial_state.conninfo == result.final_state.conninfo
|
||||||
|
|
||||||
|
- name: Change conninfo
|
||||||
|
<<: *task_parameters
|
||||||
|
postgresql_subscription:
|
||||||
|
<<: *pg_parameters
|
||||||
|
login_port: '{{ replica_port }}'
|
||||||
|
name: '{{ test_subscription }}'
|
||||||
|
state: present
|
||||||
|
relinfo: yes
|
||||||
|
connparams:
|
||||||
|
host: 127.0.0.1
|
||||||
|
port: '{{ master_port }}'
|
||||||
|
user: '{{ replication_role }}'
|
||||||
|
password: '{{ replication_pass }}'
|
||||||
|
dbname: '{{ test_db }}'
|
||||||
|
connect_timeout: '{{ conn_timeout }}'
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- result is changed
|
||||||
|
- result.name == '{{ test_subscription }}'
|
||||||
|
- result.queries == ["ALTER SUBSCRIPTION {{ test_subscription }} CONNECTION 'host=127.0.0.1 port={{ master_port }} user={{ replication_role }} password={{ replication_pass }} dbname={{ test_db }} connect_timeout={{ conn_timeout }}'"]
|
||||||
|
- result.initial_state.conninfo != result.final_state.conninfo
|
||||||
|
|
||||||
|
- name: Check publications
|
||||||
|
<<: *task_parameters
|
||||||
|
postgresql_subscription:
|
||||||
|
<<: *pg_parameters
|
||||||
|
login_port: '{{ replica_port }}'
|
||||||
|
name: '{{ test_subscription }}'
|
||||||
|
state: stat
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- result.name == '{{ test_subscription }}'
|
||||||
|
- result.final_state.conninfo.connect_timeout == {{ conn_timeout }}
|
||||||
|
|
||||||
|
- name: Try to change conninfo again with the same values
|
||||||
|
<<: *task_parameters
|
||||||
|
postgresql_subscription:
|
||||||
|
<<: *pg_parameters
|
||||||
|
login_port: '{{ replica_port }}'
|
||||||
|
name: '{{ test_subscription }}'
|
||||||
|
state: present
|
||||||
|
connparams:
|
||||||
|
host: 127.0.0.1
|
||||||
|
port: '{{ master_port }}'
|
||||||
|
user: '{{ replication_role }}'
|
||||||
|
password: '{{ replication_pass }}'
|
||||||
|
dbname: '{{ test_db }}'
|
||||||
|
connect_timeout: '{{ conn_timeout }}'
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- result is not changed
|
||||||
|
- result.name == '{{ test_subscription }}'
|
||||||
|
- result.queries == []
|
||||||
|
- result.initial_state.conninfo == result.final_state.conninfo
|
||||||
|
- result.final_state.conninfo.connect_timeout == {{ conn_timeout }}
|
||||||
|
|
||||||
|
####################
|
||||||
|
# Test state refresh
|
||||||
|
####################
|
||||||
|
|
||||||
|
- name: Refresh in check mode
|
||||||
|
<<: *task_parameters
|
||||||
|
postgresql_subscription:
|
||||||
|
<<: *pg_parameters
|
||||||
|
login_port: '{{ replica_port }}'
|
||||||
|
name: '{{ test_subscription }}'
|
||||||
|
state: refresh
|
||||||
|
check_mode: yes
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- result is changed
|
||||||
|
- result.name == '{{ test_subscription }}'
|
||||||
|
- result.queries == ["ALTER SUBSCRIPTION {{ test_subscription }} REFRESH PUBLICATION"]
|
||||||
|
|
||||||
|
- name: Refresh
|
||||||
|
<<: *task_parameters
|
||||||
|
postgresql_subscription:
|
||||||
|
<<: *pg_parameters
|
||||||
|
login_port: '{{ replica_port }}'
|
||||||
|
name: '{{ test_subscription }}'
|
||||||
|
state: refresh
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- result is changed
|
||||||
|
- result.name == '{{ test_subscription }}'
|
||||||
|
- result.queries == ["ALTER SUBSCRIPTION {{ test_subscription }} REFRESH PUBLICATION"]
|
||||||
|
|
||||||
|
####################
|
||||||
|
# Test relinfo param
|
||||||
|
####################
|
||||||
|
|
||||||
|
- name: Get relinfo
|
||||||
|
<<: *task_parameters
|
||||||
|
postgresql_subscription:
|
||||||
|
<<: *pg_parameters
|
||||||
|
login_port: '{{ replica_port }}'
|
||||||
|
name: '{{ test_subscription }}'
|
||||||
|
state: stat
|
||||||
|
relinfo: yes
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- result.name == '{{ test_subscription }}'
|
||||||
|
- result.final_state.relinfo[0].relname == '{{ test_table1 }}'
|
||||||
|
- result.final_state == result.initial_state
|
||||||
|
|
||||||
|
##########
|
||||||
|
# Clean up
|
||||||
|
##########
|
||||||
|
- name: Drop subscription
|
||||||
|
<<: *task_parameters
|
||||||
|
postgresql_subscription:
|
||||||
|
<<: *pg_parameters
|
||||||
|
login_port: '{{ replica_port }}'
|
||||||
|
name: '{{ test_subscription }}'
|
||||||
|
state: absent
|
|
@ -0,0 +1,84 @@
|
||||||
|
# Copyright: (c) 2019, Andrew Klychkov (@Andersson007) <aaklychkov@mail.ru>
|
||||||
|
# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
|
||||||
|
# Preparation for further tests of postgresql_subscription module.
|
||||||
|
|
||||||
|
- vars:
|
||||||
|
task_parameters: &task_parameters
|
||||||
|
become_user: '{{ pg_user }}'
|
||||||
|
become: yes
|
||||||
|
register: result
|
||||||
|
pg_parameters: &pg_parameters
|
||||||
|
login_user: '{{ pg_user }}'
|
||||||
|
login_db: '{{ test_db }}'
|
||||||
|
|
||||||
|
block:
|
||||||
|
- name: postgresql_publication - create test db
|
||||||
|
<<: *task_parameters
|
||||||
|
postgresql_db:
|
||||||
|
login_user: '{{ pg_user }}'
|
||||||
|
login_port: '{{ master_port }}'
|
||||||
|
maintenance_db: '{{ db_default }}'
|
||||||
|
name: '{{ test_db }}'
|
||||||
|
|
||||||
|
- name: postgresql_publication - create test role
|
||||||
|
<<: *task_parameters
|
||||||
|
postgresql_user:
|
||||||
|
<<: *pg_parameters
|
||||||
|
login_port: '{{ master_port }}'
|
||||||
|
name: '{{ replication_role }}'
|
||||||
|
password: '{{ replication_pass }}'
|
||||||
|
role_attr_flags: LOGIN,REPLICATION
|
||||||
|
|
||||||
|
- name: postgresql_publication - create test table
|
||||||
|
<<: *task_parameters
|
||||||
|
postgresql_table:
|
||||||
|
<<: *pg_parameters
|
||||||
|
login_port: '{{ master_port }}'
|
||||||
|
name: '{{ test_table1 }}'
|
||||||
|
columns:
|
||||||
|
- id int
|
||||||
|
|
||||||
|
- name: Master - dump schema
|
||||||
|
<<: *task_parameters
|
||||||
|
shell: pg_dumpall -p '{{ master_port }}' -s > /tmp/schema.sql
|
||||||
|
|
||||||
|
- name: Replicat restore schema
|
||||||
|
<<: *task_parameters
|
||||||
|
shell: psql -p '{{ replica_port }}' -f /tmp/schema.sql
|
||||||
|
|
||||||
|
- name: postgresql_publication - create publication
|
||||||
|
<<: *task_parameters
|
||||||
|
postgresql_publication:
|
||||||
|
<<: *pg_parameters
|
||||||
|
login_port: '{{ master_port }}'
|
||||||
|
name: '{{ test_pub }}'
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- result is changed
|
||||||
|
- result.exists == true
|
||||||
|
- result.queries == ["CREATE PUBLICATION \"{{ test_pub }}\" FOR ALL TABLES"]
|
||||||
|
- result.owner == '{{ pg_user }}'
|
||||||
|
- result.alltables == true
|
||||||
|
- result.tables == []
|
||||||
|
- result.parameters.publish != {}
|
||||||
|
|
||||||
|
- name: postgresql_publication - create one more publication
|
||||||
|
<<: *task_parameters
|
||||||
|
postgresql_publication:
|
||||||
|
<<: *pg_parameters
|
||||||
|
login_port: '{{ master_port }}'
|
||||||
|
name: '{{ test_pub2 }}'
|
||||||
|
|
||||||
|
- name: postgresql_publication - check the publication was created
|
||||||
|
<<: *task_parameters
|
||||||
|
postgresql_query:
|
||||||
|
<<: *pg_parameters
|
||||||
|
login_port: '{{ master_port }}'
|
||||||
|
query: >
|
||||||
|
SELECT * FROM pg_publication WHERE pubname = '{{ test_pub }}'
|
||||||
|
AND pubowner = '10' AND puballtables = 't'
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- result.rowcount == 1
|
|
@ -0,0 +1,30 @@
|
||||||
|
# General:
|
||||||
|
pg_user: postgres
|
||||||
|
db_default: postgres
|
||||||
|
|
||||||
|
pg_package_list:
|
||||||
|
- apt-utils
|
||||||
|
- postgresql
|
||||||
|
- postgresql-contrib
|
||||||
|
- python3-psycopg2
|
||||||
|
|
||||||
|
packages_to_remove:
|
||||||
|
- postgresql
|
||||||
|
- postgresql-contrib
|
||||||
|
- postgresql-server
|
||||||
|
- postgresql-libs
|
||||||
|
- python3-psycopg2
|
||||||
|
|
||||||
|
# Master specific defaults:
|
||||||
|
master_root_dir: '/var/lib/pgsql/master'
|
||||||
|
master_data_dir: '{{ master_root_dir }}/data'
|
||||||
|
master_postgresql_conf: '{{ master_data_dir }}/postgresql.conf'
|
||||||
|
master_pg_hba_conf: '{{ master_data_dir }}/pg_hba.conf'
|
||||||
|
master_port: 5433
|
||||||
|
|
||||||
|
# Replica specific defaults:
|
||||||
|
replica_root_dir: '/var/lib/pgsql/replica'
|
||||||
|
replica_data_dir: '{{ replica_root_dir }}/data'
|
||||||
|
replica_postgresql_conf: '{{ replica_data_dir }}/postgresql.conf'
|
||||||
|
replica_pg_hba_conf: '{{ replica_data_dir }}/pg_hba.conf'
|
||||||
|
replica_port: 5434
|
|
@ -0,0 +1,23 @@
|
||||||
|
- name: Stop services
|
||||||
|
become: yes
|
||||||
|
become_user: '{{ pg_user }}'
|
||||||
|
shell: '{{ pg_ctl }} -D {{ item.datadir }} -o "-p {{ item.port }}" -m immediate stop'
|
||||||
|
loop:
|
||||||
|
- { datadir: '{{ master_data_dir }}', port: '{{ master_port }}' }
|
||||||
|
- { datadir: '{{ replica_data_dir }}', port: '{{ replica_port }}' }
|
||||||
|
listen: stop postgresql
|
||||||
|
|
||||||
|
- name: Remove packages
|
||||||
|
apt:
|
||||||
|
name: '{{ packages_to_remove }}'
|
||||||
|
state: absent
|
||||||
|
listen: cleanup postgresql
|
||||||
|
|
||||||
|
- name: Remove FS objects
|
||||||
|
file:
|
||||||
|
state: absent
|
||||||
|
path: "{{ item }}"
|
||||||
|
loop:
|
||||||
|
- "{{ master_root_dir }}"
|
||||||
|
- "{{ replica_root_dir }}"
|
||||||
|
listen: cleanup postgresql
|
|
@ -0,0 +1,8 @@
|
||||||
|
# Copyright: (c) 2019, Andrew Klychkov (@Andersson007) <aaklychkov@mail.ru>
|
||||||
|
# GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt)
|
||||||
|
|
||||||
|
# Setup PostgreSQL master-standby replication into one container:
|
||||||
|
- import_tasks: setup_postgresql_cluster.yml
|
||||||
|
when:
|
||||||
|
- ansible_distribution == 'Ubuntu'
|
||||||
|
- ansible_distribution_major_version >= '18'
|
|
@ -0,0 +1,93 @@
|
||||||
|
# We run two servers listening different ports
|
||||||
|
# to be able to check replication (one server for master, another for standby).
|
||||||
|
|
||||||
|
- name: Install packages
|
||||||
|
apt:
|
||||||
|
name: '{{ pg_package_list }}'
|
||||||
|
notify: cleanup postgresql
|
||||||
|
|
||||||
|
- name: Create root dirs
|
||||||
|
file:
|
||||||
|
state: directory
|
||||||
|
path: "{{ item }}"
|
||||||
|
owner: postgres
|
||||||
|
group: postgres
|
||||||
|
mode: 0700
|
||||||
|
loop:
|
||||||
|
- "{{ master_root_dir }}"
|
||||||
|
- "{{ master_data_dir }}"
|
||||||
|
- "{{ replica_root_dir }}"
|
||||||
|
- "{{ replica_data_dir }}"
|
||||||
|
notify: cleanup postgresql
|
||||||
|
|
||||||
|
- name: Find initdb
|
||||||
|
shell: find /usr/lib -type f -name "initdb"
|
||||||
|
register: result
|
||||||
|
|
||||||
|
- name: Set path to initdb
|
||||||
|
set_fact:
|
||||||
|
initdb: '{{ result.stdout }}'
|
||||||
|
|
||||||
|
- name: Initialize databases
|
||||||
|
become: yes
|
||||||
|
become_user: '{{ pg_user }}'
|
||||||
|
shell: '{{ initdb }} --pgdata {{ item }}'
|
||||||
|
loop:
|
||||||
|
- "{{ master_data_dir }}"
|
||||||
|
- "{{ replica_data_dir }}"
|
||||||
|
|
||||||
|
- name: Copy config templates
|
||||||
|
template:
|
||||||
|
src: '{{ item.conf_templ }}'
|
||||||
|
dest: '{{ item.conf_dest }}'
|
||||||
|
owner: postgres
|
||||||
|
group: postgres
|
||||||
|
force: yes
|
||||||
|
loop:
|
||||||
|
- { conf_templ: master_postgresql.conf.j2, conf_dest: '{{ master_postgresql_conf }}' }
|
||||||
|
- { conf_templ: replica_postgresql.conf.j2, conf_dest: '{{ replica_postgresql_conf }}' }
|
||||||
|
- { conf_templ: pg_hba.conf.j2, conf_dest: '{{ master_pg_hba_conf }}' }
|
||||||
|
- { conf_templ: pg_hba.conf.j2, conf_dest: '{{ replica_pg_hba_conf }}' }
|
||||||
|
|
||||||
|
- name: Find pg_ctl
|
||||||
|
shell: find /usr/lib -type f -name "pg_ctl"
|
||||||
|
register: result
|
||||||
|
|
||||||
|
- name: Set path to initdb
|
||||||
|
set_fact:
|
||||||
|
pg_ctl: '{{ result.stdout }}'
|
||||||
|
|
||||||
|
- name: Start servers
|
||||||
|
become: yes
|
||||||
|
become_user: '{{ pg_user }}'
|
||||||
|
shell: '{{ pg_ctl }} -D {{ item.datadir }} -o "-p {{ item.port }}" start'
|
||||||
|
loop:
|
||||||
|
- { datadir: '{{ master_data_dir }}', port: '{{ master_port }}' }
|
||||||
|
- { datadir: '{{ replica_data_dir }}', port: '{{ replica_port }}' }
|
||||||
|
notify: stop postgresql
|
||||||
|
|
||||||
|
- name: Check connectivity to the master and get PostgreSQL version
|
||||||
|
become: yes
|
||||||
|
become_user: '{{ pg_user }}'
|
||||||
|
postgresql_ping:
|
||||||
|
db: '{{ db_default }}'
|
||||||
|
login_user: '{{ pg_user }}'
|
||||||
|
login_port: '{{ master_port }}'
|
||||||
|
register: result
|
||||||
|
|
||||||
|
- name: Check connectivity to the replica and get PostgreSQL version
|
||||||
|
become: yes
|
||||||
|
become_user: '{{ pg_user }}'
|
||||||
|
postgresql_ping:
|
||||||
|
db: '{{ db_default }}'
|
||||||
|
login_user: '{{ pg_user }}'
|
||||||
|
login_port: '{{ replica_port }}'
|
||||||
|
|
||||||
|
- name: Define server version
|
||||||
|
set_fact:
|
||||||
|
pg_major_version: '{{ result.server_version.major }}'
|
||||||
|
pg_minor_version: '{{ result.server_version.minor }}'
|
||||||
|
|
||||||
|
- name: Print PostgreSQL version
|
||||||
|
debug:
|
||||||
|
msg: 'PostgreSQL version is {{ pg_major_version }}.{{ pg_minor_version }}'
|
|
@ -0,0 +1,28 @@
|
||||||
|
# Important parameters:
|
||||||
|
listen_addresses='*'
|
||||||
|
port = {{ master_port }}
|
||||||
|
wal_level = logical
|
||||||
|
max_wal_senders = 8
|
||||||
|
track_commit_timestamp = on
|
||||||
|
max_replication_slots = 10
|
||||||
|
|
||||||
|
# Unimportant parameters:
|
||||||
|
max_connections=10
|
||||||
|
shared_buffers=8MB
|
||||||
|
dynamic_shared_memory_type=posix
|
||||||
|
log_destination='stderr'
|
||||||
|
logging_collector=on
|
||||||
|
log_directory='log'
|
||||||
|
log_filename='postgresql-%a.log'
|
||||||
|
log_truncate_on_rotation=on
|
||||||
|
log_rotation_age=1d
|
||||||
|
log_rotation_size=0
|
||||||
|
log_line_prefix='%m[%p]'
|
||||||
|
log_timezone='W-SU'
|
||||||
|
datestyle='iso,mdy'
|
||||||
|
timezone='W-SU'
|
||||||
|
lc_messages='en_US.UTF-8'
|
||||||
|
lc_monetary='en_US.UTF-8'
|
||||||
|
lc_numeric='en_US.UTF-8'
|
||||||
|
lc_time='en_US.UTF-8'
|
||||||
|
default_text_search_config='pg_catalog.english'
|
|
@ -0,0 +1,7 @@
|
||||||
|
local all all trust
|
||||||
|
local replication logical_replication trust
|
||||||
|
host replication logical_replication 127.0.0.1/32 trust
|
||||||
|
host replication logical_replication 0.0.0.0/0 trust
|
||||||
|
local all logical_replication trust
|
||||||
|
host all logical_replication 127.0.0.1/32 trust
|
||||||
|
host all logical_replication 0.0.0.0/0 trust
|
|
@ -0,0 +1,28 @@
|
||||||
|
# Important parameters:
|
||||||
|
listen_addresses='*'
|
||||||
|
port = {{ replica_port }}
|
||||||
|
wal_level = logical
|
||||||
|
max_wal_senders = 8
|
||||||
|
track_commit_timestamp = on
|
||||||
|
max_replication_slots = 10
|
||||||
|
|
||||||
|
# Unimportant parameters:
|
||||||
|
max_connections=10
|
||||||
|
shared_buffers=8MB
|
||||||
|
dynamic_shared_memory_type=posix
|
||||||
|
log_destination='stderr'
|
||||||
|
logging_collector=on
|
||||||
|
log_directory='log'
|
||||||
|
log_filename='postgresql-%a.log'
|
||||||
|
log_truncate_on_rotation=on
|
||||||
|
log_rotation_age=1d
|
||||||
|
log_rotation_size=0
|
||||||
|
log_line_prefix='%m[%p]'
|
||||||
|
log_timezone='W-SU'
|
||||||
|
datestyle='iso,mdy'
|
||||||
|
timezone='W-SU'
|
||||||
|
lc_messages='en_US.UTF-8'
|
||||||
|
lc_monetary='en_US.UTF-8'
|
||||||
|
lc_numeric='en_US.UTF-8'
|
||||||
|
lc_time='en_US.UTF-8'
|
||||||
|
default_text_search_config='pg_catalog.english'
|
Loading…
Reference in a new issue