2016-07-14 13:24:08 +00:00
|
|
|
#!/usr/bin/python
|
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
|
|
|
# (c) 2016, Dag Wieers <dag@wieers.com>
|
|
|
|
#
|
|
|
|
# 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/>.
|
|
|
|
|
2017-03-14 16:07:22 +00:00
|
|
|
ANSIBLE_METADATA = {'metadata_version': '1.0',
|
|
|
|
'status': ['preview'],
|
|
|
|
'supported_by': 'community'}
|
|
|
|
|
2017-07-07 09:52:02 +00:00
|
|
|
DOCUMENTATION = r'''
|
2016-07-14 13:24:08 +00:00
|
|
|
---
|
|
|
|
module: wakeonlan
|
2017-03-08 12:42:55 +00:00
|
|
|
version_added: '2.2'
|
2016-07-14 13:24:08 +00:00
|
|
|
short_description: Send a magic Wake-on-LAN (WoL) broadcast packet
|
|
|
|
description:
|
2017-03-08 12:42:55 +00:00
|
|
|
- The C(wakeonlan) module sends magic Wake-on-LAN (WoL) broadcast packets.
|
2016-07-14 13:24:08 +00:00
|
|
|
options:
|
|
|
|
mac:
|
|
|
|
description:
|
2017-03-08 12:42:55 +00:00
|
|
|
- MAC address to send Wake-on-LAN broadcast packet for.
|
2016-07-14 13:24:08 +00:00
|
|
|
required: true
|
|
|
|
broadcast:
|
|
|
|
description:
|
2017-03-08 12:42:55 +00:00
|
|
|
- Network broadcast address to use for broadcasting magic Wake-on-LAN packet.
|
2016-07-14 13:24:08 +00:00
|
|
|
default: 255.255.255.255
|
|
|
|
port:
|
|
|
|
description:
|
2017-03-08 12:42:55 +00:00
|
|
|
- UDP port to use for magic Wake-on-LAN packet.
|
2016-07-14 13:24:08 +00:00
|
|
|
default: 7
|
2017-07-07 09:52:02 +00:00
|
|
|
author:
|
|
|
|
- Dag Wieers (@dagwieers)
|
2016-07-14 13:24:08 +00:00
|
|
|
todo:
|
|
|
|
- Add arping support to check whether the system is up (before and after)
|
|
|
|
- Enable check-mode support (when we have arping support)
|
|
|
|
- Does not have SecureOn password support
|
|
|
|
notes:
|
|
|
|
- This module sends a magic packet, without knowing whether it worked
|
|
|
|
- Only works if the target system was properly configured for Wake-on-LAN (in the BIOS and/or the OS)
|
2017-07-07 09:52:02 +00:00
|
|
|
- Some BIOSes have a different (configurable) Wake-on-LAN boot order (i.e. PXE first).
|
2016-07-14 13:24:08 +00:00
|
|
|
'''
|
|
|
|
|
2017-07-07 09:52:02 +00:00
|
|
|
EXAMPLES = r'''
|
2017-03-08 12:42:55 +00:00
|
|
|
- name: Send a magic Wake-on-LAN packet to 00:00:5E:00:53:66
|
|
|
|
wakeonlan:
|
2016-12-01 11:17:32 +00:00
|
|
|
mac: '00:00:5E:00:53:66'
|
|
|
|
broadcast: 192.0.2.23
|
2017-01-05 20:27:11 +00:00
|
|
|
delegate_to: localhost
|
2016-12-01 11:17:32 +00:00
|
|
|
|
|
|
|
- wakeonlan:
|
|
|
|
mac: 00:00:5E:00:53:66
|
|
|
|
port: 9
|
2016-07-14 13:24:08 +00:00
|
|
|
delegate_to: localhost
|
|
|
|
'''
|
|
|
|
|
2017-07-07 09:52:02 +00:00
|
|
|
RETURN = r'''
|
2016-07-14 13:24:08 +00:00
|
|
|
# Default return values
|
|
|
|
'''
|
|
|
|
|
|
|
|
from ansible.module_utils.basic import AnsibleModule
|
|
|
|
from ansible.module_utils.pycompat24 import get_exception
|
|
|
|
import socket
|
|
|
|
import struct
|
|
|
|
|
|
|
|
|
|
|
|
def wakeonlan(module, mac, broadcast, port):
|
|
|
|
""" Send a magic Wake-on-LAN packet. """
|
|
|
|
|
|
|
|
mac_orig = mac
|
|
|
|
|
2016-12-11 02:50:09 +00:00
|
|
|
# Remove possible separator from MAC address
|
2016-07-14 13:24:08 +00:00
|
|
|
if len(mac) == 12 + 5:
|
|
|
|
mac = mac.replace(mac[2], '')
|
|
|
|
|
|
|
|
# If we don't end up with 12 hexadecimal characters, fail
|
|
|
|
if len(mac) != 12:
|
|
|
|
module.fail_json(msg="Incorrect MAC address length: %s" % mac_orig)
|
|
|
|
|
|
|
|
# Test if it converts to an integer, otherwise fail
|
|
|
|
try:
|
|
|
|
int(mac, 16)
|
|
|
|
except ValueError:
|
|
|
|
module.fail_json(msg="Incorrect MAC address format: %s" % mac_orig)
|
2017-01-27 23:45:23 +00:00
|
|
|
|
2016-07-14 13:24:08 +00:00
|
|
|
# Create payload for magic packet
|
2017-07-07 09:52:02 +00:00
|
|
|
data = b''
|
2016-07-14 13:24:08 +00:00
|
|
|
padding = ''.join(['FFFFFFFFFFFF', mac * 20])
|
|
|
|
for i in range(0, len(padding), 2):
|
2017-07-07 09:52:02 +00:00
|
|
|
data = b''.join([data, struct.pack('B', int(padding[i: i + 2], 16))])
|
2016-07-14 13:24:08 +00:00
|
|
|
|
2017-07-12 12:48:17 +00:00
|
|
|
# Broadcast payload to network
|
|
|
|
sock = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
|
|
|
|
sock.setsockopt(socket.SOL_SOCKET, socket.SO_BROADCAST, 1)
|
|
|
|
|
2017-07-07 09:52:02 +00:00
|
|
|
if not module.check_mode:
|
|
|
|
|
|
|
|
try:
|
|
|
|
sock.sendto(data, (broadcast, port))
|
|
|
|
except socket.error:
|
|
|
|
e = get_exception()
|
|
|
|
sock.close()
|
|
|
|
module.fail_json(msg=str(e))
|
2017-07-12 12:48:17 +00:00
|
|
|
|
|
|
|
sock.close()
|
2016-07-14 13:24:08 +00:00
|
|
|
|
|
|
|
|
|
|
|
def main():
|
|
|
|
module = AnsibleModule(
|
2017-07-07 09:52:02 +00:00
|
|
|
argument_spec=dict(
|
|
|
|
mac=dict(type='str', required=True),
|
|
|
|
broadcast=dict(type='str', default='255.255.255.255'),
|
|
|
|
port=dict(type='int', default=7),
|
2016-07-14 13:24:08 +00:00
|
|
|
),
|
2017-07-07 09:52:02 +00:00
|
|
|
supports_check_mode=True,
|
2016-07-14 13:24:08 +00:00
|
|
|
)
|
|
|
|
|
2017-03-08 12:42:55 +00:00
|
|
|
mac = module.params['mac']
|
|
|
|
broadcast = module.params['broadcast']
|
|
|
|
port = module.params['port']
|
|
|
|
|
2017-07-07 09:52:02 +00:00
|
|
|
wakeonlan(module, mac, broadcast, port)
|
2016-07-14 13:24:08 +00:00
|
|
|
|
|
|
|
module.exit_json(changed=True)
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
2016-12-01 11:17:32 +00:00
|
|
|
main()
|