saver.lisp 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340
  1. (in-package #:chatikbot)
  2. (defsetting *saver-default-timezone* -3 "Default timezone for *saver-notify-hour* calculation. GMT+3")
  3. (defsetting *saver-notify-hour* 11 "Notify with upcoming payments and saves at this time")
  4. (defun %saver/parse-schedule (schedule)
  5. (labels ((parse (text &optional def-min def-max)
  6. (let ((dash-pos (position #\- text))
  7. (star-pos (position #\* text))
  8. (slash-pos (position #\/ text)))
  9. (if (or dash-pos star-pos slash-pos)
  10. (let ((min (if star-pos def-min
  11. (when dash-pos (parse-integer text :end dash-pos))))
  12. (max (if star-pos def-max
  13. (when dash-pos (parse-integer text :start (1+ dash-pos) :end slash-pos))))
  14. (step (or (when slash-pos (parse-integer text :start (1+ slash-pos))) 1)))
  15. (list nil min max step))
  16. (list (sort (mapcar #'parse-integer (split-sequence:split-sequence #\, text)) #'<))))))
  17. (destructuring-bind (&optional day-sched month-sched year-sched)
  18. (split-sequence:split-sequence #\Space schedule :remove-empty-subseqs t)
  19. (list (if day-sched (parse day-sched 1 31) (list nil 1 31))
  20. (if month-sched (parse month-sched 1 12) (list nil 1 12))
  21. (if year-sched (parse year-sched 1900) (list nil 1900))))))
  22. (defun %saver/get-next-time (schedule universal-time &optional (dir t))
  23. (labels ((leapp (year)
  24. (or (and (zerop (mod year 4))
  25. (not (zerop (mod year 100))))
  26. (zerop (mod year 400))))
  27. (clamp-day-rule (day-rule month year)
  28. (destructuring-bind (lst &optional (min 0) max (step 1)) day-rule
  29. (let ((max-day (case month
  30. (2 (if (leapp year) 29 28))
  31. ((1 3 5 7 8 10 12) 31)
  32. (otherwise 30))))
  33. (if (consp lst)
  34. (list (remove-duplicates (mapcar (lambda (d) (min d max-day)) lst)))
  35. (list nil (min min) (when max (min max max-day)) step)))))
  36. (next (from rule)
  37. (destructuring-bind (lst &optional (min 0) max (step 1)) rule
  38. (if (consp lst)
  39. (let ((nv (find from lst :test (if dir #'<= #'>=) :from-end (not dir))))
  40. (if nv
  41. (values nv nil)
  42. (values (car (if dir lst (last lst))) t)))
  43. (let* ((m (mod (- from min) step))
  44. (nv (if (zerop m) from
  45. (funcall (if dir #'+ #'-) from
  46. (if dir (- step m) m)))))
  47. (cond
  48. ((and max (> nv max))
  49. (if dir
  50. (values min t)
  51. (values max nil)))
  52. ((and min (< nv min))
  53. (if dir
  54. (values min nil)
  55. (values max t)))
  56. (t (values nv nil)))))))
  57. (add (v of)
  58. (funcall (if dir #'+ #'-) v (if of 1 0)))
  59. (reset-rule (rule)
  60. (destructuring-bind (lst &optional (min 0) max (step 1)) rule
  61. (if (consp lst)
  62. (car (if dir lst (last lst)))
  63. (if dir min
  64. (let ((m (mod (- max min) step)))
  65. (- max (if (zerop m) 0 m))))))))
  66. (multiple-value-bind (second minute hour day month year dow dst-p tz) (decode-universal-time universal-time)
  67. (declare (ignore second minute hour dow dst-p tz))
  68. (destructuring-bind (&optional (day-rule '(nil 1 31)) (month-rule '(nil 1 12)) (year-rule '(nil 1900)))
  69. schedule
  70. (multiple-value-bind (next-day of-day) (next (add day (not dir))
  71. (clamp-day-rule day-rule month year))
  72. (multiple-value-bind (next-month of-month) (next (add month of-day) month-rule)
  73. (multiple-value-bind (next-year of-year) (next (add year of-month) year-rule)
  74. (unless of-year
  75. (let ((next-month (if (= next-year year) next-month (reset-rule month-rule))))
  76. (encode-universal-time
  77. 0 0 0
  78. (if (and (= next-month month) (= next-year year)) next-day
  79. (reset-rule (clamp-day-rule day-rule next-month next-year)))
  80. next-month
  81. next-year))))))))))
  82. (defstruct saver/payment name amount schedule started)
  83. (defun saver/payment-income-p (payment)
  84. (> (saver/payment-amount payment) 0))
  85. (defun saver/payment-next-time (payment moment &optional (dir t))
  86. (%saver/get-next-time (%saver/parse-schedule (saver/payment-schedule payment)) moment dir))
  87. (defun saver/payment-count-events (payment from to)
  88. (loop for ut = (saver/payment-next-time payment from) then (saver/payment-next-time payment (+ ut 86400))
  89. while (< ut to)
  90. counting ut))
  91. (defun saver/find-next-payment (payments &optional (moment (get-universal-time)))
  92. (loop for payment in payments
  93. for next-time = (saver/payment-next-time payment moment)
  94. with closest-time and closest-payment
  95. when (and next-time
  96. (or (null closest-time)
  97. (< next-time closest-time)))
  98. do (setf closest-time next-time
  99. closest-payment payment)
  100. finally (return closest-payment)))
  101. (defun saver/get-expense-info (payment incomes &optional (moment (get-universal-time)))
  102. (let* ((next-payment (saver/payment-next-time payment moment))
  103. (prev-payment (max (saver/payment-next-time payment moment nil)
  104. (or (saver/payment-started payment) 0))))
  105. (when next-payment
  106. (multiple-value-bind (total-periods total-income)
  107. (loop for income in incomes
  108. for periods = (saver/payment-count-events income prev-payment next-payment)
  109. summing periods into total-periods
  110. summing (* periods (saver/payment-amount income)) into total-income
  111. finally (return (values total-periods total-income)))
  112. (let ((payment-income-fracture (/ (saver/payment-amount payment) total-income -1)))
  113. (loop for income in incomes
  114. for periods = (saver/payment-count-events income prev-payment moment)
  115. summing periods into saved-periods
  116. summing (floor (* periods (saver/payment-amount income)
  117. payment-income-fracture)) into saved-amount
  118. finally (return
  119. (list :next-payment next-payment
  120. :prev-payment prev-payment
  121. :total-periods total-periods
  122. :total-income total-income
  123. :payment-income-fracture payment-income-fracture
  124. :saved-periods saved-periods
  125. :saved-amount (if (= total-periods saved-periods)
  126. (* -1 (saver/payment-amount payment))
  127. saved-amount)
  128. :left-periods (- total-periods saved-periods)
  129. :left-amount (if (= total-periods saved-periods) 0
  130. (- (- 0 (saver/payment-amount payment))
  131. saved-amount))))))))))
  132. (defun saver/get-income-info (income payments &optional (moment (get-universal-time)))
  133. (let ((incomes (remove-if-not #'saver/payment-income-p payments))
  134. (expenses (remove-if #'saver/payment-income-p payments)))
  135. (loop for payment in expenses
  136. for cur-info = (saver/get-expense-info payment incomes moment)
  137. for nxt-info = (saver/get-expense-info payment incomes (saver/payment-next-time income moment))
  138. summing (getf cur-info :saved-amount) into cur-saved
  139. summing (getf cur-info :left-amount) into cur-left
  140. summing (if (> (getf nxt-info :left-periods) 1)
  141. (floor (* (saver/payment-amount income) (getf cur-info :payment-income-fracture)))
  142. (getf nxt-info :left-amount)) into next-save
  143. finally (return
  144. (list :cur-saved cur-saved
  145. :cur-left cur-left
  146. :next-save next-save)))))
  147. (defun %saver/format-time (universal-time)
  148. (when universal-time
  149. (multiple-value-bind (sec min hour day month year dow dst-p tz)
  150. (decode-universal-time universal-time)
  151. (declare (ignore sec min hour dow dst-p tz))
  152. (format nil "~4,'0D-~2,'0D-~2,'0D" year month day))))
  153. ;; Database
  154. (def-db-init
  155. (db-execute "create table if not exists saver_payments (chat_id, name, amount, schedule, started, notified)")
  156. (db-execute "create unique index if not exists saver_payments_chat_id_name_idx on saver_payments (chat_id, name)"))
  157. (defun %db/saver/make-payment (row)
  158. (when row
  159. (make-saver/payment :name (nth 0 row)
  160. :amount (nth 1 row)
  161. :schedule (nth 2 row)
  162. :started (nth 3 row))))
  163. (defun db/saver/get-payments (chat-id)
  164. (mapcar #'%db/saver/make-payment
  165. (db-select "select name, amount, schedule, started from saver_payments where chat_id = ? order by amount > 0, started" chat-id)))
  166. (defun db/saver/add-payment (chat-id payment)
  167. (with-slots (name amount schedule started) payment
  168. (db-execute "insert into saver_payments (chat_id, name, amount, schedule, started) values (?, ?, ?, ?, ?)"
  169. chat-id name amount schedule started)))
  170. (defun db/saver/del-payment (chat-id name)
  171. (db-execute "delete from saver_payments where chat_id = ? and name = ?" chat-id name))
  172. (defun db/saver/get-not-notified-payments (notified)
  173. (loop for row in (db-select "select name, amount, schedule, started, chat_id from saver_payments where notified is null or notified != ?" (%saver/format-time notified))
  174. collect (cons (nth 4 row) (%db/saver/make-payment row))))
  175. (defun db/saver/set-payment-notified (chat-id name moment)
  176. (db-execute "update saver_payments set notified = ? where chat_id = ? and name = ?"
  177. (%saver/format-time moment) chat-id name))
  178. ;; Cron
  179. (defun saver/find-today-payments (&optional (moment (get-universal-time)))
  180. (remove-if-not (lambda (p) (string= (%saver/format-time moment)
  181. (%saver/format-time
  182. (saver/payment-next-time p moment))))
  183. (db/saver/get-not-notified-payments moment)
  184. :key #'cdr))
  185. (defun %saver/format-expense-notification (expense moment)
  186. (format nil "'~A' надо оплатить *~A* на сумму _~$_"
  187. (saver/payment-name expense)
  188. (%saver/format-time (saver/payment-next-time expense moment))
  189. (/ (saver/payment-amount expense) -100)))
  190. (defun %saver/format-income-notification (income payments moment)
  191. (let ((income-info (saver/get-income-info income payments moment)))
  192. (format nil "*~A* ~A. Отложи _~$_!"
  193. (%saver/format-time (saver/payment-next-time income moment))
  194. (saver/payment-name income)
  195. (/ (getf income-info :next-save) 100))))
  196. (defun %saver/is-ok-to-notify (chat-id &optional (moment (get-universal-time)))
  197. (let ((tz (or (alexandria:when-let
  198. (chat-loc (aget chat-id (and (boundp '*chat-locations*) (symbol-value '*chat-locations*))))
  199. (round (- 7.5 (aget "latitude" chat-loc)) 15)) ;; Nautical time
  200. *saver-default-timezone*)))
  201. (>= (nth 2 (multiple-value-list (decode-universal-time moment tz)))
  202. *saver-notify-hour*)))
  203. (defcron process-saver (:hour '*)
  204. (let ((moment (get-universal-time)))
  205. (loop for (chat-id . payment) in (saver/find-today-payments moment)
  206. when (%saver/is-ok-to-notify chat-id moment)
  207. do (let ((payments (db/saver/get-payments chat-id)))
  208. (db-transaction
  209. (bot-send-message chat-id
  210. (if (saver/payment-income-p payment)
  211. (%saver/format-income-notification payment payments moment)
  212. (%saver/format-expense-notification payment moment))
  213. :parse-mode "markdown")
  214. (db/saver/set-payment-notified chat-id (saver/payment-name payment) moment))))))
  215. ;; Bot subcommands
  216. (defun %saver/format-info (payments &optional (moment (get-universal-time)))
  217. (let* ((expenses (remove-if #'saver/payment-income-p payments))
  218. (incomes (remove-if-not #'saver/payment-income-p payments))
  219. (expenses-info
  220. (loop for payment in expenses
  221. for idx from 1
  222. for info = (saver/get-expense-info payment incomes moment)
  223. when info
  224. collect (format nil "~D) ~A по [[~A]]: накоплено _~$_ из _~$_, осталось _~$_ к *~A* за ~A платежа"
  225. idx
  226. (saver/payment-name payment)
  227. (saver/payment-schedule payment)
  228. (/ (getf info :saved-amount) 100)
  229. (/ (saver/payment-amount payment) -100)
  230. (/ (getf info :left-amount) 100)
  231. (%saver/format-time (saver/payment-next-time payment moment))
  232. (getf info :left-periods))))
  233. (closest-expense (saver/find-next-payment expenses moment))
  234. (incomes-info
  235. (loop for payment in incomes
  236. for idx from (1+ (length expenses))
  237. for info = (saver/get-income-info payment payments moment)
  238. when info
  239. collect (format nil "~D) ~A по [[~A]]: *~A*, отложить _~$_ из _~$_"
  240. idx
  241. (saver/payment-name payment)
  242. (saver/payment-schedule payment)
  243. (%saver/format-time (saver/payment-next-time payment moment))
  244. (/ (getf info :next-save) 100)
  245. (/ (saver/payment-amount payment) 100))))
  246. (closest-income (saver/find-next-payment incomes moment))
  247. (closest-income-info (and closest-income (saver/get-income-info closest-income payments moment))))
  248. (if (or expenses-info incomes-info)
  249. (format nil "*Платежи*~%~{~A~^~%~}~%*Следующий платёж*~%'~A': *~A*, заплатить _~$_~%~%*Накопления*~%~{~A~^~%~}~%*Следующее накопление*~%'~A': *~A*, отложить _~$_~%~%*Накоплено должно быть* _~$_"
  250. expenses-info
  251. (and closest-expense (saver/payment-name closest-expense))
  252. (and closest-expense (%saver/format-time (saver/payment-next-time closest-expense moment)))
  253. (and closest-expense (/ (saver/payment-amount closest-expense) -100))
  254. incomes-info
  255. (and closest-income (saver/payment-name closest-income))
  256. (and closest-income (%saver/format-time (saver/payment-next-time closest-income moment)))
  257. (and closest-income-info (/ (getf closest-income-info :next-save) 100))
  258. (and closest-income-info (/ (getf closest-income-info :cur-saved) 100)))
  259. "Нет активных платежей.")))
  260. (defun saver/send-info (chat-id)
  261. (let ((payments (db/saver/get-payments chat-id)))
  262. (if (find t payments :key #'saver/payment-income-p)
  263. (bot-send-message chat-id (%saver/format-info payments) :parse-mode "markdown")
  264. (bot-send-message chat-id (format nil "Нет поступлений, ничего не посчитать :( /saver add ...")))))
  265. (defparameter +saver/add-scanner+ (cl-ppcre:create-scanner "^(.+?) (-?\\d+(?:\\.\\d*)?) ((?:\\d+(?:,\\d+)*(?:-\\d+)?|\\*)(?:\\/\\d+)?(?: (?:\\d+(?:,\\d+)*(?:-\\d+)?|\\*)(?:\\/\\d+)?){0,2})(?: (\\d{4}-\\d{2}-\\d{2}))?$"))
  266. (defun saver/add-payment (chat-id args)
  267. (multiple-value-bind (matched groups) (cl-ppcre:scan-to-strings +saver/add-scanner+ (spaced args))
  268. (if matched
  269. (let ((payment (make-saver/payment :name (elt groups 0)
  270. :amount (round (* 100 (read-from-string (elt groups 1))))
  271. :schedule (elt groups 2)
  272. :started (if (elt groups 3)
  273. (destructuring-bind (year month day)
  274. (mapcar #'parse-integer
  275. (split-sequence:split-sequence
  276. #\- (elt groups 3)))
  277. (encode-universal-time 0 0 0 day month year))
  278. (get-universal-time)))))
  279. (saver/payment-next-time payment (get-universal-time))
  280. (handler-case
  281. (db/saver/add-payment chat-id payment)
  282. (error () (send-response chat-id (format nil "Платёж '~A' уже есть!"
  283. (saver/payment-name payment)))))
  284. (saver/send-info chat-id))
  285. (send-response chat-id "Неправильно. /saver add <title> <amount> <cron> [started]"))))
  286. (defun saver/del-payment (chat-id args)
  287. (handler-case
  288. (let* ((payments (db/saver/get-payments chat-id))
  289. (payment (elt payments (1- (parse-integer (car args)))))
  290. (incomes (remove-if-not #'saver/payment-income-p payments)))
  291. (db/saver/del-payment chat-id (saver/payment-name payment))
  292. (bot-send-message chat-id
  293. (format nil "'~A' удалил.~@[ Забрать _~$_ из накопленого.~]"
  294. (saver/payment-name payment)
  295. (and (not (saver/payment-income-p payment))
  296. incomes
  297. (/ (getf (saver/get-expense-info payment incomes) :saved-amount)
  298. 100)))
  299. :parse-mode "markdown"))
  300. (error (e) (send-response chat-id (format nil "/saver del <idx> [~A]" e)))))
  301. ;; Hooks
  302. (def-message-cmd-handler handler-cmd-save (:save :saver)
  303. (if (null args)
  304. (saver/send-info chat-id)
  305. (case (keyify (car args))
  306. (:add (saver/add-payment chat-id (rest args)))
  307. (:del (saver/del-payment chat-id (rest args)))
  308. (t (send-response chat-id "Надо /saver add ... или /saver del <idx>")))))