-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathsend_message.py
executable file
·138 lines (109 loc) · 4.99 KB
/
send_message.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
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
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
# vim: set expandtab tabstop=4 shiftwidth=4 :
# send_message.py - A command line utility that sends arbitrary messages to a
# running Exocortex XMPP Bridge. This can be used for testing or for
# situations in which a (relatively) simple task has not been turned into a
# bot yet, but sending output via, say, SMTP is not feasible.
# By: The Doctor <drwho at virtadpt dot net>
# License: GPLv3
# v2.1 - Reformatted many of the references to use double-quotes, like the rest
# of my stuff.
# - Changed some print()s to logging.fatal()s.
# - Changed logging.warn() to logging.warning(), per deprecation.
# v2.0 - Ported to Python 3.
# v1.0 - Initial release.
# TO-DO:
# Load modules.
import argparse
import json
import logging
import requests
import sys
# Global variables.
# Handle to an argument parser.
argparser = None
# Handle to parsed command line arguments.
args = None
# Loglevel for the tool.
loglevel = None
# URL of XMPP bridge to contact. Assembled later.
message_queue = ""
# Hash table of custom headers that need to be sent to make it a valid request.
headers = {}
# Hash table that formats the message to send.
message = {}
# Handle to a requests object.
request = None
# Functions.
# Figure out what to set the logging level to. There isn't a straightforward
# way of doing this because Python uses constants that are actually integers
# under the hood, and I'd really like to be able to do something like
# loglevel = 'logging.' + loglevel
# I can't have a pony, either. Takes a string, returns a Python loglevel.
def process_loglevel(loglevel):
if loglevel == "critical":
return 50
if loglevel == "error":
return 40
if loglevel == "warning":
return 30
if loglevel == "info":
return 20
if loglevel == "debug":
return 10
if loglevel == "notset":
return 0
# Core code...
# Set up the command line argument parser.
argparser = argparse.ArgumentParser(description="A command line utility which allows the user to send text or data to an instance of the Exocortex XMPP Bridge. An ideal use case for this tool is to make interactive jobs communicate with the rest of an exocortex.", epilog="If you want to redirect stdout or stderr from something else so this utility can transmit it, make the last argument a - (per UNIX convention) to catch them, like this:\n echo foo | %(prog)s -")
# Set up the hostname of the XMPP bridge to contact.
argparser.add_argument("--hostname", action="store", default="localhost",
help="Specify the hostname of an XMPP bridge to contact. Defaults to localhost.")
# Set up the port of the XMPP bridge to contact.
argparser.add_argument("--port", action="store", default=8003,
help="Specify the network port of an XMPP bridge to contact. Defaults to 8003/tcp.")
# Define the name of a message queue to send messages to.
argparser.add_argument("--queue", action="store", default="replies",
help="Specify a message queue of an XMPP bridge to contact. Defaults to /replies.")
# Loglevels: critical, error, warning, info, debug, notset.
argparser.add_argument("--loglevel", action="store", default="info",
help="Valid log levels: critical, error, warning, info, debug, notset. Defaults to INFO.")
# Message to send to the XMPP bridge.
argparser.add_argument("--message", action="store", nargs="*",
help="Text message to send to the XMPP bridge.")
argparser.add_argument("infile", nargs="?", type=argparse.FileType("r"),
help="Text stream to send to the XMPP bridge. Give a - to use stdin.")
# Parse the command line args.
args = argparser.parse_args()
# Figure out how to configure the logger.
if args.loglevel:
loglevel = process_loglevel(args.loglevel.lower())
logging.basicConfig(level=loglevel, format="%(levelname)s: %(message)s")
logger = logging.getLogger(__name__)
if not args.message and not args.infile:
logging.fatal("You need to specify either a message on the command line or a - to refer to the utility's stdin.")
logger.debug("Command line arguments presented to the script:")
logger.debug(str(args))
# Assemble the URL of the XMPP bridge to contact.
message_queue = "http://" + args.hostname + ":" + str(args.port) + "/" + args.queue.strip("/")
# Set up custom headers.
headers = {"Content-type": "application/json"}
# Build the message to send.
message["name"] = args.queue.strip("/")
if args.message:
message["reply"] = " ".join(word for word in args.message)
if args.infile:
# Yeah, this is a little hacky, but it's clear enough to maintain later.
message["reply"] = args.infile.readlines()
message["reply"] = " ".join(word for word in message["reply"])
# Attempt to contact the message queue and send a message.
try:
logger.debug("Sending message to queue: " + message_queue)
request = requests.put(message_queue, headers=headers,
data=json.dumps(message))
logger.debug("Response from server: " + request.text)
except:
logger.warning("Connection attempt to message queue failed.")
# Fin.
sys.exit(0)