#!/usr/bin/env python ### # Copyright (c) 2002, Jeremiah Fincher # All rights reserved. # # Redistribution and use in source and binary forms, with or without # modification, are permitted provided that the following conditions are met: # # * Redistributions of source code must retain the above copyright notice, # this list of conditions, and the following disclaimer. # * Redistributions in binary form must reproduce the above copyright notice, # this list of conditions, and the following disclaimer in the # documentation and/or other materials provided with the distribution. # * Neither the name of the author of this software nor the name of # contributors to this software may be used to endorse or promote products # derived from this software without specific prior written consent. # # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" # AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE # IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE # ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE # LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR # CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF # SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS # INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN # CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) # ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE # POSSIBILITY OF SUCH DAMAGE. ### """ Various math-related commands. """ from baseplugin import * import re import new import math import cmath from itertools import imap import utils import privmsgs import callbacks def configure(onStart, afterConnect, advanced): # This will be called by setup.py to configure this module. onStart and # afterConnect are both lists. Append to onStart the commands you would # like to be run when the bot is started; append to afterConnect the # commands you would like to be run when the bot has finished connecting. from questions import expect, anything, something, yn onStart.append('load Math') example = utils.wrapLines(""" @calc e**(i*pi) + 1 0 @calc 1+2+3 6 @rpn 1 2 3 + + 6 """) class Math(callbacks.Privmsg): ### # So this is how the 'calc' command works: # First, we make a nice little safe environment for evaluation; basically, # the names in the 'math' and 'cmath' modules. Then, we remove the ability # of a random user to get ints evaluated: this means we have to turn all # int literals (even octal numbers and hexadecimal numbers) into floats. # Then we delete all square brackets, underscores, and whitespace, so no # one can do list comprehensions or call __...__ functions. ### _mathEnv = {'__builtins__': new.module('__builtins__'), 'i': 1j} _mathEnv.update(math.__dict__) _mathEnv.update(cmath.__dict__) _mathRe = re.compile(r'((?:(? 0: imag = '+%si' % imag if real == 0: return imag.lstrip('+') else: return '%s%s' % (real, imag) def calc(self, irc, msg, args): """ Returns the value of the evaluted . The syntax is Python syntax; the type of arithmetic is floating point. """ text = privmsgs.getArgs(args) text = text.translate(string.ascii, '_[] \t') text = text.replace('lambda', '') def handleMatch(m): s = m.group(1) if s.startswith('0x'): i = int(s, 16) elif s.startswith('0') and '.' not in s: try: i = int(s, 8) except ValueError: i = int(s) else: i = float(s) return str(complex(i)) text = self._mathRe.sub(handleMatch, text) try: x = complex(eval(text, self._mathEnv, self._mathEnv)) irc.reply(msg, self._complexToString(x)) except OverflowError: irc.error(msg, 'Go get scanez, this is a *real* math problem!') except TypeError: irc.error(msg, 'Something in there wasn\'t a valid number.') except Exception, e: irc.error(msg, debug.exnToString(e)) _rpnEnv = { 'dup': lambda s: s.extend([s.pop()]*2), } def rpn(self, irc, msg, args): """ Returns the value of an RPN expression. """ stack = [] for arg in args: try: stack.append(complex(arg)) except ValueError: # Not a float. if arg in self._mathEnv: f = self._mathEnv[arg] if callable(f): called = False arguments = [] while not called and stack: arguments.append(stack.pop()) try: stack.append(f(*arguments)) called = True except TypeError: pass if not called: irc.error(msg, 'Not enough arguments for %s' % arg) return else: stack.append(f) elif arg in self._rpnEnv: self._rpnEnv[arg](stack) else: arg2 = stack.pop() arg1 = stack.pop() s = '%s%s%s' % (arg1, arg, arg2) stack.append(eval(s, self._mathEnv, self._mathEnv)) if len(stack) == 1: irc.reply(msg, str(self._complexToString(complex(stack[0])))) else: s = ', '.join(imap(self._complexToString, imap(complex, stack))) irc.reply(msg, 'Stack: [%s]' % s) Class = Math # vim:set shiftwidth=4 tabstop=8 expandtab textwidth=78: