Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Submit feedback
Contribute to GitLab
Sign in
Toggle navigation
haskell-gargantext
Project
Project
Details
Activity
Releases
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
Christian Merten
haskell-gargantext
Commits
e1dbfd70
Commit
e1dbfd70
authored
Aug 31, 2022
by
Przemyslaw Kaminski
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
[nodeStory] insert/update/delete state handling
This doesn't work with locks yet.
parent
9166bb01
Changes
4
Hide whitespace changes
Inline
Side-by-side
Showing
4 changed files
with
171 additions
and
89 deletions
+171
-89
0.0.6.1.sql
devops/postgres/upgrade/0.0.6.1.sql
+7
-0
Ngrams.hs
src/Gargantext/API/Ngrams.hs
+0
-9
Tools.hs
src/Gargantext/API/Ngrams/Tools.hs
+14
-12
NodeStory.hs
src/Gargantext/Core/NodeStory.hs
+150
-68
No files found.
devops/postgres/upgrade/0.0.6.1.sql
View file @
e1dbfd70
-- Start a new transaction. In case data migration goes wrong, we are
-- back to our original table.
BEGIN
;
-- we will migrate data here
-- we will migrate data here
-- rename old table and create a new one
-- rename old table and create a new one
...
@@ -56,3 +60,6 @@ INSERT INTO public.node_stories
...
@@ -56,3 +60,6 @@ INSERT INTO public.node_stories
FROM
node_stories_old
FROM
node_stories_old
CROSS
JOIN
jsonb_each
(
archive
->
'state'
->
'NgramsTerms'
)
AS
j
CROSS
JOIN
jsonb_each
(
archive
->
'state'
->
'NgramsTerms'
)
AS
j
JOIN
ngrams
ON
terms
=
j
.
key
;
JOIN
ngrams
ON
terms
=
j
.
key
;
-- finally, write out the stuff
COMMIT
;
src/Gargantext/API/Ngrams.hs
View file @
e1dbfd70
...
@@ -259,15 +259,6 @@ setListNgrams listId ngramsType ns = do
...
@@ -259,15 +259,6 @@ setListNgrams listId ngramsType ns = do
saveNodeStory
saveNodeStory
currentVersion
::
HasNodeStory
env
err
m
=>
ListId
->
m
Version
currentVersion
listId
=
do
--nls <- getRepo [listId]
pool
<-
view
connPool
nls
<-
liftBase
$
getNodeStory
pool
listId
pure
$
nls
^.
unNodeStory
.
at
listId
.
_Just
.
a_version
newNgramsFromNgramsStatePatch
::
NgramsStatePatch'
->
[
Ngrams
]
newNgramsFromNgramsStatePatch
::
NgramsStatePatch'
->
[
Ngrams
]
newNgramsFromNgramsStatePatch
p
=
newNgramsFromNgramsStatePatch
p
=
[
text2ngrams
(
unNgramsTerm
n
)
[
text2ngrams
(
unNgramsTerm
n
)
...
...
src/Gargantext/API/Ngrams/Tools.hs
View file @
e1dbfd70
...
@@ -19,6 +19,7 @@ import Control.Lens (_Just, (^.), at, view, At, Index, IxValue)
...
@@ -19,6 +19,7 @@ import Control.Lens (_Just, (^.), at, view, At, Index, IxValue)
import
Control.Monad.Reader
import
Control.Monad.Reader
import
Data.HashMap.Strict
(
HashMap
)
import
Data.HashMap.Strict
(
HashMap
)
import
Data.Hashable
(
Hashable
)
import
Data.Hashable
(
Hashable
)
import
Data.Pool
(
withResource
)
import
Data.Set
(
Set
)
import
Data.Set
(
Set
)
import
Data.Validity
import
Data.Validity
import
Gargantext.API.Ngrams.Types
import
Gargantext.API.Ngrams.Types
...
@@ -202,15 +203,16 @@ migrateFromDirToDb :: (CmdM env err m, HasNodeStory env err m)
...
@@ -202,15 +203,16 @@ migrateFromDirToDb :: (CmdM env err m, HasNodeStory env err m)
=>
m
()
=>
m
()
migrateFromDirToDb
=
do
migrateFromDirToDb
=
do
pool
<-
view
connPool
pool
<-
view
connPool
listIds
<-
liftBase
$
getNodesIdWithType
pool
NodeList
withResource
pool
$
\
c
->
do
printDebug
"[migrateFromDirToDb] listIds"
listIds
listIds
<-
liftBase
$
getNodesIdWithType
c
NodeList
(
NodeStory
nls
)
<-
NSF
.
getRepoReadConfig
listIds
printDebug
"[migrateFromDirToDb] listIds"
listIds
printDebug
"[migrateFromDirToDb] nls"
nls
(
NodeStory
nls
)
<-
NSF
.
getRepoReadConfig
listIds
_
<-
mapM
(
\
(
nId
,
a
)
->
do
printDebug
"[migrateFromDirToDb] nls"
nls
n
<-
liftBase
$
nodeExists
pool
nId
_
<-
mapM
(
\
(
nId
,
a
)
->
do
case
n
of
n
<-
liftBase
$
nodeExists
c
nId
False
->
pure
()
case
n
of
True
->
liftBase
$
upsertNodeArchive
pool
nId
a
False
->
pure
()
)
$
Map
.
toList
nls
True
->
liftBase
$
upsertNodeStories
c
nId
a
--
_
<-
nodeStoryIncs
(
Just
$
NodeStory
nls
)
listIds
)
$
Map
.
toList
nls
pure
()
--
_
<-
nodeStoryIncs
(
Just
$
NodeStory
nls
)
listIds
pure
()
src/Gargantext/Core/NodeStory.hs
View file @
e1dbfd70
...
@@ -72,11 +72,13 @@ module Gargantext.Core.NodeStory
...
@@ -72,11 +72,13 @@ module Gargantext.Core.NodeStory
,
a_version
,
a_version
,
nodeExists
,
nodeExists
,
runPGSQuery
,
runPGSQuery
,
runPGSAdvisoryXactLock
,
getNodesIdWithType
,
getNodesIdWithType
,
readNodeStoryEnv
,
readNodeStoryEnv
,
upsertNode
Archive
,
upsertNode
Stories
,
getNodeStory
,
getNodeStory
,
nodeStoriesQuery
)
,
nodeStoriesQuery
,
currentVersion
)
where
where
-- import Debug.Trace (traceShow)
-- import Debug.Trace (traceShow)
...
@@ -84,7 +86,7 @@ import Control.Debounce (mkDebounce, defaultDebounceSettings, debounceFreq, debo
...
@@ -84,7 +86,7 @@ import Control.Debounce (mkDebounce, defaultDebounceSettings, debounceFreq, debo
import
Codec.Serialise.Class
import
Codec.Serialise.Class
import
Control.Concurrent
(
MVar
(),
withMVar
,
newMVar
,
modifyMVar_
)
import
Control.Concurrent
(
MVar
(),
withMVar
,
newMVar
,
modifyMVar_
)
import
Control.Exception
(
catch
,
throw
,
SomeException
(
..
))
import
Control.Exception
(
catch
,
throw
,
SomeException
(
..
))
import
Control.Lens
(
makeLenses
,
Getter
,
(
^.
),
(
.~
),
(
%~
),
traverse
)
import
Control.Lens
(
makeLenses
,
Getter
,
(
^.
),
(
.~
),
(
%~
),
_Just
,
at
,
traverse
,
view
)
import
Control.Monad.Except
import
Control.Monad.Except
import
Control.Monad.Reader
import
Control.Monad.Reader
import
Data.Aeson
hiding
((
.=
),
decode
)
import
Data.Aeson
hiding
((
.=
),
decode
)
...
@@ -99,15 +101,17 @@ import Database.PostgreSQL.Simple.FromField (FromField(fromField), fromJSONField
...
@@ -99,15 +101,17 @@ import Database.PostgreSQL.Simple.FromField (FromField(fromField), fromJSONField
import
Data.Profunctor.Product.TH
(
makeAdaptorAndInstance
)
import
Data.Profunctor.Product.TH
(
makeAdaptorAndInstance
)
import
GHC.Generics
(
Generic
)
import
GHC.Generics
(
Generic
)
import
Gargantext.API.Ngrams.Types
import
Gargantext.API.Ngrams.Types
import
Gargantext.Core.Types
(
NodeId
(
..
),
NodeType
)
import
Gargantext.Core.Types
(
ListId
,
NodeId
(
..
),
NodeType
)
import
Gargantext.Core.Utils.Prefix
(
unPrefix
)
import
Gargantext.Core.Utils.Prefix
(
unPrefix
)
import
Gargantext.Database.Prelude
(
CmdM
'
,
HasConnectionPool
,
HasConfig
)
import
Gargantext.Database.Prelude
(
CmdM
'
,
HasConnectionPool
(
..
)
,
HasConfig
)
import
Gargantext.Database.Query.Table.Node.Error
(
HasNodeError
())
import
Gargantext.Database.Query.Table.Node.Error
(
HasNodeError
())
import
Gargantext.Prelude
import
Gargantext.Prelude
import
Opaleye
(
DefaultFromField
(
..
),
SqlJsonb
,
fromPGSFromField
)
import
Opaleye
(
DefaultFromField
(
..
),
SqlJsonb
,
fromPGSFromField
)
import
System.IO
(
stderr
)
import
System.IO
(
stderr
)
import
qualified
Data.Map.Strict
as
Map
import
qualified
Data.Map.Strict
as
Map
import
qualified
Data.Map.Strict.Patch
as
PM
import
qualified
Data.Map.Strict.Patch
as
PM
import
qualified
Data.Set
as
Set
import
qualified
Data.Text
as
Text
import
qualified
Database.PostgreSQL.Simple
as
PGS
import
qualified
Database.PostgreSQL.Simple
as
PGS
import
qualified
Gargantext.Database.Query.Table.Ngrams
as
TableNgrams
import
qualified
Gargantext.Database.Query.Table.Ngrams
as
TableNgrams
...
@@ -267,33 +271,38 @@ $(makeAdaptorAndInstance "pNodeArchiveStory" ''NodeStoryArchivePoly)
...
@@ -267,33 +271,38 @@ $(makeAdaptorAndInstance "pNodeArchiveStory" ''NodeStoryArchivePoly)
-- type NodeStoryArchiveWrite = NodeStoryArchivePoly (Column SqlInt4) (Column SqlJsonb)
-- type NodeStoryArchiveWrite = NodeStoryArchivePoly (Column SqlInt4) (Column SqlJsonb)
-- type NodeStoryArchiveRead = NodeStoryArchivePoly (Column SqlInt4) (Column SqlJsonb)
-- type NodeStoryArchiveRead = NodeStoryArchivePoly (Column SqlInt4) (Column SqlJsonb)
type
Archive
Q
=
Archive
NgramsState'
NgramsStatePatch'
type
Archive
List
=
Archive
NgramsState'
NgramsStatePatch'
-- DB stuff
-- DB stuff
runPGSExecuteMany
::
(
PGS
.
ToRow
q
)
=>
P
ool
P
GS
.
Connection
->
PGS
.
Query
->
[
q
]
->
IO
Int64
runPGSExecuteMany
::
(
PGS
.
ToRow
q
)
=>
PGS
.
Connection
->
PGS
.
Query
->
[
q
]
->
IO
Int64
runPGSExecuteMany
pool
qs
a
=
withResource
pool
$
\
c
->
catch
(
PGS
.
executeMany
c
qs
a
)
(
printError
c
)
runPGSExecuteMany
c
qs
a
=
catch
(
PGS
.
executeMany
c
qs
a
)
printError
where
where
printError
_c
(
SomeException
e
)
=
do
printError
(
SomeException
e
)
=
do
--q' <- PGS.formatQuery c qs a
--q' <- PGS.formatQuery c qs a
--hPutStrLn stderr q'
--hPutStrLn stderr q'
throw
(
SomeException
e
)
throw
(
SomeException
e
)
runPGSQuery
::
(
PGS
.
FromRow
r
,
PGS
.
ToRow
q
)
=>
P
ool
P
GS
.
Connection
->
PGS
.
Query
->
q
->
IO
[
r
]
runPGSQuery
::
(
PGS
.
FromRow
r
,
PGS
.
ToRow
q
)
=>
PGS
.
Connection
->
PGS
.
Query
->
q
->
IO
[
r
]
runPGSQuery
pool
q
a
=
withResource
pool
$
\
c
->
catch
(
PGS
.
query
c
q
a
)
(
printError
c
)
runPGSQuery
c
q
a
=
catch
(
PGS
.
query
c
q
a
)
printError
where
where
printError
c
(
SomeException
e
)
=
do
printError
(
SomeException
e
)
=
do
q'
<-
PGS
.
formatQuery
c
q
a
q'
<-
PGS
.
formatQuery
c
q
a
hPutStrLn
stderr
q'
hPutStrLn
stderr
q'
throw
(
SomeException
e
)
throw
(
SomeException
e
)
nodeExists
::
Pool
PGS
.
Connection
->
NodeId
->
IO
Bool
runPGSAdvisoryXactLock
::
PGS
.
Connection
->
Int
->
IO
()
nodeExists
pool
nId
=
(
==
[
PGS
.
Only
True
])
runPGSAdvisoryXactLock
c
id
=
do
<$>
runPGSQuery
pool
[
sql
|
SELECT true FROM nodes WHERE id = ? LIMIT 1
|]
(
PGS
.
Only
nId
)
_
<-
runPGSQuery
c
[
sql
|
SELECT pg_advisory_xact_lock(?)
|]
(
PGS
.
Only
id
)
::
IO
[
PGS
.
Only
Bool
]
pure
()
nodeExists
::
PGS
.
Connection
->
NodeId
->
IO
Bool
nodeExists
c
nId
=
(
==
[
PGS
.
Only
True
])
<$>
runPGSQuery
c
[
sql
|
SELECT true FROM nodes WHERE id = ? LIMIT 1
|]
(
PGS
.
Only
nId
)
getNodesIdWithType
::
P
ool
P
GS
.
Connection
->
NodeType
->
IO
[
NodeId
]
getNodesIdWithType
::
PGS
.
Connection
->
NodeType
->
IO
[
NodeId
]
getNodesIdWithType
pool
nt
=
do
getNodesIdWithType
c
nt
=
do
ns
<-
runPGSQuery
pool
query
(
PGS
.
Only
nt
)
ns
<-
runPGSQuery
c
query
(
PGS
.
Only
nt
)
pure
$
map
(
\
(
PGS
.
Only
nId
)
->
NodeId
nId
)
ns
pure
$
map
(
\
(
PGS
.
Only
nId
)
->
NodeId
nId
)
ns
where
where
query
::
PGS
.
Query
query
::
PGS
.
Query
...
@@ -321,9 +330,9 @@ getNodesIdWithType pool nt = do
...
@@ -321,9 +330,9 @@ getNodesIdWithType pool nt = do
-- nodeStorySelect = selectTable nodeStoryTable
-- nodeStorySelect = selectTable nodeStoryTable
-- TODO Check ordering, "first patch in the _a_history list is the most recent"
-- TODO Check ordering, "first patch in the _a_history list is the most recent"
getNodeArchiveHistory
::
P
ool
P
GS
.
Connection
->
NodeId
->
IO
[
NgramsStatePatch'
]
getNodeArchiveHistory
::
PGS
.
Connection
->
NodeId
->
IO
[
NgramsStatePatch'
]
getNodeArchiveHistory
pool
nodeId
=
do
getNodeArchiveHistory
c
nodeId
=
do
as
<-
runPGSQuery
pool
query
(
PGS
.
Only
nodeId
)
::
IO
[(
TableNgrams
.
NgramsType
,
NgramsTerm
,
NgramsPatch
)]
as
<-
runPGSQuery
c
query
(
PGS
.
Only
nodeId
)
::
IO
[(
TableNgrams
.
NgramsType
,
NgramsTerm
,
NgramsPatch
)]
pure
$
(
\
(
ngramsType
,
terms
,
patch
)
->
fst
$
PM
.
singleton
ngramsType
(
NgramsTablePatch
$
fst
$
PM
.
singleton
terms
patch
))
<$>
as
pure
$
(
\
(
ngramsType
,
terms
,
patch
)
->
fst
$
PM
.
singleton
ngramsType
(
NgramsTablePatch
$
fst
$
PM
.
singleton
terms
patch
))
<$>
as
where
where
query
::
PGS
.
Query
query
::
PGS
.
Query
...
@@ -336,28 +345,28 @@ ngramsIdQuery :: PGS.Query
...
@@ -336,28 +345,28 @@ ngramsIdQuery :: PGS.Query
ngramsIdQuery
=
[
sql
|
SELECT id FROM ngrams WHERE terms = ?
|]
ngramsIdQuery
=
[
sql
|
SELECT id FROM ngrams WHERE terms = ?
|]
insertNodeArchiveHistory
::
P
ool
P
GS
.
Connection
->
NodeId
->
[
NgramsStatePatch'
]
->
IO
()
insertNodeArchiveHistory
::
PGS
.
Connection
->
NodeId
->
[
NgramsStatePatch'
]
->
IO
()
insertNodeArchiveHistory
_
_
[]
=
pure
()
insertNodeArchiveHistory
_
_
[]
=
pure
()
insertNodeArchiveHistory
pool
nodeId
(
h
:
hs
)
=
do
insertNodeArchiveHistory
c
nodeId
(
h
:
hs
)
=
do
let
tuples
=
mconcat
$
(
\
(
nType
,
(
NgramsTablePatch
patch
))
->
let
tuples
=
mconcat
$
(
\
(
nType
,
(
NgramsTablePatch
patch
))
->
(
\
(
term
,
p
)
->
(
\
(
term
,
p
)
->
(
nodeId
,
nType
,
term
,
p
))
<$>
PM
.
toList
patch
)
<$>
PM
.
toList
h
::
[(
NodeId
,
TableNgrams
.
NgramsType
,
NgramsTerm
,
NgramsPatch
)]
(
nodeId
,
nType
,
term
,
p
))
<$>
PM
.
toList
patch
)
<$>
PM
.
toList
h
::
[(
NodeId
,
TableNgrams
.
NgramsType
,
NgramsTerm
,
NgramsPatch
)]
tuplesM
<-
mapM
(
\
(
nId
,
nType
,
term
,
patch
)
->
do
tuplesM
<-
mapM
(
\
(
nId
,
nType
,
term
,
patch
)
->
do
ngrams
<-
runPGSQuery
pool
ngramsIdQuery
(
PGS
.
Only
term
)
ngrams
<-
runPGSQuery
c
ngramsIdQuery
(
PGS
.
Only
term
)
pure
$
(
\
(
PGS
.
Only
termId
)
->
(
nId
,
nType
,
termId
,
term
,
patch
))
<$>
(
headMay
ngrams
)
pure
$
(
\
(
PGS
.
Only
termId
)
->
(
nId
,
nType
,
termId
,
term
,
patch
))
<$>
(
headMay
ngrams
)
)
tuples
::
IO
[
Maybe
(
NodeId
,
TableNgrams
.
NgramsType
,
Int
,
NgramsTerm
,
NgramsPatch
)]
)
tuples
::
IO
[
Maybe
(
NodeId
,
TableNgrams
.
NgramsType
,
Int
,
NgramsTerm
,
NgramsPatch
)]
_
<-
runPGSExecuteMany
pool
query
$
((
\
(
nId
,
nType
,
termId
,
_term
,
patch
)
->
(
nId
,
nType
,
termId
,
patch
))
<$>
(
catMaybes
tuplesM
))
_
<-
runPGSExecuteMany
c
query
$
((
\
(
nId
,
nType
,
termId
,
_term
,
patch
)
->
(
nId
,
nType
,
termId
,
patch
))
<$>
(
catMaybes
tuplesM
))
_
<-
insertNodeArchiveHistory
pool
nodeId
hs
_
<-
insertNodeArchiveHistory
c
nodeId
hs
pure
()
pure
()
where
where
query
::
PGS
.
Query
query
::
PGS
.
Query
query
=
[
sql
|
INSERT INTO node_story_archive_history(node_id, ngrams_type_id, ngrams_id, patch) VALUES (?, ?, ?, ?)
|]
query
=
[
sql
|
INSERT INTO node_story_archive_history(node_id, ngrams_type_id, ngrams_id, patch) VALUES (?, ?, ?, ?)
|]
getNodeStory
::
P
ool
P
GS
.
Connection
->
NodeId
->
IO
NodeListStory
getNodeStory
::
PGS
.
Connection
->
NodeId
->
IO
NodeListStory
getNodeStory
pool
nId
@
(
NodeId
nodeId
)
=
do
getNodeStory
c
nId
@
(
NodeId
nodeId
)
=
do
--res <- withResource pool $ \c -> runSelect c query :: IO [NodeStoryPoly NodeId Version Int Int NgramsRepoElement]
--res <- withResource pool $ \c -> runSelect c query :: IO [NodeStoryPoly NodeId Version Int Int NgramsRepoElement]
res
<-
runPGSQuery
pool
nodeStoriesQuery
(
PGS
.
Only
nodeId
)
::
IO
[(
Version
,
TableNgrams
.
NgramsType
,
NgramsTerm
,
NgramsRepoElement
)]
res
<-
runPGSQuery
c
nodeStoriesQuery
(
PGS
.
Only
nodeId
)
::
IO
[(
Version
,
TableNgrams
.
NgramsType
,
NgramsTerm
,
NgramsRepoElement
)]
-- We have multiple rows with same node_id and different (ngrams_type_id, ngrams_id).
-- We have multiple rows with same node_id and different (ngrams_type_id, ngrams_id).
-- Need to create a map: {<node_id>: {<ngrams_type_id>: {<ngrams_id>: <data>}}}
-- Need to create a map: {<node_id>: {<ngrams_type_id>: {<ngrams_id>: <data>}}}
let
dbData
=
map
(
\
(
version
,
ngramsType
,
ngrams
,
ngrams_repo_element
)
->
let
dbData
=
map
(
\
(
version
,
ngramsType
,
ngrams
,
ngrams_repo_element
)
->
...
@@ -383,27 +392,30 @@ nodeStoriesQuery = [sql| SELECT version, ngrams_type_id, terms, ngrams_repo_elem
...
@@ -383,27 +392,30 @@ nodeStoriesQuery = [sql| SELECT version, ngrams_type_id, terms, ngrams_repo_elem
JOIN ngrams ON ngrams.id = ngrams_id
JOIN ngrams ON ngrams.id = ngrams_id
WHERE node_id = ?
|]
WHERE node_id = ?
|]
type
ArchiveStateList
=
[(
TableNgrams
.
NgramsType
,
NgramsTerm
,
NgramsRepoElement
)]
-- Functions to convert archive state (which is a Map NgramsType (Map
-- Functions to convert archive state (which is a Map NgramsType (Map
-- NgramsTerm NgramsRepoElement)) to/from a flat list
-- NgramsTerm NgramsRepoElement)) to/from a flat list
archiveStateAsList
::
NgramsState'
->
[(
TableNgrams
.
NgramsType
,
NgramsTerm
,
NgramsRepoElement
)]
archiveStateAsList
::
NgramsState'
->
ArchiveStateList
archiveStateAsList
s
=
mconcat
$
(
\
(
nt
,
ntm
)
->
(
\
(
n
,
nre
)
->
(
nt
,
n
,
nre
))
<$>
Map
.
toList
ntm
)
<$>
Map
.
toList
s
archiveStateAsList
s
=
mconcat
$
(
\
(
nt
,
ntm
)
->
(
\
(
n
,
nre
)
->
(
nt
,
n
,
nre
))
<$>
Map
.
toList
ntm
)
<$>
Map
.
toList
s
archiveStateFromList
::
ArchiveStateList
->
NgramsState'
archiveStateFromList
l
=
Map
.
fromListWith
(
<>
)
$
(
\
(
nt
,
t
,
nre
)
->
(
nt
,
Map
.
singleton
t
nre
))
<$>
l
-- | This function inserts whole new node story and archive for given node_id.
-- | This function inserts whole new node story and archive for given node_id.
insertNodeStory
::
PGS
.
Connection
->
NodeId
->
ArchiveList
->
IO
()
-- | This function inserts whole new node story and archive for given node_id.
insertNodeStory
c
nodeId
@
(
NodeId
nId
)
a
=
do
printDebug
"[insertNodeStory] _a_state"
$
a
^.
a_state
_
<-
mapM
(
\
(
ngramsType
,
ngrams
,
ngramsRepoElement
)
->
do
_
<-
mapM
(
\
(
ngramsType
,
ngrams
,
ngramsRepoElement
)
->
do
insertNodeStory
pool
nodeId
@
(
NodeId
nId
)
(
A
rchive
{
..
})
=
do
termIdM
<-
runPGSQuery
c
ngramsIdQuery
(
PGS
.
Only
ngrams
)
::
IO
[
PGS
.
Only
Int64
]
case
headMay
termIdM
of
case
headMay
termIdM
of
Nothing
->
pure
0
Nothing
->
pure
0
case
headMay
termIdM
of
Just
(
PGS
.
Only
termId
)
->
runPGSExecuteMany
c
query
[(
nId
,
a
^.
a_version
,
ngramsType
,
termId
,
ngramsRepoElement
)])
$
archiveStateAsList
$
a
^.
a_state
-- runInsert c $ insert ngramsType ngrams ngramsRepoElement) $ archiveStateAsList _a_state
-- runInsert c $ insert ngramsType ngrams ngramsRepoElement) $ archiveStateAsList _a_state
-- NOTE: It is assumed that the most recent change is the first in the
-- NOTE: It is assumed that the most recent change is the first in the
-- list, so we save these in reverse order
-- list, so we save these in reverse order
-- NOTE: It is assumed th
at the most recent chan
ge is the first in the
insertNodeArchiveHistory
c
nodeId
$
reverse
$
a
^.
a_history
pure
()
pure
()
where
where
query
::
PGS
.
Query
query
::
PGS
.
Query
...
@@ -419,19 +431,74 @@ insertNodeStory pool nodeId@(NodeId nId) (Archive {..}) = do
...
@@ -419,19 +431,74 @@ insertNodeStory pool nodeId@(NodeId nId) (Archive {..}) = do
-- , iReturning = rCount
-- , iReturning = rCount
-- , iOnConflict = Nothing }
-- , iOnConflict = Nothing }
-- | This function updates the node story and archive for given node_id.
insertArchive
::
PGS
.
Connection
->
NodeId
->
ArchiveList
->
IO
()
updateNodeStory
::
Pool
PGS
.
Connection
->
NodeId
->
ArchiveQ
->
IO
()
insertArchive
c
nodeId
a
=
do
updateNodeStory
pool
nodeId
@
(
NodeId
_nId
)
(
Archive
{
..
})
=
do
_
<-
runPGSExecuteMany
c
query
$
(
\
(
nt
,
n
,
nre
)
->
(
nodeId
,
a
^.
a_version
,
nt
,
nre
,
n
))
<$>
(
archiveStateAsList
$
a
^.
a_state
)
-- TODO This requires updating current DB state (which is spanned
pure
()
-- along many rows)
where
query
::
PGS
.
Query
query
=
[
sql
|
INSERT INTO node_stories(node_id, version, ngrams_type_id, ngrams_id, ngrams_repo_element)
SELECT ?, ?, ?, ngrams.id, ? FROM ngrams WHERE terms = ?
|]
deleteArchive
::
PGS
.
Connection
->
NodeId
->
ArchiveList
->
IO
()
deleteArchive
c
nodeId
a
=
do
_
<-
runPGSExecuteMany
c
query
$
(
\
(
nt
,
n
,
_
)
->
(
nodeId
,
nt
,
n
))
<$>
(
archiveStateAsList
$
a
^.
a_state
)
pure
()
where
query
::
PGS
.
Query
query
=
[
sql
|
WITH (SELECT id FROM ngrams WHERE terms = ?) AS ngrams
DELETE FROM node_stories
WHERE node_id = ? AND ngrams_type_id = ? AND ngrams_id IN (SELECT id FROM ngrams WHERE terms = ?)
|]
-- The idea is this: fetch all current state data from the DB
updateArchive
::
PGS
.
Connection
->
NodeId
->
ArchiveList
->
IO
()
-- (locking the rows), perform a diff and update what is necessary.
updateArchive
c
nodeId
a
=
do
-- ret <- withResource pool $ \c -> runUpdate c update
_
<-
runPGSExecuteMany
c
query
$
(
\
(
nt
,
n
,
nre
)
->
(
nre
,
nodeId
,
nt
,
n
))
<$>
(
archiveStateAsList
$
a
^.
a_state
)
pure
()
where
query
::
PGS
.
Query
query
=
[
sql
|
UPDATE node_stories
SET ngrams_repo_element = ?
WHERE node_id = ? AND ngrams_type_id = ? AND ngrams_id IN (SELECT id FROM ngrams WHERE terms = ?)
|]
-- | This function updates the node story and archive for given node_id.
updateNodeStory
::
PGS
.
Connection
->
NodeId
->
ArchiveList
->
ArchiveList
->
IO
()
updateNodeStory
c
nodeId
@
(
NodeId
_nId
)
currentArchive
newArchive
=
do
-- STEPS
-- 0. We assume we're inside an advisory lock
-- 1. Find differences (inserts/updates/deletes)
let
currentList
=
archiveStateAsList
$
currentArchive
^.
a_state
let
newList
=
archiveStateAsList
$
newArchive
^.
a_state
let
currentSet
=
Set
.
fromList
$
(
\
(
nt
,
n
,
_
)
->
(
nt
,
n
))
<$>
currentList
let
newSet
=
Set
.
fromList
$
(
\
(
nt
,
n
,
_
)
->
(
nt
,
n
))
<$>
newList
let
inserts
=
filter
(
\
(
nt
,
n
,
_
)
->
Set
.
member
(
nt
,
n
)
$
Set
.
difference
newSet
currentSet
)
newList
printDebug
"[updateNodeStory] inserts"
inserts
let
deletes
=
filter
(
\
(
nt
,
n
,
_
)
->
Set
.
member
(
nt
,
n
)
$
Set
.
difference
currentSet
newSet
)
currentList
printDebug
"[updateNodeStory] deletes"
deletes
-- updates are the things that are in new but not in current
let
updates
=
Set
.
toList
$
Set
.
difference
(
Set
.
fromList
newList
)
(
Set
.
fromList
currentList
)
printDebug
"[updateNodeStory] updates"
$
Text
.
unlines
$
(
Text
.
pack
.
show
)
<$>
updates
-- 2. Perform inserts/deletes/updates
insertArchive
c
nodeId
$
Archive
{
_a_version
=
newArchive
^.
a_version
,
_a_history
=
[]
,
_a_state
=
archiveStateFromList
inserts
}
-- TODO Use currentArchive ^. a_version in delete and report error
-- if entries with (node_id, ngrams_type_id, ngrams_id) but
-- different version are found.
deleteArchive
c
nodeId
$
Archive
{
_a_version
=
newArchive
^.
a_version
,
_a_history
=
[]
,
_a_state
=
archiveStateFromList
deletes
}
updateArchive
c
nodeId
$
Archive
{
_a_version
=
newArchive
^.
a_version
,
_a_history
=
[]
,
_a_state
=
archiveStateFromList
updates
}
-- NOTE: It is assumed that the most recent change is the first in the
-- NOTE: It is assumed that the most recent change is the first in the
-- list, so we save these in reverse order
-- list, so we save these in reverse order
insertNodeArchiveHistory
pool
nodeId
$
reverse
_
a_history
insertNodeArchiveHistory
c
nodeId
$
reverse
$
newArchive
^.
a_history
pure
()
pure
()
-- where
-- where
-- update = Update { uTable = nodeStoryTable
-- update = Update { uTable = nodeStoryTable
...
@@ -449,38 +516,43 @@ updateNodeStory pool nodeId@(NodeId _nId) (Archive {..}) = do
...
@@ -449,38 +516,43 @@ updateNodeStory pool nodeId@(NodeId _nId) (Archive {..}) = do
-- , dWhere = (\row -> node_id row .== sqlInt4 nId)
-- , dWhere = (\row -> node_id row .== sqlInt4 nId)
-- , dReturning = rCount }
-- , dReturning = rCount }
upsertNodeArchive
::
Pool
PGS
.
Connection
->
NodeId
->
ArchiveQ
->
IO
()
upsertNodeStories
::
PGS
.
Connection
->
NodeId
->
ArchiveList
->
IO
()
upsertNodeArchive
pool
nId
a
=
do
upsertNodeStories
c
nodeId
@
(
NodeId
nId
)
newArchive
=
do
(
NodeStory
m
)
<-
getNodeStory
pool
nId
printDebug
"[upsertNodeStories] START nId"
nId
case
Map
.
lookup
nId
m
of
PGS
.
begin
c
--runPGSAdvisoryXactLock c nId
(
NodeStory
m
)
<-
getNodeStory
c
nodeId
case
Map
.
lookup
nodeId
m
of
Nothing
->
do
Nothing
->
do
_
<-
insertNodeStory
pool
nId
a
_
<-
insertNodeStory
c
nodeId
newArchive
pure
()
pure
()
Just
_
->
do
Just
currentArchive
->
do
_
<-
updateNodeStory
pool
nId
a
_
<-
updateNodeStory
c
nodeId
currentArchive
newArchive
pure
()
pure
()
PGS
.
commit
c
printDebug
"[upsertNodeStories] STOP nId"
nId
writeNodeStories
::
P
ool
P
GS
.
Connection
->
NodeListStory
->
IO
()
writeNodeStories
::
PGS
.
Connection
->
NodeListStory
->
IO
()
writeNodeStories
pool
(
NodeStory
nls
)
=
do
writeNodeStories
c
(
NodeStory
nls
)
=
do
_
<-
mapM
(
\
(
nId
,
a
)
->
upsertNode
Archive
pool
nId
a
)
$
Map
.
toList
nls
_
<-
mapM
(
\
(
nId
,
a
)
->
upsertNode
Stories
c
nId
a
)
$
Map
.
toList
nls
pure
()
pure
()
-- | Returns a `NodeListStory`, updating the given one for given `NodeId`
-- | Returns a `NodeListStory`, updating the given one for given `NodeId`
nodeStoryInc
::
P
ool
P
GS
.
Connection
->
Maybe
NodeListStory
->
NodeId
->
IO
NodeListStory
nodeStoryInc
::
PGS
.
Connection
->
Maybe
NodeListStory
->
NodeId
->
IO
NodeListStory
nodeStoryInc
pool
Nothing
nId
=
getNodeStory
pool
nId
nodeStoryInc
c
Nothing
nId
=
getNodeStory
c
nId
nodeStoryInc
pool
(
Just
ns
@
(
NodeStory
nls
))
nId
=
do
nodeStoryInc
c
(
Just
ns
@
(
NodeStory
nls
))
nId
=
do
case
Map
.
lookup
nId
nls
of
case
Map
.
lookup
nId
nls
of
Nothing
->
do
Nothing
->
do
(
NodeStory
nls'
)
<-
getNodeStory
pool
nId
(
NodeStory
nls'
)
<-
getNodeStory
c
nId
pure
$
NodeStory
$
Map
.
union
nls
nls'
pure
$
NodeStory
$
Map
.
union
nls
nls'
Just
_
->
pure
ns
Just
_
->
pure
ns
nodeStoryIncs
::
P
ool
P
GS
.
Connection
->
Maybe
NodeListStory
->
[
NodeId
]
->
IO
NodeListStory
nodeStoryIncs
::
PGS
.
Connection
->
Maybe
NodeListStory
->
[
NodeId
]
->
IO
NodeListStory
nodeStoryIncs
_
Nothing
[]
=
pure
$
NodeStory
$
Map
.
empty
nodeStoryIncs
_
Nothing
[]
=
pure
$
NodeStory
$
Map
.
empty
nodeStoryIncs
pool
(
Just
nls
)
ns
=
foldM
(
\
m
n
->
nodeStoryInc
pool
(
Just
m
)
n
)
nls
ns
nodeStoryIncs
c
(
Just
nls
)
ns
=
foldM
(
\
m
n
->
nodeStoryInc
c
(
Just
m
)
n
)
nls
ns
nodeStoryIncs
pool
Nothing
(
ni
:
ns
)
=
do
nodeStoryIncs
c
Nothing
(
ni
:
ns
)
=
do
m
<-
getNodeStory
pool
ni
m
<-
getNodeStory
c
ni
nodeStoryIncs
pool
(
Just
m
)
ns
nodeStoryIncs
c
(
Just
m
)
ns
-- nodeStoryDec :: Pool PGS.Connection -> NodeListStory -> NodeId -> IO NodeListStory
-- nodeStoryDec :: Pool PGS.Connection -> NodeListStory -> NodeId -> IO NodeListStory
-- nodeStoryDec pool ns@(NodeStory nls) ni = do
-- nodeStoryDec pool ns@(NodeStory nls) ni = do
...
@@ -510,10 +582,10 @@ readNodeStoryEnv pool = do
...
@@ -510,10 +582,10 @@ readNodeStoryEnv pool = do
nodeStoryVar
::
Pool
PGS
.
Connection
->
Maybe
(
MVar
NodeListStory
)
->
[
NodeId
]
->
IO
(
MVar
NodeListStory
)
nodeStoryVar
::
Pool
PGS
.
Connection
->
Maybe
(
MVar
NodeListStory
)
->
[
NodeId
]
->
IO
(
MVar
NodeListStory
)
nodeStoryVar
pool
Nothing
nIds
=
do
nodeStoryVar
pool
Nothing
nIds
=
do
state
<-
nodeStoryIncs
pool
Nothing
nIds
state
<-
withResource
pool
$
\
c
->
nodeStoryIncs
c
Nothing
nIds
newMVar
state
newMVar
state
nodeStoryVar
pool
(
Just
mv
)
nIds
=
do
nodeStoryVar
pool
(
Just
mv
)
nIds
=
do
_
<-
modifyMVar_
mv
$
\
nsl
->
(
nodeStoryIncs
pool
(
Just
nsl
)
nIds
)
_
<-
withResource
pool
$
\
c
->
modifyMVar_
mv
$
\
nsl
->
(
nodeStoryIncs
c
(
Just
nsl
)
nIds
)
pure
mv
pure
mv
-- Debounce is useful since it could delay the saving to some later
-- Debounce is useful since it could delay the saving to some later
...
@@ -523,7 +595,10 @@ mkNodeStorySaver pool mvns = mkDebounce settings
...
@@ -523,7 +595,10 @@ mkNodeStorySaver pool mvns = mkDebounce settings
where
where
settings
=
defaultDebounceSettings
settings
=
defaultDebounceSettings
{
debounceAction
=
do
{
debounceAction
=
do
withMVar
mvns
(
\
ns
->
writeNodeStories
pool
ns
)
withResource
pool
$
\
c
->
do
withMVar
mvns
$
\
ns
->
do
--printDebug "[mkNodeStorySaver] will call writeNodeStories, ns" ns
writeNodeStories
c
ns
withMVar
mvns
(
\
ns
->
printDebug
"[mkNodeStorySaver] debounce nodestory"
ns
)
withMVar
mvns
(
\
ns
->
printDebug
"[mkNodeStorySaver] debounce nodestory"
ns
)
modifyMVar_
mvns
$
\
ns
->
pure
$
clearHistory
ns
modifyMVar_
mvns
$
\
ns
->
pure
$
clearHistory
ns
,
debounceFreq
=
1
*
minute
,
debounceFreq
=
1
*
minute
...
@@ -536,6 +611,13 @@ clearHistory (NodeStory ns) = NodeStory $ ns & (traverse . a_history) .~ emptyHi
...
@@ -536,6 +611,13 @@ clearHistory (NodeStory ns) = NodeStory $ ns & (traverse . a_history) .~ emptyHi
where
where
emptyHistory
=
[]
::
[
NgramsStatePatch'
]
emptyHistory
=
[]
::
[
NgramsStatePatch'
]
currentVersion
::
(
HasNodeStory
env
err
m
)
=>
ListId
->
m
Version
currentVersion
listId
=
do
pool
<-
view
connPool
nls
<-
withResource
pool
$
\
c
->
liftBase
$
getNodeStory
c
listId
pure
$
nls
^.
unNodeStory
.
at
listId
.
_Just
.
a_version
-- mkNodeStorySaver :: MVar NodeListStory -> Cmd err (Cmd err ())
-- mkNodeStorySaver :: MVar NodeListStory -> Cmd err (Cmd err ())
-- mkNodeStorySaver mvns = mkDebounce settings
-- mkNodeStorySaver mvns = mkDebounce settings
-- where
-- where
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment