С некоторыми типами файлов простой унифицированный diff не очень полезен. Очевидный пример - бинарные файлы, но даже некоторые якобы текстовые - например, SSL-сертификаты (.pem
) - тоже попадают в эту категорию.
Итак, может ли задача указать собственный метод отображения различий? Что-то вроде:
- name: Update SSL certificate
copy:
src: etc/ssl/mycert.pem
dest: /etc/ssl/mycert.pem
diff:
exec: >
openssl x509 -in {{ old }} -noout -text > {{ old }}.txt
openssl x509 -in {{ new }} -noout -text > {{ new }}.txt
diff -U2 {{ old }}.txt {{ new }}.txt
rm -f {{ old }}.txt {{ new }}.txt
?
"Хак" @ konstantin-suvorov замечательный, но похоже, что он работает только для модулей, которые уже есть diff-метод. К сожалению, мой другой вариант использования включает модуль (command
), чего нет.
В частности, я призываю /usr/bin/apt-get update
на Ubuntu, который обновляет apt
кеш-файлы. Я хотел бы сбросить содержимое кеша в текстовой форме до и после, чтобы увидеть, что изменилось в обновлении - если что ...
Нет готового решения. Мне было любопытно оспорить это.
Вот пользовательский обратный вызов stdout (поместите в ./callback_plugins/diffhack.py
).
Он основан на обратном вызове stdout по умолчанию default.py
.
from ansible.plugins.callback.default import CallbackModule as DefaultCallback
from subprocess import check_output, CalledProcessError, STDOUT
from tempfile import mkstemp
import os
try:
from __main__ import display
except ImportError:
display = None
class CallbackModule(DefaultCallback):
def v2_on_file_diff(self, result):
def process_diff(diff, cmd):
for d in diff:
for ab in ('after','before'):
fd, fn = mkstemp()
print fn
with open(fn, 'w') as f:
f.write(d[ab])
try:
new_cmd = cmd.replace('%s',fn)
res = check_output(new_cmd, stderr=STDOUT, shell=True)
except CalledProcessError as e:
display.warning('Error occured while calling prediff_cmd "{}": (Code {}) {}'.format(cmd, e.returncode, e.output))
res = None
os.unlink(fn)
if res:
d[ab] = res
return diff
if 'prediff_cmd' in result._task_fields['vars']:
prediff_cmd = result._task_fields['vars']['prediff_cmd']
if result._task.loop and 'results' in result._result:
for res in result._result['results']:
if 'diff' in res and res['diff'] and res.get('changed', False):
res['diff'] = process_diff(res['diff'], prediff_cmd)
elif 'diff' in result._result and result._result['diff'] and result._result.get('changed', False):
result._result['diff'] = process_diff(result._result['diff'], prediff_cmd)
return super(CallbackModule, self).v2_on_file_diff(result)
Затем вы можете установить prediff_cmd
как переменная задачи с %s
заменяется временным файлом:
- copy:
src: cert.pem
dest: /tmp/cert.pem
vars:
prediff_cmd: openssl x509 -in %s -noout -text | head -n 10
И установите обратный вызов по умолчанию, например:
ANSIBLE_STDOUT_CALLBACK=diffhack ansible-playbook -vv --check --diff test.yml
Чтобы увидеть этот результат:
TASK [copy] ***************************
/var/folders/4c/4t2731dn3csd9b4sv4d2xm5m0000gn/T/tmpS5fOdj
/var/folders/4c/4t2731dn3csd9b4sv4d2xm5m0000gn/T/tmpS4UM7N
--- before: /tmp/cert.pem
+++ after: /path/to/local/cert.pem
@@ -2,9 +2,9 @@
Data:
Version: 3 (0x2)
Serial Number:
- aa:fc:53:d8:29:d2:8a:58
+ d2:05:f8:5c:61:ff:9e:d3
Signature Algorithm: sha256WithRSAEncryption
Issuer: CN=localhost
Validity
- Not Before: Oct 13 07:32:01 2017 GMT
- Not After : Oct 13 07:32:01 2018 GMT
+ Not Before: Oct 13 07:32:25 2017 GMT
+ Not After : Oct 13 07:32:25 2018 GMT