Compare commits

...

5 Commits

112
palps.rkt
View File

@@ -1,75 +1,73 @@
#! /usr/bin/env /usr/local/bin/racket
#lang racket
;; get callsign, first thing passed by BPQ
(define call (read-line (current-input-port) 'any))
;; default directory used for storing dynamic application data
(define data-dir
(build-path (find-system-path 'home-dir)
(build-path ".local" "share" "palps")))
;; drop SSID
(set! call (first (string-split call "-")))
(define (call-valid? call)
(and
(<= (string-length call) 9)
(andmap
(or/c char-numeric? char-alphabetic? (curry char=? #\-))
(string->list call))))
;; downcase callsign
(set! call (string-downcase call))
;; if the sysadmin has created a directory for that callsign, then
;; they are an authorized user.
(when (not (directory-exists? call))
(displayln "sorry, but you are not an authorized user of the PALPS system")
(displayln "visit https://gitea.farpn.net/w6vvn/palps for an explanation")
(flush-output)
(exit))
;; change to the authroized users directory
(current-directory call)
(printf "welcome ~a. now starting (r)ed, the standard text editor~n" call)
(flush-output)
(define (our-make-temporary-file [copy-from #f])
;; when deployed as a systemd socket, this program is expected to be
;; ran with PrivateTmp. however, in case this is not true, we still
;; make our own directory. "red", the restricted version of "ed",
;; has no facility for getting outside of the directory we start it
;; in.
(make-temporary-file "red.~a"
#:copy-from copy-from
#:base-dir (make-temporary-directory)))
;; unfortunately, we cannot just exec red and let it take over the
;; I/O. ed, being the standard text editor, only works with standard
;; line endings, \n. telnet and BPQ, however, use \r\n. \r\n upsets
;; ed, the standard text editor. so we need to wrap the input and
;; output ports ourselves in order to provide this translation.
(define (ed [path (our-make-temporary-file)])
(match-define
(list stdout stdin pid stderr proc)
(parameterize ([current-directory (path-only path)])
(process* "/usr/bin/red"
"-p*"
(path->string (file-name-from-path path))
#:set-pwd? #t)))
(match-define
(list stdout stdin pid stderr proc)
(process* "/usr/bin/red"
"-p*"
#:set-pwd? #t))
(define buffer (make-bytes 128))
(define buffer (make-bytes 128))
(define (loop)
(let ([evt-result
(sync
(choice-evt (read-bytes-avail!-evt buffer stdout)
(read-line-evt (current-input-port) 'any)))])
(define (loop)
(let ([evt-result
(sync
(choice-evt (read-bytes-avail!-evt buffer stdout)
(read-line-evt (current-input-port) 'any)))])
(cond
;; read-line-evt results in a string. we have something to write
;; on stdin.
[(string? evt-result)
(displayln evt-result stdin)
(flush-output stdin)
(loop)]
(cond
;; read-line-evt results in a string. we have something to write
;; on stdin.
[(string? evt-result)
(displayln evt-result stdin)
(flush-output stdin)
(loop)]
;; read-bytes-avail!-evt results in an integer. we have
;; something to read from stdout.
[(integer? evt-result)
(display (bytes->string/utf-8 (subbytes buffer 0 evt-result)))
(flush-output)
(loop)]
;; read-bytes-avail!-evt results in an integer. we have
;; something to read from stdout.
[(integer? evt-result)
(display (bytes->string/utf-8 (subbytes buffer 0 evt-result)))
(flush-output)
(loop)]
;; either event may EOF, which means that ed has died.
[(eof-object? evt-result)
;; either event may EOF, which means that ed has died.
[(eof-object? evt-result)
;; clean up
(close-output-port stdin)
(close-input-port stdout)
(close-input-port stderr)
(proc 'kill)])))
;; clean up
(close-output-port stdin)
(close-input-port stdout)
(close-input-port stderr)
(proc 'kill)
(loop)
(displayln "goodbye")
(flush-output)
(exit)
path])))
(loop))