2023-12-16 15:06:29 -05:00
|
|
|
(load "~/quicklisp/setup.lisp")
|
2024-01-04 08:45:54 -05:00
|
|
|
(ql:quickload '(:clack :websocket-driver :cl-bcrypt :datafly
|
|
|
|
:sxql :alexandria :com.inuoe.jzon))
|
2023-12-16 15:06:29 -05:00
|
|
|
|
|
|
|
(defpackage :mmo-server
|
2024-01-04 08:45:54 -05:00
|
|
|
(:use :common-lisp :clack :websocket-driver :cl-bcrypt :datafly
|
|
|
|
:sxql :alexandria :com.inuoe.jzon))
|
2023-12-16 15:06:29 -05:00
|
|
|
|
|
|
|
(in-package :mmo-server)
|
2023-12-31 15:21:34 -05:00
|
|
|
|
2024-01-01 15:26:28 -05:00
|
|
|
(defvar *connections* (make-hash-table))
|
|
|
|
|
|
|
|
(defun handle-new-connection (con)
|
2024-01-04 08:45:54 -05:00
|
|
|
(let ((user (make-hash-table)))
|
|
|
|
(setf (gethash "uid" user) (random 100000))
|
|
|
|
(setf (gethash "cmd" user) "login")
|
|
|
|
(websocket-driver:send con (com.inuoe.jzon:stringify user))
|
|
|
|
(remhash "cmd" user)
|
|
|
|
(setf (gethash con *connections*) user)))
|
2024-01-01 15:26:28 -05:00
|
|
|
|
2024-01-04 08:45:54 -05:00
|
|
|
(defun broadcast-to-room (connection message)
|
|
|
|
(let* ((json (com.inuoe.jzon:parse message))
|
|
|
|
(user (gethash connection *connections*))
|
|
|
|
(cmd (gethash "cmd" json)))
|
|
|
|
(cond
|
|
|
|
((equalp cmd "user-login")
|
2024-01-13 11:30:32 -05:00
|
|
|
(format t "user-login: ~A~%" message)
|
|
|
|
(let* ((copy (alexandria:copy-hash-table *connections*))
|
|
|
|
(qwer (make-hash-table)))
|
|
|
|
(remhash connection copy)
|
|
|
|
(setf (gethash "cmd" qwer) "user-login")
|
|
|
|
(setf (gethash "connections" qwer) copy)
|
2024-01-04 08:45:54 -05:00
|
|
|
(websocket-driver:send connection (com.inuoe.jzon:stringify copy))))
|
|
|
|
((equalp cmd "move")
|
2024-01-13 11:30:32 -05:00
|
|
|
(format t "move: ~A~%" message)
|
2024-01-04 08:45:54 -05:00
|
|
|
(setf (gethash "x" user) (gethash "x" json))
|
|
|
|
(setf (gethash "y" user) (gethash "y" json))
|
|
|
|
(setf (gethash connection *connections*) user))
|
|
|
|
(t
|
|
|
|
(format t "Unknown message: ~A~%" message)))
|
2024-01-01 16:33:39 -05:00
|
|
|
|
2024-01-13 11:30:32 -05:00
|
|
|
(when (not (equalp cmd "user-login"))
|
|
|
|
(loop :for con :being :the :hash-key :of *connections* :do
|
|
|
|
(websocket-driver:send con message)))))
|
2024-01-01 15:26:28 -05:00
|
|
|
|
|
|
|
(defun handle-close-connection (connection)
|
2024-01-04 08:45:54 -05:00
|
|
|
(let ((message (format nil "{\"cmd\":\"logout\", \"uid\":\"~a\"}"
|
|
|
|
(gethash "uid" (gethash connection *connections*)))))
|
2024-01-01 15:26:28 -05:00
|
|
|
(remhash connection *connections*)
|
|
|
|
(loop :for con :being :the :hash-key :of *connections* :do
|
|
|
|
(websocket-driver:send con message))))
|
|
|
|
|
|
|
|
(defun chat-server (env)
|
|
|
|
(let ((ws (websocket-driver:make-server env)))
|
|
|
|
(websocket-driver:on :open ws
|
|
|
|
(lambda () (handle-new-connection ws)))
|
|
|
|
|
|
|
|
(websocket-driver:on :message ws
|
|
|
|
(lambda (msg)
|
|
|
|
(broadcast-to-room ws msg)))
|
|
|
|
|
|
|
|
(websocket-driver:on :close ws
|
|
|
|
(lambda (&key code reason)
|
|
|
|
(declare (ignore code reason))
|
|
|
|
(handle-close-connection ws)))
|
2024-01-01 16:33:39 -05:00
|
|
|
(websocket-driver:on :error ws
|
|
|
|
(lambda (error)
|
|
|
|
(format t "Got an error: ~S~%" error)))
|
|
|
|
|
2024-01-01 15:26:28 -05:00
|
|
|
(lambda (responder)
|
|
|
|
(declare (ignore responder))
|
|
|
|
(websocket-driver:start-connection ws))))
|
|
|
|
|
|
|
|
(defvar *chat-handler* (clack:clackup #'chat-server :port 12345))
|