:: hark-graph-hook: notifications for graph-store [landscape] :: /- post, group-store, metadata-store, hook=hark-graph-hook /+ resource, metadata, default-agent, dbug, graph-store, graph, grouplib=group, store=hark-store :: :: ~% %hark-graph-hook-top ..part ~ |% +$ card card:agent:gall +$ versioned-state $% state-0 == :: +$ state-0 $: %0 watching=(set [resource index:post]) mentions=_& watch-on-self=_& == :: +$ notif-kind [name=@t parent-lent=@ud mode=?(%each %count) watch=?] :: ++ scry |* [[our=@p now=@da] =mold p=path] ?> ?=(^ p) ?> ?=(^ t.p) .^(mold i.p (scot %p our) i.t.p (scot %da now) t.t.p) :: ++ scry-conversion |= [[our=@p now=@da] desk=term =mark] ~+ %^ scry [our now] tube:clay /cc/[desk]/[mark]/notification-kind :: -- :: =| state-0 =* state - :: =< %- agent:dbug ^- agent:gall ~% %hark-graph-hook-agent ..card ~ |_ =bowl:gall +* this . ha ~(. +> bowl) def ~(. (default-agent this %|) bowl) met ~(. metadata bowl) grp ~(. grouplib bowl) gra ~(. graph bowl) :: ++ on-init :_ this ~[watch-graph:ha] :: ++ on-save !>(state) ++ on-load |= old=vase ^- (quip card _this) :_ this(state !<(state-0 old)) %+ welp ?: (~(has by wex.bowl) [/graph our.bowl %graph-store]) ~ ~[watch-graph:ha] %+ turn ^- (list mark) :~ %graph-validator-chat %graph-validator-link %graph-validator-publish == |= =mark ^- card =/ =wire /validator/[mark] =/ =rave:clay [%sing %c [%da now.bowl] /[mark]/notification-kind] [%pass wire %arvo %c %warp our.bowl [%home `rave]] :: ++ on-watch |= =path ^- (quip card _this) =^ cards state ?+ path (on-watch:def path) :: [%updates ~] :_ state %+ give:ha ~ :* %initial watching mentions watch-on-self == == [cards this] :: ++ on-poke ~/ %hark-graph-hook-poke |= [=mark =vase] ^- (quip card _this) |^ ?> (team:title our.bowl src.bowl) =^ cards state ?+ mark (on-poke:def mark vase) %hark-graph-hook-action (hark-graph-hook-action !<(action:hook vase)) == [cards this] :: ++ hark-graph-hook-action |= =action:hook ^- (quip card _state) |^ :- (give:ha ~[/updates] action) ?- -.action %listen (listen +.action) %ignore (ignore +.action) %set-mentions (set-mentions +.action) %set-watch-on-self (set-watch-on-self +.action) == ++ listen |= [graph=resource =index:post] ^+ state state(watching (~(put in watching) [graph index])) :: ++ ignore |= [graph=resource =index:post] ^+ state state(watching (~(del in watching) [graph index])) :: ++ set-mentions |= ment=? ^+ state state(mentions ment) :: ++ set-watch-on-self |= self=? ^+ state state(watch-on-self self) -- -- :: ++ on-agent ~/ %hark-graph-hook-agent |= [=wire =sign:agent:gall] ^- (quip card _this) |^ ?+ -.sign (on-agent:def wire sign) %kick :_ this ?. ?=([%graph ~] wire) ~ ~[watch-graph:ha] :: %fact ?. ?=(%graph-update p.cage.sign) (on-agent:def wire sign) =^ cards state (graph-update !<(update:graph-store q.cage.sign)) [cards this] == ++ add-graph |= [rid=resource =graph:graph-store] ^- (quip card _state) =/ group-rid=(unit resource) (group-from-app-resource:met %graph rid) ?~ group-rid ~& no-group+rid `state =/ is-hidden=? !(is-managed:grp u.group-rid) =/ should-watch |(is-hidden &(watch-on-self =(our.bowl entity.rid))) ?. should-watch `state :- (give:ha ~[/updates] %listen [rid ~]) state(watching (~(put in watching) [rid ~])) :: ++ graph-update |= =update:graph-store ^- (quip card _state) ?: ?=(%add-graph -.q.update) (add-graph resource.q.update graph.q.update) ?. ?=(%add-nodes -.q.update) [~ state] =* rid resource.q.update (check-nodes ~(val by nodes.q.update) rid) :: ++ check-nodes |= $: nodes=(list node:graph-store) rid=resource == =/ group=(unit resource) (group-from-app-resource:met %graph rid) ?~ group `state =/ metadata=(unit metadata:metadata-store) (peek-metadata:met %graph u.group rid) ?~ metadata `state abet:check:(abed:handle-update:ha rid nodes u.group module.u.metadata) -- :: ++ on-peek on-peek:def :: ++ on-leave on-leave:def ++ on-arvo |= [=wire =sign-arvo] ^- (quip card _this) ?+ wire (on-arvo:def wire sign-arvo) :: [%validator @ ~] :_ this =* validator i.t.wire =/ =rave:clay [%next %c [%da now.bowl] /[validator]/notification-kind] [%pass wire %arvo %c %warp our.bowl [%home `rave]]~ == ++ on-fail on-fail:def -- :: |_ =bowl:gall :: :: ++ give |= [paths=(list path) =update:hook] ^- (list card) [%give %fact paths hark-graph-hook-update+!>(update)]~ :: ++ watch-graph ^- card [%pass /graph %agent [our.bowl %graph-store] %watch /updates] :: ++ poke-hark |= =action:store ^- card =- [%pass / %agent [our.bowl %hark-store] %poke -] hark-action+!>(action) :: ++ is-mention |= contents=(list content:post) ^- ? ?. mentions %.n ?~ contents %.n ?. ?=(%mention -.i.contents) $(contents t.contents) ?: =(our.bowl ship.i.contents) %.y $(contents t.contents) :: ++ handle-update |_ $: rid=resource :: input updates=(list node:graph-store) group=resource module=term hark-pokes=(list action:store) :: output new-watches=(list index:graph-store) == ++ update-core . :: ++ abed |= [r=resource upds=(list node:graph-store) grp=resource mod=term] update-core(rid r, updates upds, group grp, module mod) :: ++ get-conversion ^- tube:clay =+ %^ scry [our now]:bowl ,mark=(unit mark) /gx/graph-store/graph-mark/(scot %p entity.rid)/[name.rid]/noun ?~ mark |=(v=vase !>(~)) (scry-conversion [our now]:bowl q.byk.bowl u.mark) :: ++ abet ^- (quip card _state) :_ state(watching (~(uni in watching) (silt (turn new-watches (lead rid))))) ^- (list card) %+ welp (turn (flop hark-pokes) poke-hark) %- zing %+ turn (flop new-watches) |=(=index:graph-store (give ~[/updates] [%listen rid index])) :: ++ hark |= =action:store ^+ update-core update-core(hark-pokes [action hark-pokes]) :: ++ new-watch |= =index:graph-store update-core(new-watches [index new-watches]) :: ++ check |- ^+ update-core ?~ updates update-core =/ core=_update-core (check-node i.updates) =. updates.core t.updates $(update-core core) :: ++ check-node-children |= =node:graph-store ^+ update-core ?: ?=(%empty -.children.node) update-core =/ children=(list [=atom =node:graph-store]) (tap:orm:graph-store p.children.node) |- ^+ update-core ?~ children update-core =. update-core (check-node node.i.children) $(children t.children) :: ++ check-node |= =node:graph-store ^+ update-core =. update-core (check-node-children node) =+ !< notif-kind=(unit notif-kind) (get-conversion !>([0 post.node])) ?~ notif-kind update-core =/ desc=@t ?: (is-mention contents.post.node) %mention name.u.notif-kind =* not-kind u.notif-kind =/ parent=index:post (scag parent-lent.not-kind index.post.node) =/ notif-index=index:store [%graph group rid module desc parent] ?: =(our.bowl author.post.node) (self-post node notif-index [mode watch]:not-kind) =. update-core (update-unread-count not-kind notif-index [time-sent index]:post.node) =? update-core ?| =(desc %mention) (~(has in watching) [rid parent]) == =/ =contents:store [%graph (limo post.node ~)] (add-unread notif-index [time-sent.post.node %.n contents]) update-core :: ++ update-unread-count |= [=notif-kind =index:store time=@da ref=index:graph-store] =. description.index name.notif-kind =/ =stats-index:store (to-stats-index:store index) ?- mode.notif-kind %count (hark %unread-count stats-index time) %each (hark %unread-each stats-index ref time) %none update-core == :: ++ self-post |= $: =node:graph-store =index:store mode=?(%count %each) watch=? == ^+ update-core =/ =stats-index:store (to-stats-index:store index) =. update-core (hark %seen-index time-sent.post.node stats-index) =? update-core ?=(%count mode) (hark %read-count stats-index) =? update-core &(watch watch-on-self) (new-watch index.post.node) update-core :: ++ add-unread |= [=index:store =notification:store] (hark %add-note index notification) :: -- --