summaryrefslogtreecommitdiff
path: root/action_plugins
diff options
context:
space:
mode:
Diffstat (limited to 'action_plugins')
-rw-r--r--action_plugins/syncthing_init.py79
-rw-r--r--action_plugins/syncthing_validate.py98
2 files changed, 177 insertions, 0 deletions
diff --git a/action_plugins/syncthing_init.py b/action_plugins/syncthing_init.py
new file mode 100644
index 0000000..eb6cf9f
--- /dev/null
+++ b/action_plugins/syncthing_init.py
@@ -0,0 +1,79 @@
+from ansible.plugins.action import ActionBase
+import re
+import yaml
+from yaml.loader import SafeLoader
+
+class ActionModule(ActionBase):
+ def run(self, tmp=None, task_vars=None):
+ if task_vars is None:
+ task_vars = dict()
+ result = super(ActionModule, self).run(tmp, task_vars)
+ result['changed'] = False
+ result['failed'] = False
+
+ error_msgs = []
+
+ ### Syncthing variables for non ansible hosts #########################
+ peers = {}
+ with open('syncthing_data.yaml') as f:
+ data = yaml.load(f, Loader=SafeLoader)
+ for peer in data:
+ peers[peer['name']] = peer
+ if not 'address' in peer.keys():
+ peer['address'] = 'dynamic'
+
+ ### Syncthing host vars ###############################################
+ for hostname, hostvars in task_vars['hostvars'].items() :
+ if 'syncthing' in hostvars.keys():
+ syncthing = hostvars['syncthing']
+ peer = {
+ 'address': 'dynamic',
+ 'id': '0000000-0000000-0000000-0000000-0000000-0000000-0000000-0000000',
+ 'shared': [],
+ }
+ if 'address' in syncthing.keys():
+ peer['address'] = syncthing['address']
+ for shared in syncthing['shared']:
+ peer['shared'].append({ 'name': shared['name'], 'path': shared['path'], 'peers': shared['peers']})
+ if 'syncthing' in hostvars['ansible_local']:
+ peer['id'] = hostvars['ansible_local']['syncthing']['id']
+ peers[hostname] = peer
+
+ ### Compiling host configuration ######################################
+ config = {}
+ if task_vars['ansible_host'] in peers.keys():
+ myself = peers[task_vars['ansible_host']]
+ config = {
+ 'config_path': "",
+ 'folders_to_create': [],
+ 'packages': [],
+ 'peers': {},
+ 'service': "syncthing",
+ 'shared': myself['shared'],
+ 'user_group': "syncthing",
+ }
+ if task_vars['ansible_distribution'] == 'FreeBSD':
+ config['config_path'] = "/usr/local/etc/syncthing/config.xml"
+ config['folders_to_create'] = ["/usr/local/etc/syncthing/", "/var/syncthing"]
+ config['packages'] = ["p5-libwww", "syncthing"]
+ elif task_vars['ansible_distribution'] == 'Gentoo':
+ config['config_path'] = "/var/lib/syncthing/.config/syncthing/config.xml"
+ config['folders_to_create'] = ["/var/lib/syncthing/.config/syncthing"]
+ config['packages'] = ["net-p2p/syncthing"]
+ else:
+ error_msgs.append(f"syncthing role does not support {task_vars['ansible_distribution']} hosts yet")
+ for shared in myself['shared']:
+ for peer in shared['peers']:
+ if not peer in config['peers'].keys():
+ config['peers'][peer] = { 'id': peers[peer]['id'], 'address': peers[peer]['address'] }
+
+ ### Results compilation ##############################################
+ if error_msgs != []:
+ result['msg'] = ' ; '.join(error_msgs)
+ result['failed'] = True
+
+ result['ansible_facts'] = {
+ 'syncthing_config': config,
+ }
+
+ return result
diff --git a/action_plugins/syncthing_validate.py b/action_plugins/syncthing_validate.py
new file mode 100644
index 0000000..df2c274
--- /dev/null
+++ b/action_plugins/syncthing_validate.py
@@ -0,0 +1,98 @@
+from ansible.plugins.action import ActionBase
+import re
+import yaml
+from yaml.loader import SafeLoader
+
+class ActionModule(ActionBase):
+ def run(self, tmp=None, task_vars=None):
+ if task_vars is None:
+ task_vars = dict()
+ result = super(ActionModule, self).run(tmp, task_vars)
+ result['changed'] = False
+ result['failed'] = False
+
+ error_msgs = []
+
+ ### syncthing_data.yaml file validation ###############################
+ peers = {}
+ with open('syncthing_data.yaml') as f:
+ data = yaml.load(f, Loader=SafeLoader)
+ if not isinstance(data, list):
+ error_msgs.append(f"The syncthing_data.yaml file must contain a list")
+ else:
+ for peer in data:
+ if not isinstance(peer, dict):
+ error_msgs.append(f"syncthing_data.yaml must contain a list of dicts")
+ else:
+ for key in peer.keys():
+ if not key in ['address', 'id', 'name', 'shared']:
+ error_msgs.append(f"Invalid key {key} in dict for syncthing_data.yaml entry")
+ if 'address' in peer.keys():
+ if not isinstance(peer['address'], str):
+ error_msgs.append(f"Invalid address in syncthing_data.yaml, must be of type string")
+ elif re.match('^tcp://', peer['address']) == None:
+ error_msgs.append(f"Invalid address in syncthing_data.yaml, must be of format tcp://<hostname or ip address>/")
+ if 'id' in peer.keys():
+ if not isinstance(peer['id'], str):
+ error_msgs.append(f"Invalid id in syncthing_data.yaml, must be of type string")
+ elif re.match('^(?:[A-Z0-9]{7}-){7}[A-Z0-9]{7}$', peer['id']) == None:
+ error_msgs.append(f"Invalid id in syncthing_data.yaml, must be of valid")
+ if 'name' in peer.keys():
+ if not isinstance(peer['name'], str):
+ error_msgs.append(f"Invalid name in syncthing_data.yaml, must be of type string")
+ elif not re.match('^[A-Za-z0-9\.]+$', peer['name']) == None:
+ error_msgs.append(f"Invalid name in syncthing_data.yaml, must be name a valid hostname")
+ if 'shared' in peer.keys():
+ # TODO validate shared and populate peers
+ pass
+
+ ### host_vars validations #############################################
+ for hostname, hostvars in task_vars['hostvars'].items() :
+ if 'syncthing' in hostvars.keys():
+ if not isinstance(task_vars['syncthing'], dict):
+ error_msgs.append(f"The syncthing variable must be of type dict for host {hostname}")
+ else:
+ syncthing = hostvars['syncthing']
+ for key in syncthing.keys():
+ if not key in ['address', 'shared']:
+ error_msgs.append(f"Invalid key {key} in the syncthing dict for host {hostname}")
+ peer = {
+ 'address': 'dynamic',
+ 'shared': [],
+ }
+ if 'address' in syncthing.keys():
+ if not isinstance(syncthing['address'], str):
+ error_msgs.append(f"Invalid address for host {hostname}: must be of type string")
+ elif re.match('^tcp://', syncthing['address']) == None:
+ error_msgs.append(f"Invalid address for host {hostname}: must be of format tcp://<hostname or ip address>/")
+ else:
+ peer['address'] = syncthing['address']
+ if 'shared' not in syncthing.keys():
+ error_msgs.append(f"Invalid syncthing entry for host {hostname}: no shared key in dict")
+ elif not isinstance(syncthing['shared'], list):
+ error_msgs.append(f"Invalid shared syncthing entry for host {hostname}: must be of type list")
+ elif len(syncthing['shared']) == 0:
+ error_msgs.append(f"Invalid shared syncthing entry for host {hostname}: must be a non empty list")
+ else:
+ for shared in syncthing['shared']:
+ if not isinstance(shared, dict):
+ error_msgs.append(f"Invalid shared syncthing entry for host {hostname}: shared needs to be a dict")
+ else:
+ for key in shared.keys():
+ if not key in ['name', 'path', 'peers']:
+ error_msgs.append(f"Invalid key {key} in the shared syncthing array for host {hostname}")
+ if 'name' not in shared.keys():
+ error_msgs.append(f"Invalid shared syncthing entry for host {hostname}: no name key in dict")
+ elif not isinstance(shared['name'], str):
+ error_msgs.append(f"Invalid shared name for host {hostname}: must be of type string")
+ #elif not shared['name'] in task_vars['hostvars']:
+ # error_msgs.append(f"Invalid shared name for host {hostname}: must be an ansible host, or defined in syncthing_data.yaml")
+ # TODO keep validating each key
+
+ ### Results compilation ##############################################
+ if error_msgs != []:
+ result['msg'] = ' ; '.join(error_msgs)
+ result['failed'] = True
+
+ return result
+