2020-09-29 07:56:09 +00:00
|
|
|
#!/usr/bin/python
|
|
|
|
# coding: utf-8
|
|
|
|
|
|
|
|
# (c) 2018, Jan Christian Grünhage <jan.christian@gruenhage.xyz>
|
|
|
|
# (c) 2020, Famedly GmbH
|
|
|
|
# GNU Affero General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/agpl-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 = '''
|
|
|
|
---
|
|
|
|
author: "Johanna Dorothea Reichmann (@transcaffeine)"
|
|
|
|
module: matrix_member
|
|
|
|
short_description: Manage matrix room membership
|
|
|
|
description:
|
|
|
|
- Manage the membership status of a given set of matrix users. Invitations (`state=member`), kicks and bans can be issued. With the `exclusive=True` flag, all other members in the room can be auto-kicked.
|
|
|
|
options:
|
|
|
|
hs_url:
|
|
|
|
description:
|
|
|
|
- URL of the homeserver, where the CS-API is reachable
|
|
|
|
required: true
|
|
|
|
token:
|
|
|
|
description:
|
|
|
|
- auth token
|
|
|
|
required: true
|
|
|
|
room_id:
|
|
|
|
description:
|
|
|
|
- ID of the room to manage
|
|
|
|
user_ids:
|
|
|
|
description:
|
|
|
|
- List of matrix IDs to set their state
|
|
|
|
required: true
|
|
|
|
state:
|
|
|
|
description:
|
|
|
|
- In which state all listed members should be: member|kicked|banned
|
|
|
|
required: true
|
|
|
|
exclusive:
|
|
|
|
description:
|
|
|
|
- If state=member, the module ensure only the specified user_ids are in the room by kicking every other user present in the room.
|
|
|
|
required: false
|
|
|
|
requirements:
|
|
|
|
- matrix-nio (Python library)
|
|
|
|
'''
|
|
|
|
|
|
|
|
EXAMPLES = '''
|
|
|
|
- name: Invite two users by matrix-ID into a room
|
|
|
|
matrix_member:
|
|
|
|
hs_url: "https://matrix.org"
|
|
|
|
token: "{{ matrix_auth_token }}"
|
|
|
|
room_id: "{{ matrix_room_id }}"
|
|
|
|
state: member
|
|
|
|
user_ids:
|
|
|
|
- @user1:matrix.org
|
|
|
|
- @user2:homeserver.tld
|
|
|
|
'''
|
|
|
|
|
|
|
|
RETURN = '''
|
|
|
|
members:
|
|
|
|
description: Dictionary of all members in the given room who are either invited or joined
|
|
|
|
returned: When auth_token is valid
|
|
|
|
type: dict[str]
|
|
|
|
'''
|
|
|
|
import traceback
|
|
|
|
import asyncio
|
|
|
|
|
|
|
|
from ansible.module_utils.basic import AnsibleModule, missing_required_lib
|
|
|
|
|
|
|
|
MATRIX_IMP_ERR = None
|
|
|
|
|
|
|
|
try:
|
|
|
|
from nio import AsyncClient, RoomBanError, RoomUnbanError, RoomKickError, RoomInviteError, RoomGetStateError
|
|
|
|
except ImportError:
|
|
|
|
MATRIX_IMP_ERR = traceback.format_exc()
|
2020-10-02 17:50:02 +00:00
|
|
|
MATRIX_FOUND = False
|
2020-09-29 07:56:09 +00:00
|
|
|
else:
|
2020-10-02 17:50:02 +00:00
|
|
|
MATRIX_FOUND = True
|
2020-09-29 07:56:09 +00:00
|
|
|
|
|
|
|
async def get_room_members(client, room_id, res):
|
|
|
|
member_resp = await client.room_get_state(room_id)
|
|
|
|
|
|
|
|
if isinstance(member_resp, RoomGetStateError):
|
|
|
|
res.msg = "Could not get room state for roomId={0}".format(room_id)
|
|
|
|
raise Exception()
|
|
|
|
else:
|
|
|
|
return dict(list(map(lambda m: (m['state_key'],m['content']['membership']), filter(lambda e: e['type'] == 'm.room.member' and
|
|
|
|
e['content']['membership'] in ['invite', 'join', 'leave', 'ban'], member_resp.events))))
|
|
|
|
|
|
|
|
async def ban_from_room(client, room_id, user_id, res):
|
|
|
|
ban_resp = await client.room_ban(room_id, user_id)
|
|
|
|
if isinstance(ban_resp, RoomBanError):
|
|
|
|
res['msg'] = "Could not ban user={0} from roomId={1}".format(user_id, room_id)
|
|
|
|
raise Exception()
|
2020-10-02 17:50:02 +00:00
|
|
|
res['changed'] = True
|
|
|
|
res['banned'].append(user_id)
|
2020-09-29 07:56:09 +00:00
|
|
|
|
|
|
|
async def unban_from_room(client, room_id, user_id, res):
|
|
|
|
ban_resp = await client.room_unban(room_id, user_id)
|
|
|
|
if isinstance(ban_resp, RoomUnbanError):
|
|
|
|
res['msg'] = "Could not unban user={0} from roomId={1}".format(user_id, room_id)
|
|
|
|
raise Exception()
|
2020-10-02 17:50:02 +00:00
|
|
|
res['changed'] = True
|
|
|
|
res['unbanned'].append(user_id)
|
2020-09-29 07:56:09 +00:00
|
|
|
|
|
|
|
async def kick_from_room(client, room_id, user_id, res):
|
|
|
|
kick_resp = await client.room_kick(room_id, user_id)
|
|
|
|
if isinstance(kick_resp, RoomKickError):
|
|
|
|
res['msg'] = "Could not kick user={0} from roomId={1}".format(user_id, room_id)
|
|
|
|
raise Exception()
|
2020-10-02 17:50:02 +00:00
|
|
|
res['changed'] = True
|
|
|
|
res['kicked'].append(user_id)
|
2020-09-29 07:56:09 +00:00
|
|
|
|
|
|
|
async def invite_to_room(client, room_id, user_id, res):
|
|
|
|
invite_resp = await client.room_invite(room_id, user_id)
|
|
|
|
if isinstance(invite_resp, RoomInviteError):
|
|
|
|
res['msg'] = "Could not invite user={0} to roomId={1}".format(user_id, room_id)
|
|
|
|
raise Exception()
|
2020-10-02 17:50:02 +00:00
|
|
|
res['changed'] = True
|
|
|
|
res['invited'].append(user_id)
|
2020-09-29 07:56:09 +00:00
|
|
|
|
|
|
|
|
|
|
|
async def run_module():
|
|
|
|
module_args = dict(
|
|
|
|
hs_url=dict(type='str', required=True),
|
|
|
|
token=dict(type='str', required=True, no_log=True),
|
|
|
|
state=dict(choices=['member', 'kicked', 'banned'], required=True),
|
|
|
|
room_id=dict(type='str', required=True),
|
|
|
|
user_ids=dict(type='list', required=True, elements='str'),
|
|
|
|
exclusive=dict(type='bool', required=False, default=False),
|
|
|
|
)
|
|
|
|
|
|
|
|
result = dict(
|
|
|
|
changed=False,
|
|
|
|
banned=[],
|
|
|
|
unbanned=[],
|
|
|
|
kicked=[],
|
|
|
|
invited=[],
|
|
|
|
members=[],
|
|
|
|
msg="",
|
|
|
|
)
|
|
|
|
|
|
|
|
module = AnsibleModule(
|
|
|
|
argument_spec=module_args,
|
|
|
|
supports_check_mode=True
|
|
|
|
)
|
2020-10-02 17:50:02 +00:00
|
|
|
if not MATRIX_FOUND:
|
2020-09-29 07:56:09 +00:00
|
|
|
module.fail_json(msg=missing_required_lib('matrix-nio'), exception=MATRIX_IMP_ERR)
|
|
|
|
|
|
|
|
if module.check_mode:
|
|
|
|
return result
|
|
|
|
|
|
|
|
action = module.params['state']
|
|
|
|
room_id = module.params['room_id']
|
|
|
|
user_ids = module.params['user_ids']
|
|
|
|
|
|
|
|
# Check for valid parameter combination
|
|
|
|
if module.params['exclusive'] and action != 'member':
|
|
|
|
module.fail_json(msg='exclusive=True can only be used with state=member')
|
|
|
|
|
|
|
|
# Create client object
|
|
|
|
client = AsyncClient(module.params['hs_url'])
|
|
|
|
client.access_token = module.params['token']
|
|
|
|
|
|
|
|
# Query all room members (invited users count as member, as they _can_ be in the room)
|
|
|
|
room_members = await get_room_members(client, room_id, result)
|
|
|
|
present_members = {m: s for m, s in room_members.items() if s in ['join', 'invite']}.keys()
|
|
|
|
banned_members = {m: s for m, s in room_members.items() if s == 'ban'}.keys()
|
|
|
|
|
|
|
|
if not module.params['exclusive']:
|
|
|
|
# Handle non-exclusive invite|kick|ban
|
|
|
|
try:
|
|
|
|
for user_id in user_ids:
|
|
|
|
if action == 'member' and user_id not in present_members:
|
|
|
|
if user_id in banned_members:
|
|
|
|
await unban_from_room(client, room_id, user_id, result)
|
|
|
|
await invite_to_room(client, room_id, user_id, result)
|
|
|
|
elif action == 'kicked' and user_id in present_members:
|
|
|
|
await kick_from_room(client, room_id, user_id, result)
|
|
|
|
elif action == 'kicked' and user_id in banned_members:
|
|
|
|
await unban_from_room(client, room_id, user_id, result)
|
|
|
|
elif action == 'banned' and user_id not in banned_members:
|
|
|
|
await ban_from_room(client, room_id, user_id, result)
|
|
|
|
except:
|
|
|
|
module.fail_json(**result)
|
|
|
|
else:
|
|
|
|
# Handle exclusive mode: get state and make lists of users to be kicked or invited
|
|
|
|
to_invite = list(filter(lambda m: m not in present_members, user_ids))
|
|
|
|
to_kick = list(filter(lambda m: m not in user_ids, present_members))
|
|
|
|
|
|
|
|
try:
|
|
|
|
for user_id in to_invite:
|
|
|
|
await invite_to_room(client, room_id, user_id, result)
|
|
|
|
for user_id in to_kick:
|
|
|
|
await kick_from_room(client, room_id, user_id, result)
|
|
|
|
except:
|
|
|
|
module.fail_json(**result)
|
|
|
|
|
|
|
|
# Get all current members from the room
|
|
|
|
try:
|
|
|
|
room_members_after = await get_room_members(client, room_id, result)
|
|
|
|
result['members'] = {m: s for m, s in room_members_after.items() if s in ['join', 'invite']}.keys()
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
|
|
|
|
await client.close()
|
|
|
|
|
|
|
|
module.exit_json(**result)
|
|
|
|
|
|
|
|
|
|
|
|
def main():
|
|
|
|
asyncio.run(run_module())
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
main()
|