2011-09-06 00:45:53 +00:00
|
|
|
# Copyright 2010-2011 Florent Le Coz <louiz@louiz.org>
|
|
|
|
#
|
|
|
|
# This file is part of Poezio.
|
|
|
|
#
|
|
|
|
# Poezio is free software: you can redistribute it and/or modify
|
2011-09-11 15:10:05 +00:00
|
|
|
# it under the terms of the zlib license. See the COPYING file.
|
2010-02-13 15:18:39 +00:00
|
|
|
|
2010-05-18 13:29:02 +00:00
|
|
|
"""
|
2013-03-08 18:39:34 +00:00
|
|
|
Various useful functions.
|
2010-05-18 13:29:02 +00:00
|
|
|
"""
|
2010-07-20 17:25:41 +00:00
|
|
|
|
2010-07-14 18:26:01 +00:00
|
|
|
from datetime import datetime, timedelta
|
2012-08-06 13:38:09 +00:00
|
|
|
from sleekxmpp import JID, InvalidJID
|
2010-02-13 15:18:39 +00:00
|
|
|
import base64
|
|
|
|
import os
|
|
|
|
import mimetypes
|
|
|
|
import hashlib
|
2010-02-14 03:51:03 +00:00
|
|
|
import subprocess
|
2010-07-14 18:26:01 +00:00
|
|
|
import time
|
2011-11-15 20:21:01 +00:00
|
|
|
import string
|
2011-02-10 11:39:15 +00:00
|
|
|
import shlex
|
2010-03-18 19:43:44 +00:00
|
|
|
|
2010-08-04 00:17:53 +00:00
|
|
|
ROOM_STATE_NONE = 11
|
|
|
|
ROOM_STATE_CURRENT = 10
|
|
|
|
ROOM_STATE_PRIVATE = 15
|
|
|
|
ROOM_STATE_MESSAGE = 12
|
|
|
|
ROOM_STATE_HL = 13
|
|
|
|
|
2010-02-13 15:18:39 +00:00
|
|
|
def get_base64_from_file(path):
|
2010-05-18 13:29:02 +00:00
|
|
|
"""
|
|
|
|
Convert the content of a file to base64
|
2013-03-08 18:39:34 +00:00
|
|
|
|
|
|
|
:param str path: The path of the file to convert.
|
|
|
|
:return: A tuple of (encoded data, mime type, sha1 hash) if
|
|
|
|
the file exists and does not exceeds the upper size limit of 16384.
|
|
|
|
:return: (None, None, error message) if it fails
|
|
|
|
:rtype: :py:class:`tuple`
|
|
|
|
|
2010-05-18 13:29:02 +00:00
|
|
|
"""
|
2010-02-13 15:18:39 +00:00
|
|
|
if not os.path.isfile(path):
|
|
|
|
return (None, None, "File does not exist")
|
|
|
|
size = os.path.getsize(path)
|
|
|
|
if size > 16384:
|
|
|
|
return (None, None,"File is too big")
|
2010-05-18 13:29:02 +00:00
|
|
|
fdes = open(path, 'rb')
|
|
|
|
data = fdes.read()
|
2010-02-13 15:18:39 +00:00
|
|
|
encoded = base64.encodestring(data)
|
|
|
|
sha1 = hashlib.sha1(data).hexdigest()
|
|
|
|
mime_type = mimetypes.guess_type(path)[0]
|
|
|
|
return (encoded, mime_type, sha1)
|
2010-02-14 03:51:03 +00:00
|
|
|
|
|
|
|
def get_output_of_command(command):
|
2010-05-18 13:29:02 +00:00
|
|
|
"""
|
2013-03-08 18:39:34 +00:00
|
|
|
Runs a command and returns its output.
|
|
|
|
|
|
|
|
:param str command: The command to run.
|
|
|
|
:return: The output or None
|
|
|
|
:rtype: :py:class:`str`
|
2010-05-18 13:29:02 +00:00
|
|
|
"""
|
2010-02-14 03:51:03 +00:00
|
|
|
try:
|
2011-01-22 14:03:54 +00:00
|
|
|
return subprocess.check_output(command.split()).decode('utf-8').split('\n')
|
|
|
|
except subprocess.CalledProcessError:
|
2010-02-14 03:51:03 +00:00
|
|
|
return None
|
|
|
|
|
|
|
|
def is_in_path(command, return_abs_path=False):
|
|
|
|
"""
|
2013-03-08 18:39:34 +00:00
|
|
|
Check if *command* is in the $PATH or not.
|
|
|
|
|
|
|
|
:param str command: The command to be checked.
|
|
|
|
:param bool return_abs_path: Return the absolute path of the command instead
|
|
|
|
of True if the command is found.
|
|
|
|
:return: True if the command is found, the command path if the command is found
|
|
|
|
and *return_abs_path* is True, otherwise False.
|
|
|
|
|
2010-02-14 03:51:03 +00:00
|
|
|
"""
|
|
|
|
for directory in os.getenv('PATH').split(os.pathsep):
|
|
|
|
try:
|
|
|
|
if command in os.listdir(directory):
|
|
|
|
if return_abs_path:
|
|
|
|
return os.path.join(directory, command)
|
|
|
|
else:
|
|
|
|
return True
|
|
|
|
except OSError:
|
|
|
|
# If the user has non directories in his path
|
|
|
|
pass
|
|
|
|
return False
|
|
|
|
|
2010-05-18 13:29:02 +00:00
|
|
|
DISTRO_INFO = {
|
2010-02-14 03:51:03 +00:00
|
|
|
'Arch Linux': '/etc/arch-release',
|
|
|
|
'Aurox Linux': '/etc/aurox-release',
|
|
|
|
'Conectiva Linux': '/etc/conectiva-release',
|
|
|
|
'CRUX': '/usr/bin/crux',
|
|
|
|
'Debian GNU/Linux': '/etc/debian_version',
|
|
|
|
'Fedora Linux': '/etc/fedora-release',
|
|
|
|
'Gentoo Linux': '/etc/gentoo-release',
|
|
|
|
'Linux from Scratch': '/etc/lfs-release',
|
|
|
|
'Mandrake Linux': '/etc/mandrake-release',
|
|
|
|
'Slackware Linux': '/etc/slackware-version',
|
|
|
|
'Solaris/Sparc': '/etc/release',
|
|
|
|
'Source Mage': '/etc/sourcemage_version',
|
|
|
|
'SUSE Linux': '/etc/SuSE-release',
|
|
|
|
'Sun JDS': '/etc/sun-release',
|
|
|
|
'PLD Linux': '/etc/pld-release',
|
|
|
|
'Yellow Dog Linux': '/etc/yellowdog-release',
|
|
|
|
# many distros use the /etc/redhat-release for compatibility
|
|
|
|
# so Redhat is the last
|
|
|
|
'Redhat Linux': '/etc/redhat-release'
|
|
|
|
}
|
2010-05-18 13:29:02 +00:00
|
|
|
|
2010-02-14 03:51:03 +00:00
|
|
|
def get_os_info():
|
2010-05-18 13:29:02 +00:00
|
|
|
"""
|
|
|
|
Returns a detailed and well formated string containing
|
|
|
|
informations about the operating system
|
2013-03-08 18:39:34 +00:00
|
|
|
|
|
|
|
:rtype: str
|
2010-05-18 13:29:02 +00:00
|
|
|
"""
|
|
|
|
if os.name == 'posix':
|
2010-02-14 03:51:03 +00:00
|
|
|
executable = 'lsb_release'
|
|
|
|
params = ' --description --codename --release --short'
|
|
|
|
full_path_to_executable = is_in_path(executable, return_abs_path = True)
|
|
|
|
if full_path_to_executable:
|
|
|
|
command = executable + params
|
2010-05-18 13:29:02 +00:00
|
|
|
process = subprocess.Popen([command], shell=True,
|
|
|
|
stdin=subprocess.PIPE,
|
|
|
|
stdout=subprocess.PIPE,
|
|
|
|
close_fds=True)
|
|
|
|
process.wait()
|
2011-01-09 00:09:51 +00:00
|
|
|
output = process.stdout.readline().decode('utf-8').strip()
|
2010-02-14 03:51:03 +00:00
|
|
|
# some distros put n/a in places, so remove those
|
|
|
|
output = output.replace('n/a', '').replace('N/A', '')
|
|
|
|
return output
|
|
|
|
|
|
|
|
# lsb_release executable not available, so parse files
|
2010-05-18 13:29:02 +00:00
|
|
|
for distro_name in DISTRO_INFO:
|
|
|
|
path_to_file = DISTRO_INFO[distro_name]
|
2010-02-14 03:51:03 +00:00
|
|
|
if os.path.exists(path_to_file):
|
|
|
|
if os.access(path_to_file, os.X_OK):
|
|
|
|
# the file is executable (f.e. CRUX)
|
|
|
|
# yes, then run it and get the first line of output.
|
|
|
|
text = get_output_of_command(path_to_file)[0]
|
|
|
|
else:
|
2012-09-13 07:48:35 +00:00
|
|
|
fdes = open(path_to_file, encoding='utf-8')
|
2010-05-18 13:29:02 +00:00
|
|
|
text = fdes.readline().strip() # get only first line
|
|
|
|
fdes.close()
|
2010-02-14 03:51:03 +00:00
|
|
|
if path_to_file.endswith('version'):
|
|
|
|
# sourcemage_version and slackware-version files
|
|
|
|
# have all the info we need (name and version of distro)
|
|
|
|
if not os.path.basename(path_to_file).startswith(
|
|
|
|
'sourcemage') or not\
|
|
|
|
os.path.basename(path_to_file).startswith('slackware'):
|
|
|
|
text = distro_name + ' ' + text
|
|
|
|
elif path_to_file.endswith('aurox-release') or \
|
|
|
|
path_to_file.endswith('arch-release'):
|
|
|
|
# file doesn't have version
|
|
|
|
text = distro_name
|
2010-05-18 13:29:02 +00:00
|
|
|
elif path_to_file.endswith('lfs-release'):
|
|
|
|
# file just has version
|
2010-02-14 03:51:03 +00:00
|
|
|
text = distro_name + ' ' + text
|
|
|
|
os_info = text.replace('\n', '')
|
|
|
|
return os_info
|
|
|
|
|
|
|
|
# our last chance, ask uname and strip it
|
|
|
|
uname_output = get_output_of_command('uname -sr')
|
|
|
|
if uname_output is not None:
|
|
|
|
os_info = uname_output[0] # only first line
|
|
|
|
return os_info
|
|
|
|
os_info = 'N/A'
|
|
|
|
return os_info
|
2010-03-19 03:30:51 +00:00
|
|
|
|
|
|
|
def datetime_tuple(timestamp):
|
|
|
|
"""
|
2013-03-08 18:39:34 +00:00
|
|
|
Convert a timestamp using strptime and the format: %Y%m%dT%H:%M:%S.
|
2010-03-19 03:30:51 +00:00
|
|
|
|
2013-03-08 18:39:34 +00:00
|
|
|
Because various datetime formats are used, the following exceptions
|
2010-03-19 03:30:51 +00:00
|
|
|
are handled:
|
2013-03-08 18:39:34 +00:00
|
|
|
|
|
|
|
* Optional milliseconds appened to the string are removed
|
|
|
|
* Optional Z (that means UTC) appened to the string are removed
|
|
|
|
* XEP-082 datetime strings have all '-' chars removed to meet the above format.
|
|
|
|
|
|
|
|
:param str timestamp: The string containing the formatted date.
|
|
|
|
:return: The date.
|
|
|
|
:rtype: :py:class:`datetime.datetime`
|
|
|
|
|
2013-08-01 18:17:12 +00:00
|
|
|
>>> datetime_tuple('20130226T06:23:12')
|
2013-03-08 18:39:34 +00:00
|
|
|
datetime.datetime(2013, 2, 26, 8, 23, 12)
|
2010-03-19 03:30:51 +00:00
|
|
|
"""
|
|
|
|
timestamp = timestamp.split('.')[0]
|
|
|
|
timestamp = timestamp.replace('-', '')
|
2010-07-26 00:51:07 +00:00
|
|
|
try:
|
|
|
|
ret = datetime.strptime(timestamp, '%Y%m%dT%H:%M:%SZ')
|
|
|
|
except ValueError: # Support the deprecated format, XEP 0091 :(
|
|
|
|
ret = datetime.strptime(timestamp, '%Y%m%dT%H:%M:%S')
|
2010-07-14 18:26:01 +00:00
|
|
|
# convert UTC to local time, with DST etc.
|
|
|
|
dst = timedelta(seconds=time.altzone)
|
|
|
|
ret -= dst
|
|
|
|
return ret
|
2011-02-10 11:39:15 +00:00
|
|
|
|
2013-03-31 14:33:10 +00:00
|
|
|
def find_delayed_tag(message):
|
|
|
|
"""
|
|
|
|
Check if a message is delayed or not.
|
|
|
|
|
|
|
|
:param sleekxmpp.Message message: The message to check.
|
|
|
|
:return: A tuple containing (True, the datetime) or (False, None)
|
|
|
|
:rtype: :py:class:`tuple`
|
|
|
|
"""
|
|
|
|
|
|
|
|
delay_tag = message.find('{urn:xmpp:delay}delay')
|
|
|
|
if delay_tag is not None:
|
|
|
|
delayed = True
|
|
|
|
date = datetime_tuple(delay_tag.attrib['stamp'])
|
|
|
|
else:
|
|
|
|
# We support the OLD and deprecated XEP: http://xmpp.org/extensions/xep-0091.html
|
|
|
|
# But it sucks, please, Jabber servers, don't do this :(
|
|
|
|
delay_tag = message.find('{jabber:x:delay}x')
|
|
|
|
if delay_tag is not None:
|
|
|
|
delayed = True
|
2013-05-24 19:52:12 +00:00
|
|
|
date = datetime_tuple(delay_tag.attrib['stamp'])
|
2013-03-31 14:33:10 +00:00
|
|
|
else:
|
|
|
|
delayed = False
|
|
|
|
date = None
|
|
|
|
return (delayed, date)
|
|
|
|
|
2011-05-19 18:38:12 +00:00
|
|
|
def shell_split(st):
|
2013-03-08 18:39:34 +00:00
|
|
|
"""
|
|
|
|
Split a string correctly according to the quotes
|
|
|
|
around the elements.
|
|
|
|
|
|
|
|
:param str st: The string to split.
|
|
|
|
:return: A list of the different of the string.
|
|
|
|
:rtype: :py:class:`list`
|
|
|
|
|
|
|
|
>>> shell_split('"sdf 1" "toto 2"')
|
|
|
|
['sdf 1', 'toto 2']
|
2013-08-05 17:45:08 +00:00
|
|
|
>>> shell_split('toto "titi"')
|
|
|
|
['toto', 'titi']
|
|
|
|
>>> shell_split('toto ""')
|
|
|
|
['toto', '']
|
|
|
|
>>> shell_split('"toto titi" toto ""')
|
|
|
|
['toto titi', 'toto', '']
|
2013-03-08 18:39:34 +00:00
|
|
|
"""
|
2013-08-01 18:17:12 +00:00
|
|
|
sh = shlex.shlex(st)
|
|
|
|
ret = []
|
|
|
|
w = sh.get_token()
|
|
|
|
while w and w[2] is not None:
|
|
|
|
ret.append(w[2])
|
2011-02-10 11:39:15 +00:00
|
|
|
w = sh.get_token()
|
2013-08-01 18:17:12 +00:00
|
|
|
return ret
|
|
|
|
|
|
|
|
def find_argument(pos, text, quoted=True):
|
|
|
|
"""
|
|
|
|
Split an input into a list of arguments, return the number of the
|
|
|
|
argument selected by pos.
|
|
|
|
|
|
|
|
If the position searched is outside the string, or in a space between words,
|
|
|
|
then it will return the position of an hypothetical new argument.
|
|
|
|
|
|
|
|
See the doctests of the two methods for example behaviors.
|
|
|
|
|
|
|
|
:param int pos: The position to search.
|
|
|
|
:param str text: The text to analyze.
|
|
|
|
:param quoted: Whether to take quotes into account or not.
|
|
|
|
:rtype: int
|
|
|
|
"""
|
|
|
|
if quoted:
|
|
|
|
return find_argument_quoted(pos, text)
|
|
|
|
else:
|
|
|
|
return find_argument_unquoted(pos, text)
|
|
|
|
|
|
|
|
def find_argument_quoted(pos, text):
|
|
|
|
"""
|
|
|
|
>>> find_argument_quoted(4, 'toto titi tata')
|
|
|
|
3
|
|
|
|
>>> find_argument_quoted(4, '"toto titi" tata')
|
|
|
|
0
|
|
|
|
>>> find_argument_quoted(8, '"toto" "titi tata"')
|
|
|
|
1
|
|
|
|
>>> find_argument_quoted(8, '"toto" "titi tata')
|
|
|
|
1
|
|
|
|
>>> find_argument_quoted(18, '"toto" "titi tata" ')
|
|
|
|
2
|
|
|
|
"""
|
|
|
|
sh = shlex.shlex(text)
|
|
|
|
count = -1
|
|
|
|
w = sh.get_token()
|
|
|
|
while w and w[2] is not None:
|
|
|
|
count += 1
|
|
|
|
if w[0] <= pos < w[1]:
|
|
|
|
return count
|
|
|
|
w = sh.get_token()
|
|
|
|
|
|
|
|
return count + 1
|
|
|
|
|
|
|
|
def find_argument_unquoted(pos, text):
|
|
|
|
"""
|
|
|
|
>>> find_argument_unquoted(2, 'toto titi tata')
|
|
|
|
0
|
|
|
|
>>> find_argument_unquoted(3, 'toto titi tata')
|
|
|
|
0
|
|
|
|
>>> find_argument_unquoted(6, 'toto titi tata')
|
|
|
|
1
|
|
|
|
>>> find_argument_unquoted(4, 'toto titi tata')
|
|
|
|
3
|
|
|
|
>>> find_argument_unquoted(25, 'toto titi tata')
|
|
|
|
3
|
|
|
|
"""
|
|
|
|
ret = text.split()
|
|
|
|
search = 0
|
|
|
|
argnum = 0
|
|
|
|
for i, elem in enumerate(ret):
|
|
|
|
elem_start = text.find(elem, search)
|
|
|
|
elem_end = elem_start + len(elem)
|
|
|
|
search = elem_end
|
|
|
|
if elem_start <= pos < elem_end:
|
|
|
|
return i
|
|
|
|
argnum = i
|
|
|
|
return argnum + 1
|
2011-02-13 21:26:05 +00:00
|
|
|
|
2011-11-15 20:21:01 +00:00
|
|
|
def parse_str_to_secs(duration=''):
|
2011-11-18 22:34:38 +00:00
|
|
|
"""
|
2013-03-08 18:39:34 +00:00
|
|
|
Parse a string of with a number of d, h, m, s.
|
|
|
|
|
|
|
|
:param str duration: The formatted string.
|
|
|
|
:return: The number of seconds represented by the string
|
|
|
|
:rtype: :py:class:`int`
|
|
|
|
|
2011-11-18 22:34:38 +00:00
|
|
|
>>> parse_str_to_secs("1d3m1h")
|
|
|
|
90180
|
|
|
|
"""
|
2011-11-15 20:37:04 +00:00
|
|
|
values = {'s': 1, 'm': 60, 'h': 3600, 'd': 86400}
|
2011-11-15 20:21:01 +00:00
|
|
|
result = 0
|
|
|
|
tmp = '0'
|
|
|
|
for char in duration:
|
|
|
|
if char in string.digits:
|
|
|
|
tmp += char
|
|
|
|
elif char in values:
|
|
|
|
tmp_i = int(tmp)
|
|
|
|
result += tmp_i * values[char]
|
|
|
|
tmp = '0'
|
|
|
|
else:
|
2012-07-30 17:05:20 +00:00
|
|
|
return 0
|
2011-11-15 20:25:28 +00:00
|
|
|
if tmp != '0':
|
|
|
|
result += int(tmp)
|
2011-11-15 20:21:01 +00:00
|
|
|
return result
|
|
|
|
|
|
|
|
def parse_secs_to_str(duration=0):
|
2011-11-18 22:34:38 +00:00
|
|
|
"""
|
2013-03-08 18:39:34 +00:00
|
|
|
Do the reverse operation of :py:func:`parse_str_to_secs`.
|
|
|
|
|
2011-11-18 22:34:38 +00:00
|
|
|
Parse a number of seconds to a human-readable string.
|
|
|
|
The string has the form XdXhXmXs. 0 units are removed.
|
2013-03-08 18:39:34 +00:00
|
|
|
|
|
|
|
:param int duration: The duration, in seconds.
|
|
|
|
:return: A formatted string containing the duration.
|
|
|
|
:rtype: :py:class:`str`
|
|
|
|
|
2011-11-18 22:34:38 +00:00
|
|
|
>>> parse_secs_to_str(3601)
|
2013-08-01 18:17:12 +00:00
|
|
|
'1h1s'
|
2011-11-18 22:34:38 +00:00
|
|
|
"""
|
2011-11-15 20:21:01 +00:00
|
|
|
secs, mins, hours, days = 0, 0, 0, 0
|
|
|
|
result = ''
|
|
|
|
secs = duration % 60
|
|
|
|
mins = (duration % 3600) // 60
|
|
|
|
hours = (duration % 86400) // 3600
|
|
|
|
days = duration // 86400
|
|
|
|
|
|
|
|
result += '%sd' % days if days else ''
|
|
|
|
result += '%sh' % hours if hours else ''
|
|
|
|
result += '%sm' % mins if mins else ''
|
|
|
|
result += '%ss' % secs if secs else ''
|
2012-08-02 14:24:10 +00:00
|
|
|
if not result:
|
|
|
|
result = '0s'
|
2011-11-15 20:21:01 +00:00
|
|
|
return result
|
|
|
|
|
2013-03-11 01:04:20 +00:00
|
|
|
def format_tune_string(infos):
|
|
|
|
"""
|
|
|
|
Contruct a string from a dict created from an "User tune" event.
|
|
|
|
|
|
|
|
:param dict infos: The informations
|
|
|
|
:return: The formatted string
|
|
|
|
:rtype: :py:class:`str`
|
|
|
|
"""
|
|
|
|
elems = []
|
|
|
|
track = infos.get('track')
|
|
|
|
if track:
|
|
|
|
elems.append(track)
|
|
|
|
title = infos.get('title')
|
|
|
|
if title:
|
|
|
|
elems.append(title)
|
|
|
|
else:
|
|
|
|
elems.append('Unknown title')
|
|
|
|
elems.append('-')
|
|
|
|
artist = infos.get('artist')
|
|
|
|
if artist:
|
|
|
|
elems.append(artist)
|
|
|
|
else:
|
|
|
|
elems.append('Unknown artist')
|
|
|
|
|
|
|
|
rating = infos.get('rating')
|
|
|
|
if rating:
|
|
|
|
elems.append('[ ' + rating + '/10' + ' ]')
|
|
|
|
length = infos.get('length')
|
|
|
|
if length:
|
|
|
|
length = int(length)
|
|
|
|
secs = length % 60
|
|
|
|
mins = length // 60
|
|
|
|
secs = str(secs).zfill(2)
|
|
|
|
mins = str(mins).zfill(2)
|
|
|
|
elems.append('[' + mins + ':' + secs + ']')
|
|
|
|
return ' '.join(elems)
|
|
|
|
|
2013-03-12 15:12:47 +00:00
|
|
|
def format_gaming_string(infos):
|
|
|
|
"""
|
|
|
|
Construct a string from a dict containing the "user gaming"
|
|
|
|
informations.
|
|
|
|
(for now, only use address and name)
|
|
|
|
|
|
|
|
:param dict infos: The informations
|
|
|
|
:returns: The formatted string
|
|
|
|
:rtype: :py:class:`str`
|
|
|
|
"""
|
|
|
|
name = infos.get('name')
|
|
|
|
if not name:
|
|
|
|
return ''
|
|
|
|
|
|
|
|
server_address = infos.get('server_address')
|
|
|
|
if server_address:
|
|
|
|
return '%s on %s' % (name, server_address)
|
|
|
|
return name
|
|
|
|
|
2013-03-08 18:39:34 +00:00
|
|
|
def safeJID(*args, **kwargs):
|
2011-11-18 22:35:11 +00:00
|
|
|
"""
|
2013-03-08 18:39:34 +00:00
|
|
|
Construct a :py:class:`sleekxmpp.JID` object from a string.
|
2012-08-06 13:38:09 +00:00
|
|
|
|
2013-03-08 18:39:34 +00:00
|
|
|
Used to avoid tracebacks during is stringprep fails
|
|
|
|
(fall back to a JID with an empty string).
|
|
|
|
"""
|
2012-08-06 13:38:09 +00:00
|
|
|
try:
|
|
|
|
return JID(*args, **kwargs)
|
|
|
|
except InvalidJID:
|
|
|
|
return JID('')
|
2013-08-01 18:17:12 +00:00
|
|
|
|
|
|
|
|
|
|
|
if __name__ == "__main__":
|
|
|
|
import doctest
|
|
|
|
doctest.testmod()
|