2 Module : Gargantext.Core.Viz.Graph.Bridgeness
3 Description : Bridgeness filter
4 Copyright : (c) CNRS, 2017-Present
5 License : AGPL + CECILL v3
6 Maintainer : team@gargantext.org
7 Stability : experimental
10 Let be a graph Bridgeness filters inter-communities links in two ways.
11 If the partitions are known, filtering is uniform to expose the communities clearly for the beginners.
16 filters inter-communities links.
18 TODO use Map LouvainNodeId (Map LouvainNodeId)
21 {-# LANGUAGE BangPatterns #-}
23 module Gargantext.Core.Viz.Graph.Bridgeness -- (bridgeness)
26 import Gargantext.Core.Methods.Similarities (Similarity(..))
27 -- import Data.IntMap (IntMap)
28 import Data.Map (Map, fromListWith, lookup, toList, mapWithKey, elems)
29 import Data.Maybe (catMaybes)
30 import Data.Ord (Down(..))
31 import Debug.Trace (trace)
32 import Gargantext.Prelude
33 import Graph.Types (ClusterNode(..))
34 -- import qualified Data.IntMap as IntMap
35 import qualified Data.List as List
36 import qualified Data.Map as Map
37 -- import qualified Data.Set as Set
39 ----------------------------------------------------------------------
41 type Partitions = Map (Int, Int) Double -> IO [ClusterNode]
42 ----------------------------------------------------------------------
43 nodeId2comId :: ClusterNode -> (NodeId, CommunityId)
44 nodeId2comId (ClusterNode i1 i2) = (i1, i2)
47 type CommunityId = Int
49 ----------------------------------------------------------------------
50 ----------------------------------------------------------------------
51 data Bridgeness = Bridgeness_Basic { bridgeness_partitions :: [ClusterNode]
52 , bridgeness_filter :: Double
54 | Bridgeness_Advanced { bridgeness_similarity :: Similarity
55 , bridgness_confluence :: Confluence
58 type Confluence = Map (NodeId, NodeId) Double
61 bridgeness :: Bridgeness
62 -> Map (NodeId, NodeId) Double
63 -> Map (NodeId, NodeId) Double
64 bridgeness (Bridgeness_Advanced sim c) m = Map.fromList
65 $ List.filter (\x -> if sim == Conditional then snd x > 0.2 else snd x > 0.02)
66 $ map (\(ks, (v1,_v2)) -> (ks,v1))
67 -- $ List.take (if sim == Conditional then 2*n else 3*n)
68 -- $ List.sortOn (Down . (snd . snd))
70 $ trace ("bridgeness3 m c" <> show (m,c))
71 $ Map.intersectionWithKey
72 (\k v1 v2 -> trace ("intersectionWithKey " <> (show (k, v1, v2))) (v1, v2)) m c
78 !n = trace ("bridgeness m size: " <> (show $ List.length m'))
80 $ (fromIntegral $ List.length m') / (log $ fromIntegral nodesNumber :: Double)
83 nodesNumber = Set.size $ Set.fromList $ as <> bs
85 (as, bs) = List.unzip $ Map.keys m
89 bridgeness (Bridgeness_Basic ns b) m = Map.fromList
93 $ groupEdges (Map.fromList $ map nodeId2comId ns) m
95 groupEdges :: (Ord a, Ord b1)
98 -> Map (a, a) [((b1, b1), b2)]
99 groupEdges m = fromListWith (<>)
103 n1n2_m = (,) <$> lookup n1 m <*> lookup n2 m
104 n1n2_d = Just [((n1,n2),d)]
105 in (,) <$> n1n2_m <*> n1n2_d
109 -- | TODO : sortOn Confluence
110 filterComs :: (Ord n1, Eq n2)
112 -> Map (n2, n2) [(a3, n1)]
113 -> Map (n2, n2) [(a3, n1)]
114 filterComs _b m = Map.filter (\n -> length n > 0) $ mapWithKey filter' m
119 | otherwise = take n $ List.sortOn (Down . snd) a
122 n = round $ 100 * a' / t
123 a'= fromIntegral $ length a
125 t = fromIntegral $ length $ List.concat $ elems m
127 --------------------------------------------------------------
130 map2intMap :: Map (Int, Int) a -> IntMap (IntMap a)
131 map2intMap m = IntMap.fromListWith (<>)
132 $ map (\((k1,k2), v) -> if k1 < k2
133 then (k1, IntMap.singleton k2 v)
134 else (k2, IntMap.singleton k1 v)
138 look :: (Int,Int) -> IntMap (IntMap a) -> Maybe a
139 look (k1,k2) m = if k1 < k2
140 then case (IntMap.lookup k1 m) of
141 Just m' -> IntMap.lookup k2 m'
147 Compute the median of a list
148 From: https://hackage.haskell.org/package/dsp-0.2.5.1/docs/src/Numeric.Statistics.Median.html
149 Compute the center of the list in a more lazy manner
150 and thus halves memory requirement.
153 median :: (Ord a, Fractional a) => [a] -> a
154 median [] = panic "medianFast: empty list has no median"
156 let recurse (x0:_) (_:[]) = x0
157 recurse (x0:x1:_) (_:_:[]) = (x0+x1)/2
158 recurse (_:xs) (_:_:ys) = recurse xs ys
160 panic "median: this error cannot occur in the way 'recurse' is called"