|
| 1 | +(defun demo-server-load-handlers () |
| 2 | + (let (handlers) |
| 3 | + (dolist (item (directory-files "../testdata")) |
| 4 | + (when (string-suffix-p ".json" item) |
| 5 | + (with-temp-buffer |
| 6 | + (insert-file-contents (file-name-concat ".." "testdata" item)) |
| 7 | + (let ((tmp (buffer-substring-no-properties |
| 8 | + (line-beginning-position) (1+ (line-end-position))))) |
| 9 | + (delete-region (line-beginning-position) (1+ (line-end-position))) |
| 10 | + |
| 11 | + (when (string-match (rx line-start (group (+? digit)) (literal "|") |
| 12 | + (group (+? anychar)) line-end) |
| 13 | + tmp) |
| 14 | + (push `(,(match-string 2 tmp) |
| 15 | + ,(string-to-number (match-string 1 tmp)) |
| 16 | + ,(buffer-string)) |
| 17 | + handlers)))))) |
| 18 | + handlers)) |
| 19 | + |
| 20 | +(defun demo-server-eval-sexprs (data) |
| 21 | + (with-temp-buffer |
| 22 | + (insert data) |
| 23 | + (goto-char (point-min)) |
| 24 | + (let ((last 0) tmp) |
| 25 | + (while (string-match |
| 26 | + (rx line-start (*? anychar) |
| 27 | + (group (literal "#(") (*? anychar) (literal ")")) |
| 28 | + (? ",") line-end) |
| 29 | + (buffer-string) last) |
| 30 | + (setq last (match-beginning 1)) |
| 31 | + (setq tmp (match-string 1 (buffer-string))) |
| 32 | + (goto-char (1+ (match-beginning 1))) |
| 33 | + (delete-region (1+ (match-beginning 1)) (1+ (match-end 1))) |
| 34 | + (insert (format "%s" (eval (read (substring tmp 1))))))) |
| 35 | + (buffer-string))) |
| 36 | + |
| 37 | +(defun demo-server-router (proc req-line) |
| 38 | + (unwind-protect |
| 39 | + (process-send-string |
| 40 | + proc |
| 41 | + ;; somewhat-(cond) |
| 42 | + (let (default code) |
| 43 | + (catch 'done |
| 44 | + (dolist (handler (or (get 'demo-server-load-handlers 'cache) |
| 45 | + (progn |
| 46 | + (put 'demo-server-load-handlers 'cache |
| 47 | + (demo-server-load-handlers)) |
| 48 | + (get 'demo-server-load-handlers 'cache)))) |
| 49 | + (let ((path (nth 0 handler)) (data (nth 2 handler))) |
| 50 | + (when (string-match path req-line) |
| 51 | + (setq code (nth 1 handler)) |
| 52 | + (setq default |
| 53 | + (string-join `(,(format "HTTP/1.1 %s" code) |
| 54 | + "Content-Type: application/json" "" |
| 55 | + ;; re-eval per request |
| 56 | + ,(demo-server-eval-sexprs data)) |
| 57 | + "\r\n")) |
| 58 | + (throw 'done t))))) |
| 59 | + (message "%s %s %s" |
| 60 | + (format-time-string "[%d/%b/%Y %H:%M:%S]" (current-time)) |
| 61 | + (when (string-match "\n" req-line) |
| 62 | + (substring req-line 0 (match-beginning 0))) |
| 63 | + (or code "404")) |
| 64 | + (or default |
| 65 | + (string-join '("HTTP/1.1 404 Not Found" |
| 66 | + "Content-Type: text/plain" "" |
| 67 | + "Page not found!\n") "\r\n")))) |
| 68 | + (process-send-eof proc))) |
| 69 | + |
| 70 | +(defun demo-server-filter (proc string) |
| 71 | + "Handle the incoming data from the network connection." |
| 72 | + (let ((req-line (decode-coding-string string 'utf-8)) |
| 73 | + (handlers )) |
| 74 | + (demo-server-router proc req-line))) |
| 75 | + |
| 76 | +(defun demo-server-sentinel (proc event) |
| 77 | + "Handle the closing of a network connection." |
| 78 | + (when (string-match "closed" event) |
| 79 | + (message "Connection closed: %s" proc))) |
| 80 | + |
| 81 | +(let ((proc-name "syncthing-demo-server")) |
| 82 | + (mapcar (lambda (proc) |
| 83 | + (when (string= (process-name proc) proc-name) |
| 84 | + (message "Restart!") |
| 85 | + (delete-process proc))) |
| 86 | + (process-list)) |
| 87 | + (put 'demo-server-load-handlers 'cache nil) |
| 88 | + (make-network-process |
| 89 | + :name proc-name |
| 90 | + :server t |
| 91 | + :host (or (getenv "DEMO_HOST") "127.0.0.1") |
| 92 | + :service (string-to-number (or (getenv "DEMO_PORT") "5000")) |
| 93 | + :family 'ipv4 |
| 94 | + :sentinel 'demo-server-sentinel |
| 95 | + :filter 'demo-server-filter)) |
0 commit comments