#!/usr/bin/env python3 # -*- coding: utf-8 -*- # To run this bot: # $ python3 logbot.py # The output folder of this bot currently is: /var/www/logs/digital-autonomy import logging from getpass import getpass from argparse import ArgumentParser import slixmpp import ssl, os, requests, urllib from datetime import datetime from bs4 import BeautifulSoup import os, re, random def check_handle(handle, used_handles): if handle in used_handles: handle_is_already_used = True else: handle_is_already_used = False return handle_is_already_used def request_handle(used_handles_path): used_handles = open(used_handles_path, 'r').readlines() handles = open('handles.txt', 'r').readlines() handle = random.choice(handles).replace('\n','') # check if handle is not used yet! handle_is_already_used = False if handle in used_handles: handle_is_already_used = True while check_handle(handle, used_handles) == True: handle = random.choice(handles) # add handle to .handles.txt with open(used_handles_path, 'a+') as h: h.write(handle) return handle def rec(self, entry): output = self.output used_handles = '.handles.txt' used_handles_path = os.path.join(output, used_handles) # save entry handle = request_handle(used_handles_path) newfile_path = output + '/entries/' + handle + '.txt' with open(newfile_path, 'w') as f: f.write(entry) with open(used_handles_path, 'a+') as h: h.write(handle) print('Saved!') def delete(self, handle): used_handles = '.handles.txt' used_handles_path = os.path.join(self.output, used_handles) possible_file_paths = [ os.path.join(self.output, 'entries', handle + '.txt'), os.path.join(self.output, 'entries', handle + '.png'), os.path.join(self.output, 'entries', handle + '.jpg') ] for path in possible_file_paths: if os.path.isfile(path): cmd = f'rm { path }' print(f'> { cmd }') os.system(cmd) print(f'Removed { path }.') pass # delete handle from .handles.txt file # read with open(used_handles_path, 'r') as h: txt = h.read() txt = txt.replace(handle+'\n', "") # write with open(used_handles_path, 'w') as h: h.write(txt) # def write_to_log(self, entry): # output = self.output # # print(f'Output: { output }') # log = 'index.html' # css = 'stylesheet.css' # used_handles = '.handles.txt' # log_path = os.path.join(output, log) # css_path = os.path.join(output, css) # used_handles_path = os.path.join(output, used_handles) # # check if file exists, if not: write it! # if not os.path.isfile(log_path): # html_template = open('templates/index.html', 'r').read() # css_template = open('templates/stylesheet.css', 'r').read() # with open(log_path, 'w') as l: # l.write(html_template) # l.write(f'

{ self.groupchat }

') # with open(css_path, 'w') as c: # c.write(css_template) # with open(used_handles_path, 'w') as h: # h.write('-----') # # add entry to log # handle = request_handle(used_handles_path) # print(f'Picked a handle: { handle }') # now = datetime.now().strftime('%A %d %B (%Y)') # print(f'Now is: { now }') # post = f'''
# { handle } # { entry } # Added on { now } #
''' # print(f'Post: { post }') # with open(log_path, 'a+') as l: # l.write(post) # print('added to the log!') # with open(used_handles_path, 'a+') as h: # h.write(handle) # print('added to the .handles file!') # *spark # add annotations # def find_in_soup(self, handle, annotation): # print('--------ADD ANNOTATION ---------') # print(f'handle: { handle }') # log = 'index.html' # log_path = os.path.join(self.output, log) # html = open(log_path, 'r').read() # soup = BeautifulSoup(html, 'html.parser') # # print(soup.prettify()) # post = soup.find(id=handle) # # print(f'posts: { posts }') # # for post in posts: # print(f'post: { post }') # if post: # # annotationcontainer = post.findChildren(id="annotationcontainer", recursive=True)[0] # # print(f'annotationcontainer: { annotationcontainer }') # # print(f'annotationcontainer.contents: { annotationcontainer.contents }') # # annotationcontainer.contents.append(f'{ annotation }') # # print(f'annotationcontainer.contents: { annotationcontainer.contents }') # # new_annotation = soup.new_annotation("a", href="http://www.example.com") # new_annotation = soup.new_annotation("span") # new_annotation.append(annotation) # soup.find(id=handle).find(class_="annotationcontainer").append(new_annotation) # print(f'new soup: { str(soup) } ') # # write soup to file # with open(log_path, 'w') as l: # l.write(str(soup)) class MUCBot(slixmpp.ClientXMPP): """ A simple Slixmpp bot that will save images and messages that are marked with @bot to a folder. """ def __init__(self, use, password, groupchat, nickname, output, mode): slixmpp.ClientXMPP.__init__(self, use, password) self.groupchat = groupchat self.nick = nickname self.output = output self.mode = mode # The session_start event will be triggered when # the bot establishes its connection with the server # and the XML logs are ready for use. We want to # listen for this event so that we we can initialize # our roster. self.add_event_handler("session_start", self.start) # The groupchat_message event is triggered whenever a message # stanza is received from any chat room. If you also also # register a handler for the 'message' event, MUC messages # will be processed by both handlers. self.add_event_handler("groupchat_message", self.muc_message) def start(self, event): self.get_roster() self.send_presence() # https://xmpp.org/extensions/xep-0045.html self.plugin['xep_0045'].join_muc(self.groupchat, self.nick, # If a room password is needed, use: # password=the_room_password, wait=True) def muc_message(self, msg): # Some inspection commands #print('Message: {}'.format(msg)) # Always check that a message is not the bot itself, otherwise you will create an infinite loop responding to your own messages. if msg['mucnick'] != self.nick: if '@bot' in msg['body']: # Send some info about this bot. self.send_message( mto=self.groupchat, mbody='''Hello! RECbot here. I\'m a new version of logbot. You can log type of text messages, by including __ADD__ in your message. Or, you can send an image/sound(*spark)/video(*spark) file to this chat and it will be logged for you. Items in the log can also be deleted again, by using the unique HANDLE of each post. You can find these handles in the generated HTML page, they look like this: +//-*. Happy logging! PS. you can access these logs at https://vvvvvvaria.org/logs/.''', mtype='groupchat' ) else: # Respond to incoming __ACTION_WORDS__! # Check if output folder exists if not os.path.exists(self.output): os.mkdir(self.output) os.mkdir(self.output + '/entries/') with open(os.path.join(self.output, '.handles.txt'), 'w') as f: pass # Check if an OOB URL is included in the stanza (which is how an image is sent) # (OOB object - https://xmpp.org/extensions/xep-0066.html#x-oob) if len(msg['oob']['url']) > 0: # Send a reply self.send_message(mto=self.groupchat, mbody="Super, our log is growing. Your image is added!", mtype='groupchat') # Save the image to the output folder url = msg['oob']['url'] # grep the url in the message filename = os.path.basename(url) # grep the filename in the url output_path = os.path.join(self.output, filename) u = urllib.request.urlopen(url) # read the image data f = open(output_path, 'wb') # open the output file f.write(u.read()) # write image to file f.close() # close the output file # Add the image to the log img = f'
' write_to_log(self, img) # Include a new post in the log (only when '__ADD__' is used in the message) if '__ADD__' in msg['body']: # reply from the bot self.send_message(mto=self.groupchat, mbody=f'Noted! And added to the log. Thanks { msg["mucnick"] }!', mtype='groupchat') # Record the entry rec(self, msg['body']) # Delete a post from the log if '__DELETE__' in msg['body']: handle = re.findall("[aioeu][aioeu][aioeu][aioeu][aioeu]", msg['body'])[0] # reply from the bot self.send_message(mto=self.groupchat, mbody=f'Noted! The following post is deleted from the log: { handle }', mtype='groupchat') # Delete the entry delete(self, handle) # Include a new post in the log (only when '__ADD__' is used in the message) # if '__ANNOTATE__' in msg['body']: # handle = msg['body'].split()[1] # annotation = msg['body'].replace('__ANNOTATE__', '').replace(handle, '') # post = find_in_soup(self, handle, annotation) # # reply from the bot # self.send_message(mto=self.groupchat, # mbody="Thanks!", # mtype='groupchat') # Check if this is a book ... if '__BOOK__' in msg['body']: self.send_message(mto=self.groupchat, mbody="Oh a book, that's cool! Thanks {}!".format(msg['mucnick']), mtype='groupchat') # Start of book feature book = msg['body'].replace('@bot', '').replace('/book', '') book = re.sub(' +', ' ', book) # remove double spaces book = book.lstrip().rstrip() # remove spaces at the beginning and at the end book = book.replace(' ', '+').lower() # turn space into + and lowercase page_link = 'https://www.worldcat.org/search?q={}&qt=results_page'.format(book) page_response = requests.get(page_link, timeout=5) page_content = BeautifulSoup(page_response.content, "html.parser") try: book_title = page_content.findAll("div", {"class": "name"})[0].text book_author = page_content.findAll("div", {"class": "author"})[0].text book_publisher = page_content.findAll("div", {"class": "publisher"})[0].text response = 'BOOK: ' + book_title + ' ' + book_author + ' ' + book_publisher book_found = True except IndexError: book_found = False if book_found: # Add message to log message = 'BOOK: ' + book_title + ' ' + book_author + ' ' + book_publisher message = f'
{ message }
' write_to_log(self, message) self.send_message(mto=self.groupchat, mbody='Hope this was the book you were looking for: ' + book_title + ' ' + book_author + ' ' + book_publisher, mtype='groupchat') else: self.send_message(mto=self.groupchat, mbody='Sorry, no book found!', mtype='groupchat') # Generate HTML logfiles # By default: log if self.mode: mode = str(self.mode.lower().strip()) if mode == "log": print('> log.py') elif mode == "stream": print('> stream.py') elif mode == "distribusi": print('> distribusi.py') if __name__ == '__main__': # Setup the command line arguments. parser = ArgumentParser() # output verbosity options. parser.add_argument("-q", "--quiet", help="set logging to ERROR", action="store_const", dest="loglevel", const=logging.ERROR, default=logging.INFO) parser.add_argument("-d", "--debug", help="set logging to DEBUG", action="store_const", dest="loglevel", const=logging.DEBUG, default=logging.INFO) # Different options. parser.add_argument("-u", "--use", dest="use", help="XMPP address to use") parser.add_argument("-p", "--password", dest="password", help="password to use") parser.add_argument("-g", "--groupchat", dest="groupchat", help="groupchat to join") parser.add_argument("-n", "--nick", dest="nickname", help="nickname for the bot") parser.add_argument("-o", "--output", dest="output", help="output folder, this is where the files are stored", type=str) parser.add_argument("-m", "--mode", dest="mode", help="logmode, options include: log, stream, distribusi", type=str, default='log') args = parser.parse_args() # Setup logging. logging.basicConfig(level=args.loglevel, format='%(levelname)-8s %(message)s') if args.use is None: args.use = input("Use this XMPP address for the bot: ") if args.password is None: args.password = getpass("Password: ") if args.groupchat is None: args.groupchat = input("Groupchat XMPP address: ") if args.nickname is None: args.nickname = input("Nickname for the bot: ") if args.output is None: args.output = input("Output folder path of the log: ") # Setup the MUCBot and register plugins. Note that while plugins may # have interdependencies, the order in which you register them does # not matter. xmpp = MUCBot(args.use, args.password, args.groupchat, args.nickname, args.output, args.mode) xmpp.register_plugin('xep_0030') # Service Discovery xmpp.register_plugin('xep_0045') # Multi-User Chat xmpp.register_plugin('xep_0199') # XMPP Ping xmpp.register_plugin('xep_0066') # Process URI's (files, images) # Connect to the XMPP server and start processing XMPP stanzas. xmpp.connect() xmpp.process()