2 Module : Gargantext.API.Node
3 Description : Server API
4 Copyright : (c) CNRS, 2017-Present
5 License : AGPL + CECILL v3
6 Maintainer : team@gargantext.org
7 Stability : experimental
10 -- TODO-SECURITY: Critical
12 -- TODO-ACCESS: CanGetNode
13 -- TODO-EVENTS: No events as this is a read only query.
15 -------------------------------------------------------------------
16 -- TODO-ACCESS: access by admin only.
17 -- At first let's just have an isAdmin check.
18 -- Later: check userId CanDeleteNodes Nothing
19 -- TODO-EVENTS: DeletedNodes [NodeId]
20 -- {"tag": "DeletedNodes", "nodes": [Int*]}
24 {-# OPTIONS_GHC -fno-warn-orphans #-}
26 {-# LANGUAGE ScopedTypeVariables #-}
27 {-# LANGUAGE TemplateHaskell #-}
28 {-# LANGUAGE TypeOperators #-}
30 module Gargantext.API.Node
33 import Data.Aeson (FromJSON, ToJSON)
34 import Data.Aeson.TH (deriveJSON)
37 import Data.Text (Text())
38 import GHC.Generics (Generic)
40 import Test.QuickCheck (elements)
41 import Test.QuickCheck.Arbitrary (Arbitrary, arbitrary)
43 import Gargantext.API.Admin.Auth.Types (PathId(..))
44 import Gargantext.API.Admin.Auth (withAccess)
45 import Gargantext.API.Metrics
46 import Gargantext.API.Ngrams (TableNgramsApi, apiNgramsTableCorpus)
47 import Gargantext.API.Ngrams.Types (TabType(..))
48 import Gargantext.API.Node.File
49 import Gargantext.API.Node.New
50 import Gargantext.API.Prelude
51 import Gargantext.API.Table
52 import Gargantext.Core.Types (NodeTableResult)
53 import Gargantext.Core.Types.Individu (User(..))
54 import Gargantext.Core.Types.Main (Tree, NodeTree)
55 import Gargantext.Core.Utils.Prefix (unPrefix)
56 import Gargantext.Database.Action.Flow.Pairing (pairing)
57 import Gargantext.Database.Admin.Types.Hyperdata
58 import Gargantext.Database.Admin.Types.Node
59 import Gargantext.Database.Prelude -- (Cmd, CmdM)
60 import Gargantext.Database.Query.Facet (FacetDoc, OrderBy(..))
61 import Gargantext.Database.Query.Table.Node
62 import Gargantext.Database.Query.Table.Node.Children (getChildren)
63 import Gargantext.Database.Query.Table.Node.Error (HasNodeError(..))
64 import Gargantext.Database.Query.Table.Node.Update (Update(..), update)
65 import Gargantext.Database.Query.Table.Node.UpdateOpaleye (updateHyperdata)
66 import Gargantext.Database.Query.Table.NodeNode
67 import Gargantext.Database.Query.Tree (tree, TreeMode(..))
68 import Gargantext.Prelude
69 import Gargantext.Core.Viz.Phylo.API (PhyloAPI, phyloAPI)
70 import qualified Gargantext.API.Node.Share as Share
71 import qualified Gargantext.API.Node.Update as Update
72 import qualified Gargantext.API.Search as Search
73 import qualified Gargantext.Database.Action.Delete as Action (deleteNode)
74 import qualified Gargantext.Database.Query.Table.Node.Update as U (update, Update(..))
77 import qualified Gargantext.Core.Text.List.Learn as Learn
78 import qualified Data.Vector as Vec
83 type NodesAPI = Delete '[JSON] Int
86 -- Be careful: really delete nodes
87 -- Access by admin only
88 nodesAPI :: [NodeId] -> GargServer NodesAPI
89 nodesAPI = deleteNodes
91 ------------------------------------------------------------------------
92 -- | TODO-ACCESS: access by admin only.
93 -- At first let's just have an isAdmin check.
94 -- Later: CanAccessAnyNode or (CanGetAnyNode, CanPutAnyNode)
95 -- To manage the Users roots
98 -- TODO needs design discussion.
99 type Roots = Get '[JSON] [Node HyperdataUser]
100 :<|> Put '[JSON] Int -- TODO
102 -- | TODO: access by admin only
103 roots :: GargServer Roots
104 roots = getNodesWithParentId Nothing
105 :<|> pure (panic "not implemented yet") -- TODO use patch map to update what we need
107 -------------------------------------------------------------------
108 -- | Node API Types management
109 -- TODO-ACCESS : access by users
110 -- No ownership check is needed if we strictly follow the capability model.
112 -- CanGetNode (Node, Children, TableApi, TableNgramsApiGet, PairingApi, ChartApi,
114 -- CanRenameNode (or part of CanEditNode?)
115 -- CanCreateChildren (PostNodeApi)
116 -- CanEditNode / CanPutNode TODO not implemented yet
118 -- CanPatch (TableNgramsApi)
122 type NodeAPI a = Get '[JSON] (Node a)
123 :<|> "rename" :> RenameApi
124 :<|> PostNodeApi -- TODO move to children POST
126 :<|> ReqBody '[JSON] a :> Put '[JSON] Int
127 :<|> "update" :> Update.API
128 :<|> Delete '[JSON] Int
129 :<|> "children" :> ChildrenApi a
132 :<|> "table" :> TableApi
133 :<|> "ngrams" :> TableNgramsApi
135 :<|> "category" :> CatApi
136 :<|> "search" :> (Search.API Search.SearchResult)
137 :<|> "share" :> Share.API
140 :<|> "pairwith" :> PairWith
141 :<|> "pairs" :> Pairs
142 :<|> "pairing" :> PairingApi
145 :<|> "metrics" :> ScatterAPI
146 :<|> "chart" :> ChartApi
148 :<|> "tree" :> TreeApi
149 :<|> "phylo" :> PhyloAPI
150 -- :<|> "add" :> NodeAddAPI
151 :<|> "move" :> MoveAPI
152 :<|> "unpublish" :> Share.Unpublish
154 :<|> "file" :> FileApi
155 :<|> "async" :> FileAsyncApi
157 -- TODO-ACCESS: check userId CanRenameNode nodeId
158 -- TODO-EVENTS: NodeRenamed RenameNode or re-use some more general NodeEdited...
159 type RenameApi = Summary " Rename Node"
160 :> ReqBody '[JSON] RenameNode
163 type PostNodeApi = Summary " PostNode Node with ParentId as {id}"
164 :> ReqBody '[JSON] PostNode
165 :> Post '[JSON] [NodeId]
167 type ChildrenApi a = Summary " Summary children"
168 :> QueryParam "type" NodeType
169 :> QueryParam "offset" Int
170 :> QueryParam "limit" Int
171 -- :> Get '[JSON] [Node a]
172 :> Get '[JSON] (NodeTableResult a)
174 ------------------------------------------------------------------------
175 type NodeNodeAPI a = Get '[JSON] (Node a)
177 nodeNodeAPI :: forall proxy a. (JSONB a, ToJSON a)
182 -> GargServer (NodeNodeAPI a)
183 nodeNodeAPI p uId cId nId = withAccess (Proxy :: Proxy (NodeNodeAPI a)) Proxy uId (PathNodeNode cId nId) nodeNodeAPI'
185 nodeNodeAPI' :: GargServer (NodeNodeAPI a)
186 nodeNodeAPI' = getNodeWith nId p
188 ------------------------------------------------------------------------
189 -- TODO: make the NodeId type indexed by `a`, then we no longer need the proxy.
190 nodeAPI :: forall proxy a.
197 -> GargServer (NodeAPI a)
198 nodeAPI p uId id' = withAccess (Proxy :: Proxy (NodeAPI a)) Proxy uId (PathNode id') nodeAPI'
200 nodeAPI' :: GargServer (NodeAPI a)
201 nodeAPI' = getNodeWith id' p
203 :<|> postNode uId id'
204 :<|> postNodeAsyncAPI uId id'
206 :<|> Update.api uId id'
207 :<|> Action.deleteNode (RootId $ NodeId uId) id'
208 :<|> getChildren id' p
212 :<|> apiNgramsTableCorpus id'
227 :<|> phyloAPI id' uId
228 :<|> moveNode (RootId $ NodeId uId) id'
229 -- :<|> nodeAddAPI id'
230 -- :<|> postUpload id'
231 :<|> Share.unPublish id'
234 :<|> fileAsyncApi uId id'
237 ------------------------------------------------------------------------
238 data RenameNode = RenameNode { r_name :: Text }
241 ------------------------------------------------------------------------
242 ------------------------------------------------------------------------
243 type CatApi = Summary " To Categorize NodeNodes: 0 for delete, 1/null neutral, 2 favorite"
244 :> ReqBody '[JSON] NodesToCategory
247 data NodesToCategory = NodesToCategory { ntc_nodesId :: [NodeId]
248 , ntc_category :: Int
252 -- TODO unPrefix "ntc_" FromJSON, ToJSON, ToSchema, adapt frontend.
253 instance FromJSON NodesToCategory
254 instance ToJSON NodesToCategory
255 instance ToSchema NodesToCategory
257 catApi :: CorpusId -> GargServer CatApi
260 putCat :: CorpusId -> NodesToCategory -> Cmd err [Int]
261 putCat cId cs' = nodeNodesCategory $ map (\n -> (cId, n, ntc_category cs')) (ntc_nodesId cs')
263 ------------------------------------------------------------------------
264 -- TODO adapt FacetDoc -> ListDoc (and add type of document as column)
265 -- Pairing utilities to move elsewhere
266 type PairingApi = Summary " Pairing API"
267 :> QueryParam "view" TabType
268 -- TODO change TabType -> DocType (CorpusId for pairing)
269 :> QueryParam "offset" Int
270 :> QueryParam "limit" Int
271 :> QueryParam "order" OrderBy
272 :> Get '[JSON] [FacetDoc]
275 type Pairs = Summary "List of Pairs"
276 :> Get '[JSON] [AnnuaireId]
277 pairs :: CorpusId -> GargServer Pairs
279 ns <- getNodeNode cId
280 pure $ map _nn_node2_id ns
282 type PairWith = Summary "Pair a Corpus with an Annuaire"
283 :> "annuaire" :> Capture "annuaire_id" AnnuaireId
284 :> QueryParam "list_id" ListId
287 pairWith :: CorpusId -> GargServer PairWith
288 pairWith cId aId lId = do
289 r <- pairing cId aId lId
290 _ <- insertNodeNode [ NodeNode cId aId Nothing Nothing]
294 ------------------------------------------------------------------------
295 type TreeAPI = QueryParams "type" NodeType :> Get '[JSON] (Tree NodeTree)
297 treeAPI :: NodeId -> GargServer TreeAPI
298 treeAPI = tree TreeAdvanced
300 ------------------------------------------------------------------------
301 -- | TODO Check if the name is less than 255 char
302 rename :: NodeId -> RenameNode -> Cmd err [Int]
303 rename nId (RenameNode name') = U.update (U.Rename nId name')
305 putNode :: forall err a. (HasNodeError err, JSONB a, ToJSON a)
309 putNode n h = fromIntegral <$> updateHyperdata n h
311 -------------------------------------------------------------
312 type MoveAPI = Summary "Move Node endpoint"
313 :> Capture "parent_id" ParentId
320 moveNode _u n p = update (Move n p)
321 -------------------------------------------------------------
324 $(deriveJSON (unPrefix "r_" ) ''RenameNode )
325 instance ToSchema RenameNode
326 instance Arbitrary RenameNode where
327 arbitrary = elements [RenameNode "test"]
330 -------------------------------------------------------------