# HG changeset patch
# User Jeff Hammel <k0scist@gmail.com>
# Date 1435702692 25200
# Node ID a2d199008a837f12a9d37467448346c7b6c70d5c
# Parent  25622fb5906d54f2c0feab69bdb44896e9dced76# Parent  eec5b7abff2bd5fa4b7715f876d1f8ff1ce93578
wtf

diff -r 25622fb5906d -r a2d199008a83 .bashrc
--- a/.bashrc	Wed May 27 15:55:29 2015 -0700
+++ b/.bashrc	Tue Jun 30 15:18:12 2015 -0700
@@ -463,20 +463,6 @@
 
 ### functions for version control systems
 
-svndance(){
-# do the svn import dance!
-if (( $# ))
-then
-    svn import $1
-    cd ..
-    rm -rf $OLDPWD
-    svn co $1 $OLDPWD
-    cd $OLDPWD
-else
-    return 1
-fi
-}
-
 difffiles() {
     # which files are diffed; could use `lsdiff`
     grep '^+++ ' $@ | sed 's/+++ b\///'
@@ -492,12 +478,6 @@
     combinediff <(git diff) <(git diff --cached)
 }
 
-hg-add-commit() {
-    # add a file + commit
-    MESSAGE=$(hg add $@)
-    hg commit -m "${MESSAGE}"
-}
-
 hg-update-all() {
     # update all hg repositories in the current directory
     for i in *;
@@ -607,16 +587,7 @@
 ### include overrides for commands
 source ~/.bash_overrides
 
-### cognet customizations
-export BASTION_SERVER=bastion-east-1c.cognet.tv
-export BASTION_USER=jhammel
-export KEYS_DIR=/home/jhammel/.ssh
-COGNET_BASHRC=~/cognet/bin/.bashrc
-if [ -e "${COGNET_BASHRC}" ]
-then
-    . ${COGNET_BASHRC}
-fi
-
+### deactivate any virtualenvs
 if type deactivate &> /dev/null
 then
 deactivate
diff -r 25622fb5906d -r a2d199008a83 .fluxbox/keys
--- a/.fluxbox/keys	Wed May 27 15:55:29 2015 -0700
+++ b/.fluxbox/keys	Tue Jun 30 15:18:12 2015 -0700
@@ -78,6 +78,7 @@
 Control Mod1 v :ExecCommand xterm -e '/home/jhammel/k0s/bin/textshaper -c -i'
 Control Mod1 x :ExecCommand xkill
 Control Mod1 y :ExecCommand /home/jhammel/python/pypi.py "$(xclip -o)"| xclip -i # pypi package summary
+Control Mod1 z :ExecCommand xzoom
 Control Mod1 0 :ExecCommand xclip -o | sed 's/^[-+]//' | xclip -i # strip leading +s
 Control Shift / :ExecCommand /home/jhammel/bin/keyshelp.sh # hotkeys help
 
@@ -88,7 +89,6 @@
 # v: ?
 # w: ?
 # y: ?
-# z: zoom?
 
 # `l` could go to launch to free up `s`
 # alternately, `l` -> `local resource`,
diff -r 25622fb5906d -r a2d199008a83 bin/log.sh
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/bin/log.sh	Tue Jun 30 15:18:12 2015 -0700
@@ -0,0 +1,13 @@
+#!/bin/bash
+
+BASEDIR=${HOME}/cisco/log
+STAMP=$(date +"%Y%m%d")
+FILENAME="${BASEDIR}/${STAMP}.txt"
+
+if [[ -e "${FILENAME}" ]]
+then
+    echo "emacs -nw +`wc -l ${FILENAME}` ${FILENAME}"
+    emacs -nw +`wc -l "${FILENAME}"`
+else
+    emacs -nw ${FILENAME}
+fi
diff -r 25622fb5906d -r a2d199008a83 bin/tofile.sh
--- a/bin/tofile.sh	Wed May 27 15:55:29 2015 -0700
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,1 +0,0 @@
-/home/jhammel/docs/projects/ims/workflow/tofile.sh
\ No newline at end of file
diff -r 25622fb5906d -r a2d199008a83 python/example/sendmail.py
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/python/example/sendmail.py	Tue Jun 30 15:18:12 2015 -0700
@@ -0,0 +1,87 @@
+#!/usr/bin/env python
+# -*- coding: utf-8 -*-
+
+"""
+
+See:
+- http://pymotw.com/2/smtplib/
+- http://www.mkyong.com/python/how-do-send-email-in-python-via-smtplib/
+- https://github.com/CognitiveNetworks/daily-reports/blob/master/send_added_tv_count.py
+"""
+
+import argparse
+import email.utils
+import os
+import smtplib
+import sys
+from email.mime.text import MIMEText
+
+__all__ = ['main']
+
+class MailSender(object):
+
+    def __init__(self, host, sender, password, port=587, type='plain'):
+        self.host = host
+        self.sender = sender
+        self.password = password
+        self.port = port
+        self.type = type
+
+    def __call__(self, message, *recipients, **headers):
+
+        assert recipients
+
+        # construct the message
+        msg = MIMEText(message, self.type)
+        headers.setdefault('From', self.sender)
+        headers.setdefault('To', ','.join(recipients))
+        for key, value in headers.items():
+            msg[key] = value
+
+        # connect to mail server
+        server = smtplib.SMTP(self.host, self.port)
+        try:
+            server.set_debuglevel(True)
+
+            # identify ourselves, prompting server for supported features
+            server.ehlo()
+
+            # If we can encrypt this session, do it
+            if server.has_extn('STARTTLS'):
+                server.starttls()
+                server.ehlo() # re-identify ourselves over TLS connection
+
+            # login
+            server.login(self.sender, self.password)
+
+            # send the email
+            server.sendmail(self.sender, recipients, msg.as_string())
+        finally:
+            server.quit()
+
+def main(args=sys.argv[1:]):
+
+    # parse command line
+    parser = argparse.ArgumentParser(description=__doc__)
+    parser.add_argument('host')
+    parser.add_argument('sender')
+    parser.add_argument('password')
+    parser.add_argument('-r', '--recipients', dest='recipients',
+                        nargs='+', required=True,
+                        help="recipients")
+    parser.add_argument('-m', '--message', dest='message', required=True)
+    parser.add_argument('--port', dest='port', type=int, default=587,
+                        help="port to connect to [DEFAULT: %(default)s]")
+    options = parser.parse_args(args)
+
+    message = options.message
+
+    # instantiate sender
+    sender = MailSender(options.host, options.sender, options.password, options.port)
+
+    # send email
+    sender(message, *options.recipients)
+
+
+if __name__ == '__main__':
+    main()
diff -r 25622fb5906d -r a2d199008a83 python/example/slice.py
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/python/example/slice.py	Tue Jun 30 15:18:12 2015 -0700
@@ -0,0 +1,65 @@
+#!/usr/bin/env python
+# -*- coding: utf-8 -*-
+
+"""
+slice a arbitrarily sized list
+"""
+
+# imports
+import argparse
+import os
+import subprocess
+import sys
+
+__all__ = ['slice', 'main', 'Parser']
+
+def slice(container, n_chunks):
+    size = int(len(container)/(n_chunks-1))
+    retval = []
+    start = 0
+    for i in range(n_chunks-1):
+        retval.append(container[start:start+size])
+        start += size
+    retval.append(container[start:])
+    return retval
+
+class Parser(argparse.ArgumentParser):
+    """CLI option parser"""
+    def __init__(self, **kwargs):
+        kwargs.setdefault('description', __doc__)
+        argparse.ArgumentParser.__init__(self, **kwargs)
+        self.add_argument('N', type=int,
+                          help="number of chunks")
+        self.add_argument('-M', '--len', dest='length', type=int, default=29,
+                          help="length of list [DEFAULT: %(default)s]")
+        self.options = None
+
+    def parse_args(self, *args, **kw):
+        options = argparse.ArgumentParser.parse_args(self, *args, **kw)
+        self.validate(options)
+        self.options = options
+        return options
+
+    def validate(self, options):
+        """validate options"""
+
+def main(args=sys.argv[1:]):
+    """CLI"""
+
+    # parse command line options
+    parser = Parser()
+    options = parser.parse_args(args)
+
+    # generate list
+    seq = range(options.length)
+
+    # chunk list
+    output = slice(seq, options.N)
+
+    # print output
+    for chunk in output:
+        print (",".join([str(i) for i in chunk]))
+
+if __name__ == '__main__':
+    main()
+
diff -r 25622fb5906d -r a2d199008a83 python/ffslice.py
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/python/ffslice.py	Tue Jun 30 15:18:12 2015 -0700
@@ -0,0 +1,168 @@
+#!/usr/bin/env python
+# -*- coding: utf-8 -*-
+
+"""
+slice a clip with `ffmpeg`:
+
+ffmpeg -ss <00:00:00> -t <sec> -i <input_file> -q 1  <output_file> #
+
+--ss = start time ;
+
+-t = length of video to create from the start time in seconds.
+"""
+
+# imports
+import argparse
+import os
+import subprocess
+import sys
+
+# module globals
+__all__ = ['main', 'Parser']
+here = os.path.dirname(os.path.realpath(__file__))
+string = (str, unicode)
+
+
+def ensure_dir(directory):
+    """ensure a directory exists"""
+    if os.path.exists(directory):
+        assert os.path.isdir(directory)
+        return directory
+    os.makedirs(directory)
+    return directory
+
+
+def convert_seconds_to_hhmmss(time_sec):
+    """converts `time_sec` to (hh,mm,ss)"""
+
+    hh = int(time_sec / 3600)
+    mm = int((time_sec % 3600) / 60)
+    ss = int((time_sec % 3600) % 60)
+    return (hh,mm,ss)
+
+
+def format_seconds_to_hhmmss(time_sec, separator=':'):
+    """converts `time_sec` to string 'hh:mm:ss'"""
+    return separator.join(['{:0>2d}'.format(i)
+                           for i in convert_seconds_to_hhmmss(time_sec)])
+
+
+def duration(clip, ffprobe='ffprobe'):
+    """get the duration in seconds using `ffprobe` from ffmpeg"""
+
+    command = [ffprobe, clip]
+    assert os.path.exists(clip), "Missing clip, duration not available"
+
+    # probe the file
+    try:
+        output = subprocess.check_output(command, stderr=subprocess.STDOUT)
+        # (for some reason, ffprobe sends output to stderr)
+    except subprocess.CalledProcessError as e:
+        print (e.output)
+        raise
+
+    duration = None
+    for line in output.splitlines():
+        # Parse the output of `ffprobe`
+        # look for a line like:
+        #   Duration: 00:00:59.73, start: 0.000000, bitrate: 7783 kb/s
+
+        line = line.strip()
+        if line.startswith("Duration:"):
+            if duration:
+                raise AssertionError("Duplicate duration - already found: {}".format(duration))
+            line = line.split(',')[0]
+            duration = line.split(':', 1)[1].strip()
+
+    if duration:
+        hh, mm, ss = [float(i) for i in duration.split(':')]
+        duration = 3600*hh + 60*mm + ss
+
+        return duration
+
+
+class FFSliceParser(argparse.ArgumentParser):
+    """fflice CLI option parser"""
+
+    default_slice_time = 300.
+
+    def __init__(self):
+        argparse.ArgumentParser.__init__(self, description=__doc__)
+        self.add_argument('clips', metavar='clip', nargs='+', help="clips to slice")
+        self.add_argument('-d', '--directory', dest='directory',
+                          default=os.getcwd(),
+                          help="output directory [DEFAULT: %(default)s]")
+        self.add_argument('--durations', '--print-durations', dest='print_durations',
+                          action='store_true', default=False,
+                          help="print durations and exit")
+        self.add_argument('-n', dest='number', type=int,
+                          help="number of slices")
+        self.add_argument('-t', '--time', dest='slice_time',
+                          type=float,
+                          help="time of each slice [DEFAULT: {}]".format(self.default_slice_time))
+        self.add_argument('--dry-run', dest='dry_run',
+                          action='store_true', default=False,
+                          help="print out what will be done")
+        self.add_argument('--hhmmss', dest='hhmmss',
+                          action='store_true', default=False,
+                          help="display times in 'hh:mm:ss' format; thedefault is in seconds")
+
+        self.add_argument('-v', '--verbose', dest='verbose',
+                          action='store_true', default=False,
+                          help="be verbose")
+        self.options = None
+
+    def parse_args(self, *args, **kw):
+        options = argparse.ArgumentParser.parse_args(self, *args, **kw)
+        self.validate(options)
+        self.options = options
+        return options
+
+    def validate(self, options):
+        """validate options"""
+
+        missing = [clip for clip in options.clips
+                   if not os.path.exists(clip)]
+        if missing:
+            self.error("Not found: {}".format(', '.join(missing)))
+
+        if options.slice_time and options.number:
+            self.error("Cannot specify slice time and number of slices")
+            # TODO: allow specification of both
+        elif options.slice_time is None and options.number is None:
+            options.slice_time = self.default_slice_time
+
+        ensure_dir(options.directory)
+
+    def format_seconds(self, seconds):
+        """format seconds to string"""
+        if self.options.hhmmss:
+            return format_seconds_to_hhmmss(seconds)
+        return '{:.2}'.format(seconds)
+
+
+def main(args=sys.argv[1:]):
+    """CLI"""
+
+    # parse command line options
+    parser = FFSliceParser()
+    options = parser.parse_args(args)
+
+    # compute durations
+    durations = {clip: duration(clip) for clip in options.clips}
+    if options.print_durations:
+        returncode = 0
+        total = 0.
+        for clip in options.clips:
+            _duration = durations[clip]
+            if _duration is None:
+                print ("Duration not found: '{}'".format(clip))
+                returncode = 1
+            else:
+                print ('{} : {}'.format(clip,
+                                        parser.format_seconds(_duration)))
+        sys.exit(returncode)
+
+if __name__ == '__main__':
+    main()
+
diff -r 25622fb5906d -r a2d199008a83 python/slice.py
--- a/python/slice.py	Wed May 27 15:55:29 2015 -0700
+++ /dev/null	Thu Jan 01 00:00:00 1970 +0000
@@ -1,65 +0,0 @@
-#!/usr/bin/env python
-# -*- coding: utf-8 -*-
-
-"""
-slice a arbitrarily sized list
-"""
-
-# imports
-import argparse
-import os
-import subprocess
-import sys
-
-__all__ = ['slice', 'main', 'Parser']
-
-def slice(container, n_chunks):
-    size = int(len(container)/(n_chunks-1))
-    retval = []
-    start = 0
-    for i in range(n_chunks-1):
-        retval.append(container[start:start+size])
-        start += size
-    retval.append(container[start:])
-    return retval
-
-class Parser(argparse.ArgumentParser):
-    """CLI option parser"""
-    def __init__(self, **kwargs):
-        kwargs.setdefault('description', __doc__)
-        argparse.ArgumentParser.__init__(self, **kwargs)
-        self.add_argument('N', type=int,
-                          help="number of chunks")
-        self.add_argument('-M', '--len', dest='length', type=int, default=29,
-                          help="length of list [DEFAULT: %(default)s]")
-        self.options = None
-
-    def parse_args(self, *args, **kw):
-        options = argparse.ArgumentParser.parse_args(self, *args, **kw)
-        self.validate(options)
-        self.options = options
-        return options
-
-    def validate(self, options):
-        """validate options"""
-
-def main(args=sys.argv[1:]):
-    """CLI"""
-
-    # parse command line options
-    parser = Parser()
-    options = parser.parse_args(args)
-
-    # generate list
-    seq = range(options.length)
-
-    # chunk list
-    output = slice(seq, options.N)
-
-    # print output
-    for chunk in output:
-        print (",".join([str(i) for i in chunk]))
-
-if __name__ == '__main__':
-    main()
-