(module status mzscheme (require mzlib/file mzlib/list mzlib/string mzlib/date web-server/servlet web-server/servlet/servlet-structs web-server/managers/timeouts net/uri-codec handin-server/private/md5 handin-server/private/logger handin-server/private/config handin-server/private/hooker) (define get-user-data (let ([users-file (build-path server-dir "users.ss")]) (lambda (user) (get-preference (string->symbol user) (lambda () #f) #f users-file)))) (define (clean-str s) (regexp-replace #rx" *$" (regexp-replace #rx"^ *" s "") "")) (define (aget alist key) (cond [(assq key alist) => cdr] [else #f])) (define (make-page title . body) `(html (head (title ,title)) (body ([bgcolor "white"]) (h1 ((align "center")) ,title) ,@body))) (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))) stringstring (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))) stringdir 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 (with-input-from-file file (lambda () (read-bytes (file-size 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 status 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"])))))))))] [user (clean-str (aget (request-bindings request) 'user))] [passwd (aget (request-bindings request) 'passwd)] [user-data (get-user-data user)]) (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 status 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 (start initial-request) (parameterize ([current-session (web-counter)]) (login-page null (aget (request-bindings initial-request) 'handin) #f))) (define interface-version 'v2) (define name "status") (define (instance-expiration-handler failed-request) (let ([this (servlet-url->url-string/no-continuation (request->servlet-url failed-request))]) `(html (head (meta [(http-equiv "refresh") (content ,(format "3;URL=~a" this))])) (body "Your session has expired, " (a ([href ,this]) "restarting") " in 3 seconds.")))) (define manager (create-timeout-manager instance-expiration-handler 600 600)) (provide interface-version start name manager))