wait_for 4.23 KB
Newer Older
1
#!/usr/bin/python
Jeroen Hoekx committed
2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25
# -*- coding: utf-8 -*-

# (c) 2012, Jeroen Hoekx <jeroen@hoekx.be>
#
# This file is part of Ansible
#
# Ansible is free software: you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
#
# Ansible is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with Ansible.  If not, see <http://www.gnu.org/licenses/>.

import socket
import datetime
import time
import sys

26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63
DOCUMENTATION = '''
---
module: wait_for
short_description: Waits for a given port to become accessible on a server.
description:
     - This is useful for when services are not immediately available after
       their init scripts return - which is true of certain Java application
       servers. It is also useful when starting guests with the M(virt) module and
       needing to pause until they are ready.
version_added: "0.7"
options:
  host:
    description:
      - hostname or IP address to wait for
    required: false
    default: "127.0.0.1"
    aliases: []
  timeout:
    description:
      - maximum number of seconds to wait for
    required: false
    default: 300
  delay:
    description:
      - number of seconds to wait before starting to poll
    required: false
    default: 0
  port:
    description:
      - port number to poll
    required: true
  state:
    description:
      - either C(started), or C(stopped) depending on whether the module should 
        poll for the port being open or closed.
    choices: [ "started", "stopped" ]
    default: "started"
notes: []
64
requirements: []
65 66 67
author: Jeroen Hoekx
'''

Michael DeHaan committed
68 69
EXAMPLES = '''
# wait 300 seconds for port 8000 to become open on the host, don't start checking for 10 seconds
70
- wait_for: port=8000 delay=10"
Michael DeHaan committed
71 72
'''

Jeroen Hoekx committed
73 74 75 76
def main():

    module = AnsibleModule(
        argument_spec = dict(
77
            host=dict(default='127.0.0.1'),
Jeroen Hoekx committed
78
            timeout=dict(default=300),
79
            connect_timeout=dict(default=5),
80 81 82
            delay=dict(default=0),
            port=dict(required=True),
            state=dict(default='started', choices=['started', 'stopped']),
Jeroen Hoekx committed
83 84 85 86 87
        ),
    )

    params = module.params

88
    host = params['host']
Jeroen Hoekx committed
89
    timeout = int(params['timeout'])
90
    connect_timeout = int(params['connect_timeout'])
91
    delay = int(params['delay'])
Jeroen Hoekx committed
92
    port = int(params['port'])
93
    state = params['state']
Jeroen Hoekx committed
94

95 96
    start = datetime.datetime.now()

97 98 99
    if delay:
        time.sleep(delay)

100
    if state == 'stopped':
101
        ### first wait for the host to go down
102
        end = start + datetime.timedelta(seconds=timeout)
Jeroen Hoekx committed
103

104 105
        while datetime.datetime.now() < end:
            s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
106
            s.settimeout(connect_timeout)
107 108
            try:
                s.connect( (host, port) )
109
                s.shutdown(socket.SHUT_RDWR)
110 111 112 113 114
                s.close()
                time.sleep(1)
            except:
                break
        else:
115 116
            elapsed = datetime.datetime.now() - start
            module.fail_json(msg="Timeout when waiting for %s:%s to stop." % (host, port), elapsed=elapsed.seconds)
Jeroen Hoekx committed
117

118
    elif state == 'started':
119
        ### wait for the host to come up
120
        end = start + datetime.timedelta(seconds=timeout)
121 122 123

        while datetime.datetime.now() < end:
            s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
124
            s.settimeout(connect_timeout)
125 126
            try:
                s.connect( (host, port) )
127
                s.shutdown(socket.SHUT_RDWR)
128 129 130 131
                s.close()
                break
            except:
                time.sleep(1)
132
                pass
133
        else:
134 135
            elapsed = datetime.datetime.now() - start
            module.fail_json(msg="Timeout when waiting for %s:%s" % (host, port), elapsed=elapsed.seconds)
136

137 138
    elapsed = datetime.datetime.now() - start
    module.exit_json(state=state, port=port, elapsed=elapsed.seconds)
Jeroen Hoekx committed
139 140 141 142

# this is magic, see lib/ansible/module_common.py
#<<INCLUDE_ANSIBLE_MODULE_COMMON>>
main()