urbit/pkg/arvo/sys/vane/behn.hoon

385 lines
10 KiB
Plaintext
Raw Normal View History

:: %behn, just a timer
!:
2016-11-24 07:25:07 +03:00
!? 164
::
=, behn
2020-12-06 11:38:37 +03:00
|= our=ship
=> |%
2020-12-08 03:47:06 +03:00
+$ move [p=duct q=(wite note gift)]
+$ note :: out request $->
$~ [%b %wait *@da] ::
$% $: %b :: to self
2020-12-08 03:47:06 +03:00
$>(%wait task) :: set timer
== ::
$: %d :: to %dill
2020-12-08 03:47:06 +03:00
$>(%flog task:dill) :: log output
== == ::
+$ sign
2020-12-08 03:22:26 +03:00
$~ [%behn %wake ~]
$% [%behn $>(%wake gift)]
==
::
+$ behn-state
2020-05-27 09:59:07 +03:00
$: %2
timers=(tree [key=@da val=(qeu duct)])
unix-duct=duct
next-wake=(unit @da)
2019-05-01 00:58:51 +03:00
drips=drip-manager
==
::
2020-05-27 09:59:07 +03:00
++ timer-map ((ordered-map ,@da ,(qeu duct)) lte)
2020-05-20 07:40:39 +03:00
::
2019-05-01 00:58:51 +03:00
+$ drip-manager
$: count=@ud
movs=(map @ud vase)
==
::
+$ timer [date=@da =duct]
--
2016-11-24 07:25:07 +03:00
::
=>
~% %behn ..part ~
|%
2019-01-30 04:15:54 +03:00
++ per-event
=| moves=(list move)
2020-12-06 11:38:37 +03:00
|= [[now=@da =duct] state=behn-state]
2018-12-03 22:45:50 +03:00
::
2019-01-30 04:15:54 +03:00
|%
:: %entry-points
::
2019-02-12 02:34:08 +03:00
:: +born: urbit restarted; refresh :next-wake and store wakeup timer duct
2019-01-30 04:15:54 +03:00
::
2019-02-12 02:34:08 +03:00
++ born set-unix-wake(next-wake.state ~, unix-duct.state duct)
:: +crud: handle failure of previous arvo event
::
2019-01-30 04:15:54 +03:00
++ crud
|= [tag=@tas error=tang]
2019-01-30 04:15:54 +03:00
^+ [moves state]
:: behn must get activated before other vanes in a %wake
::
?. =(%wake tag)
~& %behn-crud-not-wake^tag
[[duct %slip %d %flog %crud tag error]~ state]
::
?: =(~ timers.state)
~|(%behn-crud-no-timer^tag^error !!)
::
(wake `error)
2019-02-12 02:34:08 +03:00
:: +rest: cancel the timer at :date, then adjust unix wakeup
:: +wait: set a new timer at :date, then adjust unix wakeup
2019-01-30 04:15:54 +03:00
::
2019-02-12 02:34:08 +03:00
++ rest |=(date=@da set-unix-wake(timers.state (unset-timer [date duct])))
++ wait |=(date=@da set-unix-wake(timers.state (set-timer [date duct])))
:: +huck: give back immediately
::
:: Useful if you want to continue working after other moves finish.
::
++ huck
2020-07-08 09:08:27 +03:00
|= syn=sign-arvo
=< [moves state]
2020-07-08 09:08:27 +03:00
event-core(moves [duct %give %heck syn]~)
2019-05-01 00:58:51 +03:00
:: +drip: XX
::
++ drip
|= mov=vase
=< [moves state]
^+ event-core
=. moves
[duct %pass /drip/(scot %ud count.drips.state) %b %wait +(now)]~
=. movs.drips.state
(~(put by movs.drips.state) count.drips.state mov)
=. count.drips.state +(count.drips.state)
event-core
:: +take-drip: XX
::
++ take-drip
|= [num=@ud error=(unit tang)]
=< [moves state]
^+ event-core
=/ drip (~(got by movs.drips.state) num)
=. movs.drips.state (~(del by movs.drips.state) num)
=/ =move
2020-02-27 02:57:59 +03:00
=/ card [%give %meta drip]
?~ error
2020-02-27 02:57:59 +03:00
[duct card]
=/ =tang
(weld u.error `tang`[leaf/"drip failed" ~])
:: XX should be
:: [duct %hurl fail/tang card]
::
[duct %pass /drip-slog %d %flog %crud %drip-fail tang]
event-core(moves [move moves])
:: +trim: in response to memory pressue
::
++ trim [moves state]
:: +vega: learn of a kernel upgrade
::
2019-02-12 02:34:08 +03:00
++ vega [moves state]
:: +wake: unix says wake up; process the elapsed timer and set :next-wake
2019-01-30 04:15:54 +03:00
::
++ wake
|= error=(unit tang)
2019-01-30 04:15:54 +03:00
^+ [moves state]
=. next-wake.state ~
2019-07-31 02:11:19 +03:00
:: no-op on spurious but innocuous unix wakeups
2019-02-12 02:34:08 +03:00
::
2020-05-20 07:40:39 +03:00
?: =(~ timers.state)
2019-07-31 02:11:19 +03:00
~? ?=(^ error) %behn-wake-no-timer^u.error
[moves state]
:: if we errored, pop the timer and notify the client vane of the error
::
?^ error
=< set-unix-wake
2020-05-27 09:59:07 +03:00
=^ =timer timers.state pop-timer
2020-05-20 07:40:39 +03:00
(emit-vane-wake duct.timer error)
2019-02-12 02:34:08 +03:00
:: if unix woke us too early, retry by resetting the unix wakeup timer
::
2020-05-27 09:59:07 +03:00
=/ [=timer later-timers=_timers.state] pop-timer
2020-05-20 07:40:39 +03:00
?: (gth date.timer now)
set-unix-wake
2019-02-12 02:34:08 +03:00
:: pop first timer, tell vane it has elapsed, and adjust next unix wakeup
::
=< set-unix-wake
2020-05-27 09:59:07 +03:00
(emit-vane-wake(timers.state later-timers) duct.timer ~)
2019-01-30 04:15:54 +03:00
:: %utilities
::
::+|
::
++ event-core .
2019-02-12 02:34:08 +03:00
:: +emit-vane-wake: produce a move to wake a vane; assumes no prior moves
::
++ emit-vane-wake
|= [=^duct error=(unit tang)]
event-core(moves [duct %give %wake error]~)
2019-02-12 02:34:08 +03:00
:: +emit-doze: set new unix wakeup timer in state and emit move to unix
::
2019-01-08 00:47:23 +03:00
:: We prepend the unix %doze event so that it is handled first. Arvo must
:: handle this first because the moves %behn emits will get handled in
:: depth-first order. If we're handling a %wake which causes a move to a
2019-01-08 00:47:23 +03:00
:: different vane and a %doze event to send to unix, Arvo needs to process
2019-01-08 00:58:08 +03:00
:: the %doze first because otherwise if the move to the other vane calls
:: back into %behn and emits a second %doze, the second %doze would be
:: handled by unix first which is incorrect.
::
2019-02-12 02:34:08 +03:00
++ emit-doze
|= =date=(unit @da)
^+ event-core
:: no-op if .unix-duct has not yet been set
::
?~ unix-duct.state
event-core
2019-02-12 02:34:08 +03:00
:: make sure we don't try to wake up in the past
::
=? date-unit ?=(^ date-unit) `(max now u.date-unit)
::
%_ event-core
next-wake.state date-unit
moves [[unix-duct.state %give %doze date-unit] moves]
==
:: +set-unix-wake: set or unset next unix wakeup timer based on :i.timers
::
++ set-unix-wake
=< [moves state]
~% %set-unix-wake ..part ~ |-
2019-02-12 02:34:08 +03:00
^+ event-core
2019-01-30 04:15:54 +03:00
::
=* next-wake next-wake.state
=* timers timers.state
:: if no timers, cancel existing wakeup timer or no-op
::
=/ first=(unit [date=@da *]) (pry:timer-map timers.state)
2020-05-27 09:59:07 +03:00
?~ first
?~ next-wake
2019-02-12 02:34:08 +03:00
event-core
(emit-doze ~)
:: if :next-wake is in the past or not soon enough, reset it
::
?^ next-wake
2020-05-27 09:59:07 +03:00
?: &((gte date.u.first u.next-wake) (lte now u.next-wake))
2019-02-12 02:34:08 +03:00
event-core
2020-05-27 09:59:07 +03:00
(emit-doze `date.u.first)
:: there was no unix wakeup timer; set one
::
2020-05-27 09:59:07 +03:00
(emit-doze `date.u.first)
:: +pop-timer: dequeue and produce earliest timer
::
++ pop-timer
^+ [*timer timers.state]
=^ [date=@da dux=(qeu ^duct)] timers.state (pop:timer-map timers.state)
=^ dut dux ~(get to dux)
:- [date dut]
?: =(~ dux)
timers.state
2020-05-27 09:59:07 +03:00
(put:timer-map timers.state date dux)
:: +set-timer: set a timer, maintaining order
2019-01-30 04:15:54 +03:00
::
++ set-timer
~% %set-timer ..part ~
2019-01-30 04:15:54 +03:00
|= t=timer
2020-05-20 07:40:39 +03:00
^+ timers.state
2020-05-27 09:59:07 +03:00
=/ found (find-ducts date.t)
(put:timer-map timers.state date.t (~(put to found) duct.t))
:: +find-ducts: get timers at date
::
:: TODO: move to +ordered-map
::
++ find-ducts
|= date=@da
^- (qeu ^duct)
?~ timers.state ~
?: =(date key.n.timers.state)
val.n.timers.state
?: (lte date key.n.timers.state)
$(timers.state l.timers.state)
$(timers.state r.timers.state)
2019-01-30 04:15:54 +03:00
:: +unset-timer: cancel a timer; if it already expired, no-op
::
++ unset-timer
2019-02-12 02:34:08 +03:00
|= t=timer
2020-05-20 07:40:39 +03:00
^+ timers.state
2020-05-27 09:59:07 +03:00
=/ [found=? dux=(qeu ^duct)]
=/ dux (find-ducts date.t)
|- ^- [found=? dux=(qeu ^duct)]
?~ dux |+~
?: =(duct.t n.dux) &+~(nip to `(qeu ^duct)`dux)
=^ found-left=? l.dux $(dux l.dux)
2020-05-27 09:59:07 +03:00
?: found-left &+dux
=^ found-rite=? r.dux $(dux r.dux)
[found-rite dux]
2020-05-27 09:59:07 +03:00
?. found timers.state
?: =(~ dux)
+:(del:timer-map timers.state date.t)
(put:timer-map timers.state date.t dux)
--
2019-01-30 04:15:54 +03:00
--
::
=| behn-state
=* state -
2020-12-06 11:38:37 +03:00
|= [now=@da eny=@uvJ rof=roof]
2019-01-30 04:15:54 +03:00
=* behn-gate .
^?
|%
2020-12-08 03:47:06 +03:00
:: +call: handle a +task:behn request
2019-01-30 04:15:54 +03:00
::
++ call
~% %behn-call ..part ~
2019-01-30 04:15:54 +03:00
|= $: hen=duct
2020-02-11 01:03:03 +03:00
dud=(unit goof)
2020-12-08 03:47:06 +03:00
wrapped-task=(hobo task)
2019-01-30 04:15:54 +03:00
==
^- [(list move) _behn-gate]
::
2020-12-08 03:47:06 +03:00
=/ =task ((harden task) wrapped-task)
2020-12-06 11:38:37 +03:00
=/ event-core (per-event [now hen] state)
2019-01-30 04:15:54 +03:00
::
=^ moves state
2020-12-08 05:01:48 +03:00
::
:: handle error notifications
::
?^ dud
(crud:event-core -.task tang.u.dud)
::
2019-01-30 04:15:54 +03:00
?- -.task
%born born:event-core
%rest (rest:event-core date=p.task)
2019-05-01 00:58:51 +03:00
%drip (drip:event-core move=p.task)
2020-07-08 09:08:27 +03:00
%huck (huck:event-core syn.task)
%trim trim:event-core
%vega vega:event-core
2019-01-30 04:15:54 +03:00
%wait (wait:event-core date=p.task)
%wake (wake:event-core error=~)
2019-01-30 04:15:54 +03:00
==
[moves behn-gate]
:: +load: migrate an old state to a new behn version
2016-11-24 07:25:07 +03:00
::
++ load
|= old=behn-state
2019-01-30 04:15:54 +03:00
^+ behn-gate
2019-08-01 21:37:28 +03:00
behn-gate(state old)
:: +scry: view timer state
::
:: TODO: not referentially transparent w.r.t. elapsed timers,
:: which might or might not show up in the product
2016-11-24 07:25:07 +03:00
::
++ scry
2020-12-08 00:52:12 +03:00
^- roon
|= [lyc=gang car=term bem=beam]
2016-11-24 07:25:07 +03:00
^- (unit (unit cage))
2020-12-08 00:52:12 +03:00
=* ren car
2020-11-24 00:06:50 +03:00
=* why=shop &/p.bem
=* syd q.bem
=* lot=coin $/r.bem
=* tyl s.bem
::
::TODO don't special-case whey scry
::
2020-05-07 11:51:08 +03:00
?: &(=(ren %$) =(tyl /whey))
=/ maz=(list mass)
:~ timers+&+timers.state
==
``mass+!>(maz)
:: only respond for the local identity, %$ desk, current timestamp
::
?. ?& =(&+our why)
=([%$ %da now] lot)
=(%$ syd)
==
~
:: /bx/debug/timers (list [@da duct]) all timers and their ducts
:: /bx/timers (list @da) all timer timestamps
:: /bx/timers/next (unit @da) the very next timer to fire
:: /bx/timers/[da] (list @da) all timers up to and including da
::
?. ?=(%x ren) ~
?+ tyl [~ ~]
[%debug %timers ~]
:^ ~ ~ %noun
!> ^- (list [@da duct])
2020-05-27 09:59:07 +03:00
%- zing
%+ turn (tap:timer-map timers)
|= [date=@da q=(qeu duct)]
%+ turn ~(tap to q)
|=(d=duct [date d])
::
[%timers ~]
:^ ~ ~ %noun
!> ^- (list @da)
%- zing
%+ turn (tap:timer-map timers)
|= [date=@da q=(qeu duct)]
(reap ~(wyt in q) date)
::
[%timers %next ~]
:^ ~ ~ %noun
!> ^- (unit @da)
(bind (pry:timer-map timers) head)
::
[%timers @ ~]
?~ til=(slaw %da i.t.tyl)
[~ ~]
:^ ~ ~ %noun
!> ^- (list @da)
2020-06-25 20:25:44 +03:00
=/ tiz=(list [date=@da q=(qeu duct)])
(tap:timer-map timers)
|- ^- (list @da)
?~ tiz ~
?: (gth date.i.tiz u.til) ~
%+ weld
(reap ~(wyt in q.i.tiz) date.i.tiz)
$(tiz t.tiz)
==
2016-11-24 07:25:07 +03:00
::
++ stay state
2019-01-30 04:15:54 +03:00
++ take
2020-12-06 11:38:37 +03:00
|= [tea=wire hen=duct dud=(unit goof) hin=sign]
2019-01-30 04:15:54 +03:00
^- [(list move) _behn-gate]
?^ dud
~|(%behn-take-dud (mean tang.u.dud))
::
2019-05-01 00:58:51 +03:00
?> ?=([%drip @ ~] tea)
2020-12-06 11:38:37 +03:00
=/ event-core (per-event [now hen] state)
2019-05-01 00:58:51 +03:00
=^ moves state
2020-12-06 11:38:37 +03:00
(take-drip:event-core (slav %ud i.t.tea) error.hin)
2019-05-01 00:58:51 +03:00
[moves behn-gate]
2016-11-24 07:25:07 +03:00
--