Ansible: stdout ของคำสั่ง Store ในตัวแปรใหม่?


86

ภายใน playbook ของฉันฉันต้องการสร้างตัวแปรที่ถือเอาท์พุทของคำสั่งภายนอก หลังจากนั้นฉันต้องการใช้ประโยชน์จากตัวแปรนั้นในเทมเพลตสองสามรายการ

นี่คือส่วนที่เกี่ยวข้องของ Playbook:

  tasks:
    - name: Create variable from command
      command: "echo Hello"
      register: command_output
    - debug: msg="{{command_output.stdout}}"

    - name: Copy test service
      template: src=../templates/test.service.j2 dest=/tmp/test.service
    - name: Enable test service
      shell: systemctl enable /tmp/test.service
    - name: Start test service
      shell: systemctl start test.service

และสมมติว่านี่คือเทมเพลตของฉัน:

[Unit]
Description=MyApp
After=docker.service
Requires=docker.service

[Service]
TimeoutStartSec=0
ExecStartPre=-/usr/bin/docker kill busybox1
ExecStartPre=-/usr/bin/docker rm busybox1
ExecStartPre=/usr/bin/docker pull busybox
ExecStart=/usr/bin/docker run --name busybox1 busybox /bin/sh -c "while true; do echo {{ string_to_echo }}; sleep 1; done"

[Install]
WantedBy=multi-user.target

(สังเกตเห็น{{ string_to_echo }})

ดังนั้นสิ่งที่ฉันโดยทั่วไปที่กำลังมองหาวิธีที่จะเก็บเนื้อหาของcommand_output.stdout(ซึ่งถูกสร้างขึ้น / ดึงในช่วงงานแรก) string_to_echoในตัวแปรใหม่
ตัวแปรนั้นฉันต้องการใช้ในหลายเทมเพลตในภายหลัง

ฉันเดาว่าฉันสามารถใช้{{command_output.stdout}}ในเทมเพลตของฉันได้ แต่ฉันต้องการกำจัดสิ่งนั้นออก.stdoutเพื่อให้อ่านง่าย

คำตอบ:



71

ไม่จำเป็นต้องตั้งข้อเท็จจริง

    - shell: cat "hello"
      register: cat_contents

    - shell: echo "I cat hello"
      when: cat_contents.stdout == "hello"

14
สิ่งนี้มีประโยชน์ แต่หมายความว่าในภายหลังเมื่อคุณใช้ตัวแปรคุณต้องจำไว้ว่าจะใช้.stdoutด้วย
Tim Malone

21

การปรับเปลี่ยนเล็กน้อยนอกเหนือจากคำตอบของ @ udondan ฉันต้องการใช้ชื่อตัวแปรที่ลงทะเบียนซ้ำด้วยset_factเพื่อช่วยให้ไม่เกะกะน้อยที่สุด

ดังนั้นถ้าฉันจะลงทะเบียนโดยใช้ตัวแปรpskฉันจะใช้ชื่อตัวแปรเดียวกันกับการสร้างไฟล์set_fact.

ตัวอย่าง

- name: generate PSK
  shell: openssl rand -base64 48
  register: psk
  delegate_to: 127.0.0.1
  run_once: true

- set_fact: 
    psk={{ psk.stdout }}

- debug: var=psk
  run_once: true

จากนั้นเมื่อฉันเรียกใช้:

$ ansible-playbook -i inventory setup_ipsec.yml

 PLAY                                                                                                                                                                                [all] *************************************************************************************************************************************************************************

 TASK [Gathering                                                                                                                                                                     Facts] *************************************************************************************************************************************************************
 ok: [hostc.mydom.com]
 ok: [hostb.mydom.com]
 ok: [hosta.mydom.com]

 TASK [libreswan : generate                                                                                                                                                          PSK] ****************************************************************************************************************************************************
 changed: [hosta.mydom.com -> 127.0.0.1]

 TASK [libreswan :                                                                                                                                                                   set_fact] ********************************************************************************************************************************************************
 ok: [hosta.mydom.com]
 ok: [hostb.mydom.com]
 ok: [hostc.mydom.com]

 TASK [libreswan :                                                                                                                                                                   debug] ***********************************************************************************************************************************************************
 ok: [hosta.mydom.com] => {
     "psk": "6Tx/4CPBa1xmQ9A6yKi7ifONgoYAXfbo50WXPc1kGcird7u/pVso/vQtz+WdBIvo"
 }

 PLAY                                                                                                                                                                                RECAP *************************************************************************************************************************************************************************
 hosta.mydom.com    : ok=4    changed=1    unreachable=0    failed=0
 hostb.mydom.com    : ok=2    changed=0    unreachable=0    failed=0
 hostc.mydom.com    : ok=2    changed=0    unreachable=0    failed=0

8

ฉันเป็นมือใหม่ใน Ansible แต่ฉันขอแนะนำวิธีแก้ไขปัญหาถัดไป:

playbook.yml

...
vars:
  command_output_full:
    stdout: will be overriden below
  command_output: {{ command_output_full.stdout }}
...
...
...
tasks:
  - name: Create variable from command
    command: "echo Hello"
    register: command_output_full
  - debug: msg="{{ command_output }}"

มันน่าจะใช้ได้ (และใช้ได้กับฉัน) เพราะ Ansible ใช้การประเมินแบบขี้เกียจ แต่ดูเหมือนว่าจะตรวจสอบความถูกต้องก่อนการเปิดตัวดังนั้นฉันจึงต้องกำหนดเป็นcommand_output_full.stdoutvars

และแน่นอนว่าหากมีจำนวนตัวแทนในvarsส่วนนี้มากเกินไปก็จะดูน่าเกลียด


7

ในกรณีที่คุณต้องการจัดเก็บคำสั่งที่ซับซ้อนเพื่อเปรียบเทียบผลลัพธ์ของข้อความตัวอย่างเช่นเพื่อเปรียบเทียบเวอร์ชันของระบบปฏิบัติการสิ่งนี้อาจช่วยคุณได้:

tasks:
       - shell: echo $(cat /etc/issue | awk {'print $7'})
         register: echo_content

       - shell: echo "It works"
         when: echo_content.stdout == "12"
         register: out
       - debug: var=out.stdout_lines

1

หากคุณต้องการดำเนินการต่อไปและดึงข้อมูลที่ต้องการจากผลลัพธ์ Playbook ให้ใช้ภาษาแบบสอบถาม JSON เช่น jmespath ตัวอย่าง:

  - name: Sample Playbook
    // Fill up your task
    no_log: True
    register: example_output

  - name: Json Query
    set_fact:
      query_result:
        example_output:"{{ example_output | json_query('results[*].name') }}"
โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.