utils.lisp 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325
  1. (in-package #:chatikbot)
  2. (defvar *bot-name* "@chatikbot" "bot name to properly handle text input")
  3. (defvar *hooks* (make-hash-table) "Hooks storage")
  4. (defun run-hooks (event &rest arguments)
  5. (let ((hooks (gethash event *hooks*)))
  6. (labels ((try-handle (func)
  7. (apply func arguments)))
  8. (unless (some #'try-handle hooks)
  9. (log:info "unhandled" event arguments)))))
  10. (defun add-hook (event hook &optional append)
  11. (let ((existing (gethash event *hooks*)))
  12. (unless (member hook existing)
  13. (setf (gethash event *hooks*)
  14. (if append (append existing (list hook))
  15. (cons hook existing))))))
  16. (defun remove-hook (event hook)
  17. (setf (gethash event *hooks*)
  18. (remove hook (gethash event *hooks*))))
  19. (defun string-to-event (key)
  20. (intern (string-upcase (substitute #\- #\_ key)) :keyword))
  21. ;; Settings
  22. (defvar *settings* nil "List of plugin's settings symbols")
  23. (defmacro defsetting (var &optional val doc)
  24. `(progn (defvar ,var ,val ,doc)
  25. (push ',var *settings*)))
  26. (defun load-settings ()
  27. (loop for (var val) in (db-select "select var, val from settings")
  28. do (setf (symbol-value (intern var))
  29. (handler-case (read-from-string val)
  30. (error (e) (log:error e))))))
  31. (defun set-setting (symbol value)
  32. (handler-case
  33. (progn
  34. (db-execute "replace into settings (var, val) values (?, ?)"
  35. (symbol-name symbol)
  36. (write-to-string value))
  37. (setf (symbol-value symbol) value))
  38. (error (e) (log:error e))))
  39. (defvar *backoff-start* 1 "Initial back-off")
  40. (defvar *backoff-max* 64 "Maximum back-off delay")
  41. (defun loop-with-error-backoff (func)
  42. (let ((backoff *backoff-start*))
  43. (loop
  44. do
  45. (handler-case
  46. (progn
  47. (funcall func)
  48. (setf backoff *backoff-start*))
  49. (error (e)
  50. (log:error e)
  51. (log:info "Backing off for" backoff)
  52. (sleep backoff)
  53. (setf backoff (min *backoff-max*
  54. (* 2 backoff))))
  55. (bordeaux-threads:timeout (e)
  56. (log:error e)
  57. (log:info "Backing off for" backoff)
  58. (sleep backoff)
  59. (setf backoff (min *backoff-max*
  60. (* 2 backoff))))))))
  61. (defun replace-all (string part replacement &key (test #'char=))
  62. "Returns a new string in which all the occurences of the part
  63. is replaced with replacement."
  64. (with-output-to-string (out)
  65. (loop with part-length = (length part)
  66. for old-pos = 0 then (+ pos part-length)
  67. for pos = (search part string
  68. :start2 old-pos
  69. :test test)
  70. do (write-string string out
  71. :start old-pos
  72. :end (or pos (length string)))
  73. when pos do (write-string replacement out)
  74. while pos)))
  75. (defmacro aget (key alist)
  76. `(cdr (assoc ,key ,alist :test #'equal)))
  77. (defun mappend (fn &rest lists)
  78. "Apply fn to each element of lists and append the results."
  79. (apply #'append (apply #'mapcar fn lists)))
  80. (defun random-elt (choices)
  81. "Choose an element from a list at random."
  82. (elt choices (random (length choices))))
  83. (defun flatten (the-list)
  84. "Append together elements (or lists) in the list."
  85. (mappend #'(lambda (x) (if (listp x) (flatten x) (list x))) the-list))
  86. (defun preprocess-input (text)
  87. (when text
  88. (let* ((first-space (position #\Space text))
  89. (first-word (subseq text 0 first-space)))
  90. (if (equal first-word *bot-name*)
  91. (preprocess-input (subseq text (1+ first-space)))
  92. (replace-all text *bot-name* "ты")))))
  93. (defun parse-cmd (text)
  94. (let* ((args (split-sequence:split-sequence #\Space (subseq text 1) :remove-empty-subseqs t))
  95. (cmd (subseq (car args) 0 (position #\@ (car args)))))
  96. (values (intern (string-upcase cmd) "KEYWORD") (rest args))))
  97. (defun http-default (url)
  98. (let ((uri (puri:uri url)))
  99. (puri:render-uri
  100. (if (null (puri:uri-scheme uri))
  101. (puri:uri (format nil "http://~A" url))
  102. uri)
  103. nil)))
  104. ;; XML processing
  105. (defun xml-request (url &key encoding parameters)
  106. (multiple-value-bind (raw-body status headers uri http-stream)
  107. (drakma:http-request (http-default url)
  108. :parameters parameters
  109. :external-format-out :utf-8
  110. :force-binary t
  111. :decode-content t)
  112. (declare (ignore status http-stream))
  113. (let ((encoding
  114. (or
  115. ;; 1. Provided encoding
  116. encoding
  117. ;; 2. Content-type header
  118. (ignore-errors
  119. (let ((ct (aget :content-type headers)))
  120. (subseq ct (1+ (position #\= ct)))))
  121. ;; 3. Parse first 1000 bytes
  122. (ignore-errors
  123. (let ((dom (plump:parse (flex:octets-to-string (subseq raw-body 0 1000)))))
  124. (or
  125. ;; 3.1 Content-type from http-equiv
  126. (ignore-errors
  127. (let ((ct (loop for meta in (get-by-tag dom "meta")
  128. for http-equiv = (plump:get-attribute meta "http-equiv")
  129. for content = (plump:get-attribute meta "content")
  130. when (equal http-equiv "Content-Type")
  131. return content)))
  132. (subseq ct (1+ (position #\= ct)))))
  133. ;; 3.2 'content' xml node attribute
  134. (ignore-errors (plump:get-attribute (plump:first-child dom) "encoding")))))
  135. ;; 4. Default 'utf-8'
  136. "utf-8")))
  137. (values
  138. (handler-bind ((flex:external-format-encoding-error
  139. (lambda (c) (use-value #\? c))))
  140. (plump:parse
  141. (flex:octets-to-string raw-body :external-format (intern encoding 'keyword))))
  142. uri))))
  143. (defun get-by-tag (node tag)
  144. (nreverse (org.shirakumo.plump.dom::get-elements-by-tag-name node tag)))
  145. (defun select-text (selector node)
  146. (ignore-errors
  147. (string-trim '(#\Newline #\Space #\Return) (plump:text (elt (clss:select selector node) 0)))))
  148. ;; JSON processing
  149. (defun json-request (url &key (method :get) parameters (object-as :alist))
  150. (multiple-value-bind (stream status headers uri http-stream)
  151. (drakma:http-request (http-default url) :method method :parameters parameters
  152. :external-format-out :utf-8
  153. :force-binary t :want-stream t :decode-content t)
  154. (declare (ignore status headers))
  155. (unwind-protect
  156. (progn
  157. (setf (flex:flexi-stream-external-format stream) :utf-8)
  158. (values (yason:parse stream :object-as object-as) uri))
  159. (ignore-errors (close http-stream)))))
  160. (defun format-ts (ts)
  161. (local-time:format-timestring nil ts
  162. :format '(:year "-" (:month 2) "-" (:day 2) " "
  163. (:hour 2) ":" (:min 2) ":" (:sec 2))))
  164. (defun google-tts (text &key (lang "en"))
  165. (let ((path #P"google_tts.mp3"))
  166. (with-open-file (s path :direction :output
  167. :element-type '(unsigned-byte 8)
  168. :if-exists :supersede
  169. :if-does-not-exist :create)
  170. (write-sequence
  171. (drakma:http-request
  172. "http://translate.google.com/translate_tts"
  173. :parameters `(("ie" . "UTF-8")
  174. ("client" . "t")
  175. ("tl" . ,lang)
  176. ("q" . ,text))
  177. :user-agent "stagefright/1.2 (Linux;Android 5.0)"
  178. :additional-headers '((:referer . "http://translate.google.com/"))
  179. :external-format-out :utf-8
  180. :force-binary t)
  181. s)
  182. path)))
  183. (defun say-it (lang words)
  184. (cons :voice
  185. (google-tts (print-with-spaces words) :lang lang)))
  186. (defun yit-info ()
  187. (labels ((get-rows (url)
  188. (rest (get-by-tag (plump:get-element-by-id (xml-request url) "apartmentList") "tr")))
  189. (row-data (row)
  190. (mapcar (lambda (e) (string-trim '(#\Newline #\Space) (plump:text e)))
  191. (get-by-tag row "td")))
  192. (format-data (data)
  193. (format nil "~{~A~^ ~}" (mapcar (lambda (n) (nth n data)) '(1 2 3 4 7 6))))
  194. (get-intresting (rows)
  195. (loop for row in rows
  196. for data = (row-data row)
  197. for rooms = (parse-integer (nth 2 data))
  198. for area = (parse-float:parse-float (replace-all (nth 3 data) "," "."))
  199. when (= rooms 3)
  200. when (< 65 area 75)
  201. collect data))
  202. (format-apts (url)
  203. (let ((apts (get-intresting (get-rows url))))
  204. (format nil "~A~%~{~A~^~%~}~%~A/~A" url (mapcar #'format-data apts)
  205. (length (remove "забронировано" apts :test #'equal :key #'(lambda (r) (nth 7 r)) ))
  206. (length apts)))))
  207. (format nil "~{~A~^~%~%~}"
  208. (mapcar #'format-apts
  209. '("http://www.yitspb.ru/yit_spb/catalog/apartments/novoorlovskiy-korpus-1-1-1"
  210. "http://www.yitspb.ru/yit_spb/catalog/apartments/novoorlovskiy-korpus-1-1-2")))))
  211. (defmacro def-message-handler (name (message) &body body)
  212. `(progn
  213. (defun ,name (,message)
  214. (let ((message-id (aget "message_id" ,message))
  215. (from-id (aget "id" (aget "from" ,message)))
  216. (chat-id (aget "id" (aget "chat" ,message)))
  217. (text (aget "text" ,message)))
  218. (declare (ignorable message-id from-id chat-id text))
  219. (handler-case (progn ,@body)
  220. (error (e)
  221. (log:error "~A" e)
  222. (bot-send-message chat-id
  223. (format nil "Ошибочка вышла~@[: ~A~]"
  224. (when (member chat-id *admins*) e)))))))
  225. (add-hook :update-message ',name)))
  226. (defmacro def-message-cmd-handler (name (&rest commands) &body body)
  227. `(def-message-handler ,name (message)
  228. (when (and text (equal #\/ (char text 0)))
  229. (multiple-value-bind (cmd args) (parse-cmd text)
  230. (when (member cmd (list ,@commands))
  231. (log:info cmd message-id chat-id from-id args)
  232. ,@body
  233. t)))))
  234. (defmacro def-message-admin-cmd-handler (name (&rest commands) &body body)
  235. `(def-message-handler ,name (message)
  236. (when (and (member chat-id *admins*)
  237. text (equal #\/ (char text 0)))
  238. (multiple-value-bind (cmd args) (parse-cmd text)
  239. (when (member cmd (list ,@commands))
  240. (log:info cmd message-id chat-id from-id args)
  241. ,@body
  242. t)))))
  243. ;; Schedule
  244. (defmacro defcron (name (&rest schedule) &body body)
  245. (let ((schedule (or schedule '(:minute '* :hour '*))))
  246. `(progn
  247. (defun ,name ()
  248. (handler-case (progn ,@body)
  249. (error (e) (log:error e))))
  250. (add-hook :starting #'(lambda ()
  251. (clon:schedule-function
  252. ',name (clon:make-scheduler
  253. (clon:make-typed-cron-schedule
  254. ,@schedule)
  255. :allow-now-p t)
  256. :name ',name :thread t)
  257. (values))))))
  258. ;; Fix bug in local-time (following symlinks in /usr/share/zoneinfo/
  259. ;; leads to bad cutoff)
  260. (in-package #:local-time)
  261. (defun reread-timezone-repository (&key (timezone-repository *default-timezone-repository-path*))
  262. (check-type timezone-repository (or pathname string))
  263. (multiple-value-bind (valid? error)
  264. (ignore-errors
  265. (truename timezone-repository)
  266. t)
  267. (unless valid?
  268. (error "REREAD-TIMEZONE-REPOSITORY was called with invalid PROJECT-DIRECTORY (~A). The error is ~A."
  269. timezone-repository error)))
  270. (let* ((root-directory timezone-repository)
  271. (cutoff-position (length (princ-to-string root-directory))))
  272. (flet ((visitor (file)
  273. (handler-case
  274. (let* ((full-name (subseq (princ-to-string file) cutoff-position))
  275. (name (pathname-name file))
  276. (timezone (%realize-timezone (make-timezone :path file :name name))))
  277. (setf (gethash full-name *location-name->timezone*) timezone)
  278. (map nil (lambda (subzone)
  279. (push timezone (gethash (subzone-abbrev subzone)
  280. *abbreviated-subzone-name->timezone-list*)))
  281. (timezone-subzones timezone)))
  282. (invalid-timezone-file () nil))))
  283. (setf *location-name->timezone* (make-hash-table :test 'equal))
  284. (setf *abbreviated-subzone-name->timezone-list* (make-hash-table :test 'equal))
  285. (cl-fad:walk-directory root-directory #'visitor :directories nil :follow-symlinks nil
  286. :test (lambda (file)
  287. (not (find "Etc" (pathname-directory file) :test #'string=))))
  288. (cl-fad:walk-directory (merge-pathnames "Etc/" root-directory) #'visitor :directories nil))))
  289. (let ((zonepath "/usr/share/zoneinfo/"))
  290. (when (directory zonepath)
  291. (local-time:reread-timezone-repository :timezone-repository zonepath)))