From b12cf754f63e49c915f8a8196a2619e42e83a93f Mon Sep 17 00:00:00 2001 From: Scott Buchanan Date: Thu, 17 May 2018 17:29:47 -0400 Subject: [PATCH] 1Password lookup plugin (#37207) * add pytest_cache to gitignore * onepassword lookup plugin * fix linter/style test complaints * second pass at making pycodestyle happy * use json module instead of jq * update copyrights, license & version added * fix python2 compatibility * doh. fix spacing issue. * use standard ansible exception * remove potentially problematic stdin argument * actually call assertion method * add support for top-level fields * make vault uuids pedantically consistent in fixture * fix new style issues * ability specify section & correct case handling * improve error handling * add onepassword_raw plugin * Add maintainer info * Move common code to module_utils/onepassword.py * Load raw data JSON data for easier use in Ansible * Put OnePass class back inside lookup plugin There is no good place for sharing code across lookups currently. * Remove debugging code in unit tests * Patche proper module in raw unit tests * Add changelog entry Co-authored-by: Scott Buchanan --- .github/BOTMETA.yml | 8 + .gitignore | 1 + changelogs/fragments/onepassword-lookup.yaml | 2 + lib/ansible/plugins/lookup/onepassword.py | 136 ++++++++ lib/ansible/plugins/lookup/onepassword_raw.py | 65 ++++ test/units/plugins/lookup/test_onepassword.py | 313 ++++++++++++++++++ 6 files changed, 525 insertions(+) create mode 100644 changelogs/fragments/onepassword-lookup.yaml create mode 100644 lib/ansible/plugins/lookup/onepassword.py create mode 100644 lib/ansible/plugins/lookup/onepassword_raw.py create mode 100644 test/units/plugins/lookup/test_onepassword.py diff --git a/.github/BOTMETA.yml b/.github/BOTMETA.yml index 587ac36ac0a32c..7ca1fdd2cab54e 100644 --- a/.github/BOTMETA.yml +++ b/.github/BOTMETA.yml @@ -903,6 +903,8 @@ files: $module_utils/network: maintainers: $team_networking labels: networking + $module_utils/onepassword.py: + maintainers: samdoran lib/ansible/playbook/handler.py: keywords: - handlers @@ -1042,6 +1044,12 @@ files: lib/ansible/plugins/netconf/: maintainers: $team_networking labels: networking + lib/ansible/plugins/lookup/onepassword.py: + maintainers: samdoran + ignored: azenk + lib/ansible/plugins/lookup/onepassword_raw.py: + maintainers: samdoran + ignored: azenk lib/ansible/plugins/shell/powershell.py: maintainers: $team_windows_core labels: diff --git a/.gitignore b/.gitignore index 893d7e00f93bce..ee3a22558e3557 100644 --- a/.gitignore +++ b/.gitignore @@ -57,6 +57,7 @@ credentials.yml # test output *.retry *.out +.pytest_cache/ .tox .cache .pytest_cache diff --git a/changelogs/fragments/onepassword-lookup.yaml b/changelogs/fragments/onepassword-lookup.yaml new file mode 100644 index 00000000000000..7a8cbe70956544 --- /dev/null +++ b/changelogs/fragments/onepassword-lookup.yaml @@ -0,0 +1,2 @@ +features: + - onepassword lookup - add lookup plugins onepassword and onepassword_raw to retrieve secrets from 1Password vault (https://github.com/ansible/ansible/pull/37207) diff --git a/lib/ansible/plugins/lookup/onepassword.py b/lib/ansible/plugins/lookup/onepassword.py new file mode 100644 index 00000000000000..c172acd5fa2efb --- /dev/null +++ b/lib/ansible/plugins/lookup/onepassword.py @@ -0,0 +1,136 @@ +# -*- coding: utf-8 -*- +# (c) 2018, Scott Buchanan +# (c) 2016, Andrew Zenk (lastpass.py used as starting point) +# (c) 2018, Ansible Project +# 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 = """ + lookup: onepassword + author: + - Scott Buchanan + - Andrew Zenk + version_added: "2.6" + requirements: + - C(op) 1Password command line utility. See U(https://support.1password.com/command-line/) + - must have already logged into 1Password using C(op) CLI + short_description: fetch field values from 1Password + description: + - onepassword wraps the C(op) command line utility to fetch specific field values from 1Password + options: + _terms: + description: identifier(s) (UUID, name or domain; case-insensitive) of item(s) to retrieve + required: True + field: + description: field to return from each matching item (case-insensitive) + default: 'password' + section: + description: item section containing the field to retrieve (case-insensitive); if absent will return first match from any section + default: None + vault: + description: vault containing the item to retrieve (case-insensitive); if absent will search all vaults + default: None +""" + +EXAMPLES = """ +- name: "retrieve password for KITT" + debug: + msg: "{{ lookup('onepassword', 'KITT') }}" + +- name: "retrieve password for Wintermute" + debug: + msg: "{{ lookup('onepassword', 'Tessier-Ashpool', section='Wintermute') }}" + +- name: "retrieve username for HAL" + debug: + msg: "{{ lookup('onepassword', 'HAL 9000', field='username', vault='Discovery') }}" +""" + +RETURN = """ + _raw: + description: field data requested +""" + +import json +import errno + +from subprocess import Popen, PIPE + +from ansible.plugins.lookup import LookupBase +from ansible.errors import AnsibleLookupError + + +class OnePass(object): + + def __init__(self, path='op'): + self._cli_path = path + + @property + def cli_path(self): + return self._cli_path + + def assert_logged_in(self): + try: + self._run(["get", "account"]) + except OSError as e: + if e.errno == errno.ENOENT: + raise AnsibleLookupError("1Password CLI tool not installed in path on control machine") + raise e + except AnsibleLookupError: + raise AnsibleLookupError("Not logged into 1Password: please run 'op signin' first") + + def get_raw(self, item_id, vault=None): + args = ["get", "item", item_id] + if vault is not None: + args += ['--vault={0}'.format(vault)] + output, dummy = self._run(args) + return output + + def get_field(self, item_id, field, section=None, vault=None): + output = self.get_raw(item_id, vault) + return self._parse_field(output, field, section) if output != '' else '' + + def _run(self, args, expected_rc=0): + p = Popen([self.cli_path] + args, stdout=PIPE, stderr=PIPE, stdin=PIPE) + out, err = p.communicate() + rc = p.wait() + if rc != expected_rc: + raise AnsibleLookupError(err) + return out, err + + def _parse_field(self, data_json, field_name, section_title=None): + data = json.loads(data_json) + if section_title is None: + for field_data in data['details'].get('fields', []): + if field_data.get('name').lower() == field_name.lower(): + return field_data.get('value', '') + for section_data in data['details'].get('sections', []): + if section_title is not None and section_title.lower() != section_data['title'].lower(): + continue + for field_data in section_data.get('fields', []): + if field_data.get('t').lower() == field_name.lower(): + return field_data.get('v', '') + return '' + + +class LookupModule(LookupBase): + + def run(self, terms, variables=None, **kwargs): + op = OnePass() + + op.assert_logged_in() + + field = kwargs.get('field', 'password') + section = kwargs.get('section') + vault = kwargs.get('vault') + + values = [] + for term in terms: + values.append(op.get_field(term, field, section, vault)) + return values diff --git a/lib/ansible/plugins/lookup/onepassword_raw.py b/lib/ansible/plugins/lookup/onepassword_raw.py new file mode 100644 index 00000000000000..9bc5e1cabd84be --- /dev/null +++ b/lib/ansible/plugins/lookup/onepassword_raw.py @@ -0,0 +1,65 @@ +# -*- coding: utf-8 -*- +# (c) 2018, Scott Buchanan +# (c) 2016, Andrew Zenk (lastpass.py used as starting point) +# (c) 2018, Ansible Project +# 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 = """ + lookup: onepassword_raw + author: + - Scott Buchanan + - Andrew Zenk + version_added: "2.6" + requirements: + - C(op) 1Password command line utility. See U(https://support.1password.com/command-line/) + - must have already logged into 1Password using op CLI + short_description: fetch raw json data from 1Password + description: + - onepassword_raw wraps C(op) command line utility to fetch an entire item from 1Password + options: + _terms: + description: identifier(s) (UUID, name, or domain; case-insensitive) of item(s) to retrieve + required: True + vault: + description: vault containing the item to retrieve (case-insensitive); if absent will search all vaults + default: None +""" + +EXAMPLES = """ +- name: "retrieve all data about Wintermute" + debug: + msg: "{{ lookup('onepassword_raw', 'Wintermute') }}" +""" + +RETURN = """ + _raw: + description: field data requested +""" + +import json + +from ansible.plugins.lookup.onepassword import OnePass +from ansible.plugins.lookup import LookupBase + + +class LookupModule(LookupBase): + + def run(self, terms, variables=None, **kwargs): + op = OnePass() + + op.assert_logged_in() + + vault = kwargs.get('vault') + + values = [] + for term in terms: + data = json.loads(op.get_raw(term, vault)) + values.append(data) + return values diff --git a/test/units/plugins/lookup/test_onepassword.py b/test/units/plugins/lookup/test_onepassword.py new file mode 100644 index 00000000000000..f1d0aa71b617e7 --- /dev/null +++ b/test/units/plugins/lookup/test_onepassword.py @@ -0,0 +1,313 @@ +# (c) 2018, Scott Buchanan +# (c) 2016, Andrew Zenk (test_lastpass.py used as starting point) +# 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 + +import json +import datetime + +try: + from urllib.parse import urlparse +except ImportError: + from urlparse import urlparse + +from argparse import ArgumentParser + + +from ansible.compat.tests import unittest +from ansible.compat.tests.mock import patch +from ansible.errors import AnsibleError +from ansible.plugins.lookup.onepassword import OnePass, LookupModule +from ansible.plugins.lookup.onepassword_raw import LookupModule as OnePasswordRawLookup + + +# Intentionally excludes metadata leaf nodes that would exist in real output if not relevant. +MOCK_ENTRIES = [ + { + 'vault_name': 'Acme "Quot\'d" Servers', + 'queries': [ + '0123456789', + 'Mock "Quot\'d" Server' + ], + 'output': { + 'uuid': '0123456789', + 'vaultUuid': '2468', + 'overview': { + 'title': 'Mock "Quot\'d" Server' + }, + 'details': { + 'sections': [{ + 'title': '', + 'fields': [ + {'t': 'username', 'v': 'jamesbond'}, + {'t': 'password', 'v': 't0pS3cret'}, + {'t': 'notes', 'v': 'Test note with\nmultiple lines and trailing space.\n\n'}, + {'t': 'tricksy "quot\'d" field\\', 'v': '"quot\'d" value'} + ] + }] + } + } + }, + { + 'vault_name': 'Acme Logins', + 'queries': [ + '9876543210', + 'Mock Website', + 'acme.com' + ], + 'output': { + 'uuid': '9876543210', + 'vaultUuid': '1357', + 'overview': { + 'title': 'Mock Website', + 'URLs': [ + {'l': 'website', 'u': 'https://acme.com/login'} + ] + }, + 'details': { + 'sections': [{ + 'title': '', + 'fields': [ + {'t': 'password', 'v': 't0pS3cret'} + ] + }] + } + } + }, + { + 'vault_name': 'Acme Logins', + 'queries': [ + '864201357' + ], + 'output': { + 'uuid': '864201357', + 'vaultUuid': '1357', + 'overview': { + 'title': 'Mock Something' + }, + 'details': { + 'fields': [ + { + 'value': 'jbond@mi6.gov.uk', + 'name': 'emailAddress' + }, + { + 'name': 'password', + 'value': 'vauxhall' + } + ] + } + } + }, +] + + +def get_mock_query_generator(require_field=None): + def _process_field(field, section_title=None): + field_name = field.get('name', field.get('t')) + field_value = field.get('value', field.get('v')) + + if require_field is None or field_name == require_field: + return entry, query, section_title, field_name, field_value + + for entry in MOCK_ENTRIES: + for query in entry['queries']: + for field in entry['output']['details'].get('fields', []): + fixture = _process_field(field) + if fixture: + yield fixture + for section in entry['output']['details'].get('sections', []): + for field in section['fields']: + fixture = _process_field(field, section['title']) + if fixture: + yield fixture + + +def get_one_mock_query(require_field=None): + generator = get_mock_query_generator(require_field) + return next(generator) + + +class MockOnePass(OnePass): + + _mock_logged_out = False + _mock_timed_out = False + + def _lookup_mock_entry(self, key, vault=None): + for entry in MOCK_ENTRIES: + if vault is not None and vault.lower() != entry['vault_name'].lower() and vault.lower() != entry['output']['vaultUuid'].lower(): + continue + + match_fields = [ + entry['output']['uuid'], + entry['output']['overview']['title'] + ] + + # Note that exactly how 1Password matches on domains in non-trivial cases is neither documented + # nor obvious, so this may not precisely match the real behavior. + urls = entry['output']['overview'].get('URLs') + if urls is not None: + match_fields += [urlparse(url['u']).netloc for url in urls] + + if key in match_fields: + return entry['output'] + + def _run(self, args, expected_rc=0): + parser = ArgumentParser() + + command_parser = parser.add_subparsers(dest='command') + + get_parser = command_parser.add_parser('get') + get_options = ArgumentParser(add_help=False) + get_options.add_argument('--vault') + get_type_parser = get_parser.add_subparsers(dest='object_type') + get_type_parser.add_parser('account', parents=[get_options]) + get_item_parser = get_type_parser.add_parser('item', parents=[get_options]) + get_item_parser.add_argument('item_id') + + args = parser.parse_args(args) + + def mock_exit(output='', error='', rc=0): + if rc != expected_rc: + raise AnsibleError(error) + if error != '': + now = datetime.date.today() + error = '[LOG] {0} (ERROR) {1}'.format(now.strftime('%Y/%m/%d %H:$M:$S'), error) + return output, error + + if args.command == 'get': + if self._mock_logged_out: + return mock_exit(error='You are not currently signed in. Please run `op signin --help` for instructions', rc=1) + + if self._mock_timed_out: + return mock_exit(error='401: Authentication required.', rc=1) + + if args.object_type == 'item': + mock_entry = self._lookup_mock_entry(args.item_id, args.vault) + + if mock_entry is None: + return mock_exit(error='Item {0} not found'.format(args.item_id)) + + return mock_exit(output=json.dumps(mock_entry)) + + if args.object_type == 'account': + # Since we don't actually ever use this output, don't bother mocking output. + return mock_exit() + + raise AnsibleError('Unsupported command string passed to OnePass mock: {0}'.format(args)) + + +class LoggedOutMockOnePass(MockOnePass): + + _mock_logged_out = True + + +class TimedOutMockOnePass(MockOnePass): + + _mock_timed_out = True + + +class TestOnePass(unittest.TestCase): + + def test_onepassword_cli_path(self): + op = MockOnePass(path='/dev/null') + self.assertEqual('/dev/null', op.cli_path) + + def test_onepassword_logged_in(self): + op = MockOnePass() + try: + op.assert_logged_in() + except: + self.fail() + + def test_onepassword_logged_out(self): + op = LoggedOutMockOnePass() + with self.assertRaises(AnsibleError): + op.assert_logged_in() + + def test_onepassword_timed_out(self): + op = TimedOutMockOnePass() + with self.assertRaises(AnsibleError): + op.assert_logged_in() + + def test_onepassword_get(self): + op = MockOnePass() + query_generator = get_mock_query_generator() + for dummy, query, dummy, field_name, field_value in query_generator: + self.assertEqual(field_value, op.get_field(query, field_name)) + + def test_onepassword_get_raw(self): + op = MockOnePass() + for entry in MOCK_ENTRIES: + for query in entry['queries']: + self.assertEqual(json.dumps(entry['output']), op.get_raw(query)) + + def test_onepassword_get_not_found(self): + op = MockOnePass() + self.assertEqual('', op.get_field('a fake query', 'a fake field')) + + def test_onepassword_get_with_section(self): + op = MockOnePass() + dummy, query, section_title, field_name, field_value = get_one_mock_query() + self.assertEqual(field_value, op.get_field(query, field_name, section=section_title)) + + def test_onepassword_get_with_vault(self): + op = MockOnePass() + entry, query, dummy, field_name, field_value = get_one_mock_query() + for vault_query in [entry['vault_name'], entry['output']['vaultUuid']]: + self.assertEqual(field_value, op.get_field(query, field_name, vault=vault_query)) + + def test_onepassword_get_with_wrong_vault(self): + op = MockOnePass() + dummy, query, dummy, field_name, dummy = get_one_mock_query() + self.assertEqual('', op.get_field(query, field_name, vault='a fake vault')) + + def test_onepassword_get_diff_case(self): + op = MockOnePass() + entry, query, section_title, field_name, field_value = get_one_mock_query() + self.assertEqual( + field_value, + op.get_field( + query, + field_name.upper(), + vault=entry['vault_name'].upper(), + section=section_title.upper() + ) + ) + + +@patch('ansible.plugins.lookup.onepassword.OnePass', MockOnePass) +class TestLookupModule(unittest.TestCase): + + def test_onepassword_plugin_multiple(self): + lookup_plugin = LookupModule() + + entry = MOCK_ENTRIES[0] + field = entry['output']['details']['sections'][0]['fields'][0] + + self.assertEqual( + [field['v']] * len(entry['queries']), + lookup_plugin.run(entry['queries'], field=field['t']) + ) + + def test_onepassword_plugin_default_field(self): + lookup_plugin = LookupModule() + + dummy, query, dummy, dummy, field_value = get_one_mock_query('password') + self.assertEqual([field_value], lookup_plugin.run([query])) + + +@patch('ansible.plugins.lookup.onepassword_raw.OnePass', MockOnePass) +class TestOnePasswordRawLookup(unittest.TestCase): + + def test_onepassword_raw_plugin_multiple(self): + raw_lookup_plugin = OnePasswordRawLookup() + + entry = MOCK_ENTRIES[0] + raw_value = entry['output'] + + self.assertEqual( + [raw_value] * len(entry['queries']), + raw_lookup_plugin.run(entry['queries']) + )