#lang scheme/base (require scheme/list scheme/path scheme/file scheme/date net/uri-codec web-server/servlet handin-server/private/md5 handin-server/private/logger handin-server/private/config handin-server/private/hooker "run-servlet.ss") (define (aget alist key) (cond [(assq key alist) => cdr] [else #f])) (define (clean-str s) (regexp-replace #rx" +$" (regexp-replace #rx"^ +" s "") "")) (define (make-page title . body) `(html (head (title ,title)) (body ([bgcolor "white"]) (h1 ((align "center")) ,title) ,@body))) (define get-user-data (let ([users-file (build-path server-dir "users.ss")]) (unless (file-exists? users-file) (log-line "WARNING: users file missing on startup: ~a" users-file)) (lambda (user) (and user (get-preference (string->symbol user) (lambda () #f) 'timestamp users-file))))) (define (relativize-path p) (path->string (find-relative-path (normalize-path server-dir) p))) (define (make-k k tag) (format "~a~atag=~a" k (if (regexp-match? #rx"^[^#]*[?]" k) "&" "?") (uri-encode tag))) ;; `look-for' can be a username as a string (will find "bar+foo" for "foo"), or ;; a regexp that should match the whole directory name (used with "^solution" ;; below) (define (find-handin-entry hi look-for) (let ([dir (assignment<->dir hi)]) (and (directory-exists? dir) (ormap (lambda (d) (let ([d (path->string d)]) (and (cond [(string? look-for) (member look-for (regexp-split #rx" *[+] *" d))] [(regexp? look-for) (regexp-match? look-for d)] [else (error 'find-handin-entry "internal error: ~e" look-for)]) (build-path dir d)))) (directory-list dir))))) (define (handin-link k user hi) (let* ([dir (find-handin-entry hi user)] [l (and dir (with-handlers ([exn:fail? (lambda (x) null)]) (parameterize ([current-directory dir]) (sort (filter (lambda (f) (and (not (equal? f "grade")) (file-exists? f))) (map path->string (directory-list))) string))))]) (if (pair? l) (cdr (append-map (lambda (f) (let ([hi (build-path dir f)]) `((br) (a ([href ,(make-k k (relativize-path hi))]) ,f) " (" ,(date->string (seconds->date (file-or-directory-modify-seconds hi)) #t) ")"))) l)) (list (format "No handins accepted so far for user ~s, assignment ~s" user hi))))) (define (solution-link k hi) (let ([soln (and (member (assignment<->dir hi) (get-conf 'inactive-dirs)) (find-handin-entry hi #rx"^solution"))] [none `((i "---"))]) (cond [(not soln) none] [(file-exists? soln) `((a ((href ,(make-k k (relativize-path soln)))) "Solution"))] [(directory-exists? soln) (parameterize ([current-directory soln]) (let ([files (sort (map path->string (filter file-exists? (directory-list))) string)]) (if (null? files) none (apply append (map (lambda (f) `((a ([href ,(make-k k (relativize-path (build-path soln f)))]) (tt ,f)) (br))) files)))))] [else none]))) (define (handin-grade user hi) (let* ([dir (find-handin-entry hi user)] [grade (and dir (let ([filename (build-path dir "grade")]) (and (file-exists? filename) (with-input-from-file filename (lambda () (read-string (file-size filename)))))))]) (or grade "--"))) (define (one-status-page user for-handin) (let* ([next (send/suspend (lambda (k) (make-page (format "User: ~a, Handin: ~a" user for-handin) `(p ,@(handin-link k user for-handin)) `(p "Grade: " ,(handin-grade user for-handin)) `(p ,@(solution-link k for-handin)) `(p (a ([href ,(make-k k "allofthem")]) ,(format "All handins for ~a" user))))))] [tag (aget (request-bindings next) 'tag)]) (if (string=? tag "allofthem") (all-status-page user) (download user tag)))) (define (all-status-page user) (define (cell . texts) `(td ([bgcolor "white"]) ,@texts)) (define (rcell . texts) `(td ([bgcolor "white"] [align "right"]) ,@texts)) (define (header . texts) `(td ([bgcolor "#f0f0f0"]) (big (strong ,@texts)))) (define ((row k active?) dir) (let ([hi (assignment<->dir dir)]) `(tr ([valign "top"]) ,(apply header hi (if active? `((br) (small (small "[active]"))) '())) ,(apply cell (handin-link k user hi)) ,(rcell (handin-grade user hi)) ,(apply cell (solution-link k hi))))) (let* ([next (send/suspend (lambda (k) (make-page (format "All Handins for ~a" user) `(table ([bgcolor "#ddddff"] [cellpadding "6"] [align "center"]) (tr () ,@(map header '(nbsp "Files" "Grade" "Solution"))) ,@(append (map (row k #t) (get-conf 'active-dirs)) (map (row k #f) (get-conf 'inactive-dirs)))))))] [tag (aget (request-bindings next) 'tag)]) (download user tag))) (define (download who tag) (define (check path elts allow-active?) (let loop ([path path] [elts (reverse elts)]) (let*-values ([(base name dir?) (split-path path)] [(name) (path->string name)] [(check) (and (pair? elts) (car elts))]) (if (null? elts) ;; must be rooted in a submission directory (why build-path instead ;; of using `path'? -- because path will have a trailing slash) (member (build-path base name) (get-conf (if allow-active? 'all-dirs 'inactive-dirs))) (and (cond [(eq? '* check) #t] [(regexp? check) (regexp-match? check name)] [(string? check) (or (equal? name check) (member check (regexp-split #rx" *[+] *" name)))] [else #f]) (loop base (cdr elts))))))) (define file (build-path server-dir tag)) (with-handlers ([exn:fail? (lambda (exn) (log-line "Status exception: ~a" (exn-message exn)) (make-page "Error" "Illegal file access"))]) ;; Make sure the user is allowed to read the requested file: (or (check file `(,who *) #t) (check file `(#rx"^solution") #f) (check file `(#rx"^solution" *) #f) (error 'download "bad file access for ~s: ~a" who file)) (log-line "Status file-get: ~s ~a" who file) (hook 'status-file-get `([username ,(string->symbol who)] [file ,file])) ;; Return the downloaded file (let* ([data (file->bytes file)] [html? (regexp-match? #rx"[.]html?$" (string-foldcase tag))] [wxme? (regexp-match? #rx#"^(?:#reader[(]lib\"read.ss\"\"wxme\"[)])?WXME" data)]) (make-response/full 200 #"Okay" (current-seconds) (cond [html? #"text/html"] [wxme? #"application/data"] [else #"text/plain"]) (list (make-header #"Content-Length" (string->bytes/latin-1 (number->string (bytes-length data)))) (make-header #"Content-Disposition" (string->bytes/utf-8 (format "~a; filename=~s" (if wxme? "attachment" "inline") (let-values ([(base name dir?) (split-path file)]) (path->string name)))))) (list data))))) (define (status-page user for-handin) (log-line "Status access: ~s" user) (hook 'status-login `([username ,(string->symbol user)])) (if for-handin (one-status-page user for-handin) (all-status-page user))) (define (login-page for-handin errmsg) (let* ([request (send/suspend (lambda (k) (make-page "Handin Status Login" `(form ([action ,k] [method "post"]) (table ([align "center"]) (tr (td ([colspan "2"] [align "center"]) (font ([color "red"]) ,(or errmsg 'nbsp)))) (tr (td "Username") (td (input ([type "text"] [name "user"] [size "20"] [value ""])))) (tr (td nbsp)) (tr (td "Password") (td (input ([type "password"] [name "passwd"] [size "20"] [value ""])))) (tr (td ([colspan "2"] [align "center"]) (input ([type "submit"] [name "post"] [value "Login"])))))))))] [bindings (request-bindings request)] [user (aget bindings 'user)] [passwd (aget bindings 'passwd)] [user (and user (clean-str user))] [user (and user (if (get-conf 'username-case-sensitive) user (string-foldcase user)))] [user-data (get-user-data user)]) (redirect/get) (cond [(and user-data (string? passwd) (let ([pw (md5 passwd)]) (or (equal? pw (car user-data)) (equal? pw (get-conf 'master-password))))) (status-page user for-handin)] [else (login-page for-handin "Bad username or password")]))) (define web-counter (let ([sema (make-semaphore 1)] [count 0]) (lambda () (dynamic-wind (lambda () (semaphore-wait sema)) (lambda () (set! count (add1 count)) (format "w~a" count)) (lambda () (semaphore-post sema)))))) (define default-context-length (error-print-context-length)) (define (dispatcher request) (error-print-context-length default-context-length) (parameterize ([current-session (web-counter)]) (login-page (aget (request-bindings request) 'handin) #f))) (provide run) (define (run) (if (get-conf 'use-https) (begin0 (parameterize ([error-print-context-length 0]) (run-servlet dispatcher #:namespace '(handin-server/private/md5 handin-server/private/logger handin-server/private/config handin-server/private/hooker handin-server/private/reloadable) #:log-file (get-conf 'web-log-file))) (log-line "*** embedded web server started")) ;; simple "server" so it's known that there is no server (lambda (msg . args) (when (eq? 'connect msg) (for-each (lambda (x) (display x (cadr args))) '(#"HTTP/1.0 200 OK\r\n" #"Content-Type: text/html\r\n" #"\r\n" #"