Monday, October 10, 2011

Tunneling UDP (e.g. Photoshop keyserver) over SSH

It's basically the same as this solution on Vincent Dumouchel's site, but wrapped up in a shell script that will run on a Mac. The theory is to use 'nc' to listen for UDP packets on localhost and forward them through a TCP connection (tunneled by SSH) to the remote host, where 'nc' will then read them off TCP and send them via UDP. Useful in a number of obscure situations.

#!/bin/sh
#
# file:        remote-key.sh
# description: forward UDP traffic to port 19283 (Adobe keysvr) over SSH
#
# Peter Desnoyers, Northeastern University, 2011
#

host=$1
keyserver=$2
port=19283      # Adobe keysrvr

# get rid of the annoying "Terminated" message on shutdown
exec 2<&-   

# lots of gross delays, because (a) we need to start the local 'nc'
# first, so it can be in the background, even though the remote isn't
# there yet to connect to, and (b) because we can't open the FIFO for
# reading until it's been opened for writing, or we'll get an EOF
#
(sleep 3; rm -f /tmp/fifo; mkfifo /tmp/fifo;
    (sleep 1; nc -l -u $port < /tmp/fifo) | nc localhost 6667 > /tmp/fifo) &
local=$!
trap "kill $local; rm -f /tmp/fifo" 0

# we keep the SSH in the foreground so that the remote end can detect
# a broken connection ('cat' will see EOF) and then clean up
# properly. 
#
ssh -L6667:localhost:6667 $host \
    'trap "rm -f /tmp/fifo; killall nc" 1; 
        rm -f /tmp/fifo; killall -q nc; mkfifo /tmp/fifo;
        ((sleep 1; nc -l -p 6667 < /tmp/fifo) | 
  nc -u '$keyserver' '$port' > /tmp/fifo) < /dev/null &
        cat > /dev/null;
        killall nc' 

Thursday, September 15, 2011

An alternative to strtok(3) in C

If you've ever tried to split strings in C you know that strtok() is an abomination, modifying the string passed to it and in general breaking most of the standard patterns for how memory is allocated and handled in reasonable C programs. The obvious solution is to use another programming language, but sometimes that isn't possible. Here's an alternative function strwrd that I give my students for a homework that involves command line parsing, which follows the standard practice of returning strings in caller-allocated storage.
/* find the next word starting at 's', delimited by characters
 * in the string 'delim', and store up to 'len' bytes into *buf
 * returns pointer to immediately after the word, or NULL if done.
 */
char *strwrd(char *s, char *buf, size_t len, char *delim)
{
    s += strspn(s, delim);
    int n = strcspn(s, delim);  /* count the span (spn) of bytes in */
    if (len-1 < n)              /* the complement (c) of *delim */
        n = len-1;
    memcpy(buf, s, n);
    buf[n] = 0;
    s += n;
    return (*s == 0) ? NULL : s;
}
which is used like this:
char line[some_length];
char argv[10][20];
int argc;
for (argc = 0; argc < 10; argc++) {
    line = strwrd(line, argv[argc], sizeof(argv[argc]), " \t");
    if (line == NULL)
        break;
}

Wednesday, July 20, 2011

Splitting body and references in Latex for NSF proposals

For those of us who prepare grant proposals for the NSF, and use LaTeX for formatting, it can be a pain to separate the body of the proposal from the references, as the two have to be uploaded separately. A combination of pdftk (the PDF toolkit) and make handles this in a pretty straightforward fashion:

1. put the following in your LaTeX source, just before generating the bibliography, to mark the last page:

\label{page:last}

2. Use a makefile that looks like this (assuming you use pdflatex):

FILE = proposal
$(FILE).pdf: $(FILE).tex
    pdflatex $(FILE)
    bibtex $(FILE)
    pdflatex $(FILE)
    pdflatex $(FILE)

DESC_END = $(shell grep page:last $(FILE).aux | tr '{}' ' ' | awk '{print $$NF}')
REFS_START = $(shell expr $(DESC_END) + 1)

description.pdf: $(FILE).pdf
        pdftk $< cat 1-$(DESC_END) output $@

references.pdf: $(FILE).pdf
        pdftk $< cat $(REFS_START)-end output $@

Thursday, June 23, 2011

GNU Argp for OSX

After using optparse in Python, I've become spoiled and never want to go back to getopt/getopt_long for C code. The only package I've found for C that handles both options and help strings in a reasonable fashion seems to be GNU Argp, but it's non-standard and works on my Linux machines but not my Mac. A quick try at pulling Argp from the glibc source and porting to the Mac was leading to more work than I wanted, so the solution was this quick-and-dirty replacement.

And yes, it commits the cardinal sin of including code in a header file, but it works fine if you're only including it from main.c, and keeps your source compatible between systems. (just put it in the local directory, add "-I ." to your compile flags, and you're done.)