-
Notifications
You must be signed in to change notification settings - Fork 220
Expand file tree
/
Copy pathPackageList.hs
More file actions
399 lines (352 loc) · 15.1 KB
/
PackageList.hs
File metadata and controls
399 lines (352 loc) · 15.1 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
{-# LANGUAGE RankNTypes, RecordWildCards, NamedFieldPuns #-}
module Distribution.Server.Features.PackageList (
ListFeature(..),
initListFeature,
PackageItem(..),
tagHistogram
) where
import Distribution.Server.Framework
import Distribution.Server.Features.Core
import Distribution.Server.Features.ReverseDependencies
import Distribution.Server.Features.Votes
import Distribution.Server.Features.DownloadCount
import Distribution.Server.Features.Tags
import Distribution.Server.Features.Users
import Distribution.Server.Features.Upload(UploadFeature(..))
import Distribution.Server.Features.Documentation (DocumentationFeature(..))
import Distribution.Server.Features.TarIndexCache (TarIndexCacheFeature(..))
import Distribution.Server.Features.PackageList.PackageRank
import Distribution.Server.Users.Users (userIdToName)
import qualified Distribution.Server.Users.UserIdSet as UserIdSet
import Distribution.Server.Users.Group(UserGroup(..), GroupDescription(..))
import Distribution.Server.Features.PreferredVersions
import qualified Distribution.Server.Packages.PackageIndex as PackageIndex
import Distribution.Server.Util.CountingMap (cmFind)
import Distribution.Server.Packages.Types
import Distribution.Server.Users.Types
import Distribution.Package
import Distribution.PackageDescription
import Distribution.PackageDescription.Configuration
import Distribution.Pretty (prettyShow)
import Distribution.Types.Version (Version)
import Distribution.Utils.ShortText (fromShortText)
import Distribution.Simple.Utils (safeLast)
import Control.Concurrent
import qualified Data.List.NonEmpty as NE
import Data.Maybe (mapMaybe)
import Data.Map (Map)
import qualified Data.Map as Map
import Data.Set (Set)
import qualified Data.Set as Set
import Data.Time.Clock (UTCTime(..))
data ListFeature = ListFeature {
listFeatureInterface :: HackageFeature,
itemUpdate :: Hook (Set PackageName) (),
constructItemIndex :: IO (Map PackageName PackageItem),
makeItemList :: [PackageName] -> IO [PackageItem],
makeItemMap :: forall a. Map PackageName a -> IO (Map PackageName (PackageItem, a)),
getAllLists :: IO (Map PackageName PackageItem)
}
instance IsHackageFeature ListFeature where
getFeatureInterface = listFeatureInterface
data PackageItem = PackageItem {
-- The name of the package
itemName :: !PackageName,
-- The tags for this package
itemTags :: !(Set Tag),
-- If the package is deprecated, what is it deprecated in favor of
itemDeprecated :: !(Maybe [PackageName]),
-- The description of the package from its Cabal file
itemDesc :: !String,
-- Maintainer of the package
itemMaintainer :: [UserName],
-- Whether the item is in the Haskell Platform
--itemPlatform :: Bool,
-- Voting score for the package
itemVotes :: !Float,
-- The total number of downloads. (For sorting, not displaying.)
-- Updated periodically.
itemDownloads :: !Int,
-- The number of direct revdeps. (Likewise.)
-- also: distinguish direct/flat?
itemRevDepsCount :: !Int,
-- Whether there's a library here.
itemHasLibrary :: !Bool,
-- How many executables (>=0) this package has.
itemNumExecutables :: !Int,
-- How many test suites (>=0) this package has.
itemNumTests :: !Int,
-- How many benchmarks (>=0) this package has.
itemNumBenchmarks :: !Int,
-- Last upload date
itemLastUpload :: !UTCTime,
-- Hotness = recent downloads + stars + 2 * no rev deps
itemHotness :: !Float,
-- Reference version (non-deprecated highest numbered version)
itemReferenceVersion :: !String,
-- heuristic way to sort packages
itemPackageRank :: !Float
}
instance MemSize PackageItem where
memSize (PackageItem a b c d e f g h i j k l _m n o r) = memSize12 a b c d e f g h i j (k, l, n, o) r
emptyPackageItem :: PackageName -> PackageItem
emptyPackageItem pkg =
PackageItem {
itemName = pkg,
itemTags = Set.empty,
itemDeprecated = Nothing,
itemDesc = "",
itemMaintainer = [],
itemVotes = 0,
itemDownloads = 0,
itemRevDepsCount = 0,
itemHasLibrary = False,
itemNumExecutables = 0,
itemNumTests = 0,
itemNumBenchmarks = 0,
itemLastUpload = UTCTime (toEnum 0) 0,
itemHotness = 0,
itemReferenceVersion = "",
itemPackageRank = 0
}
initListFeature :: ServerEnv
-> IO (CoreFeature
-> ReverseFeature
-> DownloadFeature
-> VotesFeature
-> TagsFeature
-> VersionsFeature
-> UserFeature
-> UploadFeature
-> DocumentationFeature
-> TarIndexCacheFeature
-> IO ListFeature)
initListFeature _env = do
itemCache <- newMemStateWHNF Map.empty
itemUpdate <- newHook
return $ \core@CoreFeature{..}
revs@ReverseFeature{revDirectCount, reverseHook}
download
votesf@VotesFeature{..}
tagsf@TagsFeature{..}
versions@VersionsFeature{..}
users@UserFeature{..}
uploads@UploadFeature{..}
documentation tar -> do
let (feature, modifyItem, updateDesc) =
listFeature core revs download votesf tagsf versions users uploads
itemCache itemUpdate documentation tar _env
registerHookJust packageChangeHook isPackageChangeAny $ \(pkgid, _) ->
updateDesc (packageName pkgid)
registerHookJust packageChangeHook isPackageAdd $ \pkg -> do
let pkgname = packageName . packageId $ pkg
prefsinfo <- queryGetPreferredInfo pkgname
index <- queryGetPackageIndex
let allVersions = packageVersion <$> PackageIndex.lookupPackageName index pkgname
modifyItem pkgname $ \x ->
updateReferenceVersion prefsinfo allVersions $
x
{ itemLastUpload = fst (pkgOriginalUploadInfo pkg)
}
runHook_ itemUpdate (Set.singleton pkgname)
registerHook groupChangedHook $ \(gd,_,_,_,_) ->
case fmap (mkPackageName . fst) (groupEntity gd) of
Just pkgname -> do
maintainers <- queryUserGroup (maintainersGroup pkgname)
users' <- queryGetUserDb
modifyItem pkgname (\x -> x {itemMaintainer = map (userIdToName users') (UserIdSet.toList maintainers)})
runHook_ itemUpdate (Set.singleton pkgname)
Nothing -> return ()
registerHook reverseHook $ \pkginfos -> do
let
names = Set.fromDistinctAscList $
map (pkgName . pkgInfoId . NE.head)
pkginfos
forM_ names $ \pkgname -> do
revDirect <- revDirectCount pkgname
modifyItem pkgname (updateReverseItem revDirect)
runHook_ itemUpdate names
registerHook votesUpdated $ \(pkgname, _) -> do
votes <- pkgNumScore pkgname
modifyItem pkgname (updateVoteItem votes)
runHook_ itemUpdate (Set.singleton pkgname)
registerHook tagsUpdated $ \(pkgs, _) -> do
forM_ (Set.toList pkgs) $ \pkgname -> do
tags <- queryTagsForPackage pkgname
modifyItem pkgname (updateTagItem tags)
runHook_ itemUpdate pkgs
registerHook deprecatedHook $ \(pkgname, mpkgs) -> do
modifyItem pkgname (updateDeprecation mpkgs)
runHook_ itemUpdate (Set.singleton pkgname)
registerHook updatePreferredHook $ \(pkgname, prefsinfo) -> do
index <- queryGetPackageIndex
let allVersions = packageVersion <$> PackageIndex.lookupPackageName index pkgname
modifyItem pkgname $ updateReferenceVersion prefsinfo allVersions
return feature
listFeature :: CoreFeature
-> ReverseFeature
-> DownloadFeature
-> VotesFeature
-> TagsFeature
-> VersionsFeature
-> UserFeature
-> UploadFeature
-> MemState (Map PackageName PackageItem)
-> Hook (Set PackageName) ()
-> DocumentationFeature
-> TarIndexCacheFeature
-> ServerEnv
-> (ListFeature,
PackageName -> (PackageItem -> PackageItem) -> IO (),
PackageName -> IO ())
listFeature CoreFeature{..}
ReverseFeature{revDirectCount, revPackageStats}
DownloadFeature{..}
VotesFeature{..}
TagsFeature{..}
versions@VersionsFeature{..}
UserFeature{..}
UploadFeature{..}
itemCache itemUpdate
documentation tar env
= (ListFeature{..}, modifyItem, updateDesc)
where
listFeatureInterface = (emptyHackageFeature "list") {
featurePostInit = do itemsCache
void $ forkIO periodicDownloadRefresh
, featureState = []
, featureCaches = [
CacheComponent {
cacheDesc = "per-package-name summary info",
getCacheMemSize = memSize <$> readMemState itemCache
}
]
}
where itemsCache = do
items <- constructItemIndex
writeMemState itemCache items
periodicDownloadRefresh = forever $ do
--FIXME: don't do this if nothing has changed!
threadDelay (30 * 60 * 1000000) -- 30 minutes
refreshDownloads
modifyItem pkgname token = do
hasItem <- fmap (Map.member pkgname) $ readMemState itemCache
case hasItem of
True -> modifyMemState itemCache $ Map.adjust token pkgname
False -> do
index <- queryGetPackageIndex
let pkgs = PackageIndex.lookupPackageName index pkgname
case pkgs of
[] -> return () --this shouldn't happen
_ -> modifyMemState itemCache . uncurry Map.insert =<< constructItem pkgs
updateDesc pkgname = do
index <- queryGetPackageIndex
let pkgs = PackageIndex.lookupPackageName index pkgname
case pkgs of
[] -> modifyMemState itemCache (Map.delete pkgname)
_ -> modifyItem pkgname (updateDescriptionItem $ pkgDesc $ last pkgs)
runHook_ itemUpdate (Set.singleton pkgname)
refreshDownloads = do
downs <- recentPackageDownloads
modifyMemState itemCache $ Map.mapWithKey $ \pkg item ->
updateDownload (cmFind pkg downs) item
-- Say all packages were updated here (detecting this is more laborious)
mainMap <- readMemState itemCache
runHook_ itemUpdate (Set.fromDistinctAscList $ Map.keys mainMap)
constructItemIndex :: IO (Map PackageName PackageItem)
constructItemIndex = do
index <- queryGetPackageIndex
items <- mapM constructItem $ PackageIndex.allPackagesByName index
return $ Map.fromList items
constructItem :: [PkgInfo] -> IO (PackageName, PackageItem)
constructItem pkgs = do
let pkgname = packageName pkg
desc = pkgDesc pkg
pkg = last pkgs
-- [reverse index disabled] revCount <- query . GetReverseCount $ pkgname
revCount@(ReverseCount intRevDirectCount _) <- revPackageStats pkgname
users <- queryGetUserDb
tags <- queryTagsForPackage pkgname
downs <- recentPackageDownloads
votes <- pkgNumScore pkgname
deprs <- queryGetDeprecatedFor pkgname
maintainers <- queryUserGroup (maintainersGroup pkgname)
prefsinfo <- queryGetPreferredInfo pkgname
packageR <- rankPackage versions (cmFind pkgname downs) (UserIdSet.size maintainers)
documentation tar env pkgs (safeLast pkgs) revCount
return $ (,) pkgname . updateReferenceVersion prefsinfo [pkgVersion (pkgInfoId pkg)] $ (updateDescriptionItem desc $ emptyPackageItem pkgname) {
itemTags = tags
, itemMaintainer = map (userIdToName users) (UserIdSet.toList maintainers)
, itemDeprecated = deprs
, itemDownloads = cmFind pkgname downs
, itemVotes = votes
, itemLastUpload = fst (pkgOriginalUploadInfo pkg)
, itemRevDepsCount = intRevDirectCount
, itemHotness = votes + fromIntegral (cmFind pkgname downs) + fromIntegral intRevDirectCount * 2
, itemPackageRank = packageR
}
------------------------------
makeItemList :: [PackageName] -> IO [PackageItem]
makeItemList pkgnames = do
mainMap <- readMemState itemCache
return $ mapMaybe (flip Map.lookup mainMap) pkgnames
makeItemMap :: Map PackageName a -> IO (Map PackageName (PackageItem, a))
makeItemMap pkgmap = do
mainMap <- readMemState itemCache
return $ Map.intersectionWith (,) mainMap pkgmap
getAllLists :: IO (Map PackageName PackageItem)
getAllLists = readMemState itemCache
tagHistogram :: [PackageItem] -> Map Tag Int
tagHistogram = Map.fromListWith (+) . map (flip (,) 1) . concatMap (Set.toList . itemTags)
updateDescriptionItem :: GenericPackageDescription -> PackageItem -> PackageItem
updateDescriptionItem genDesc item =
let desc = flattenPackageDescription genDesc
in item {
itemDesc = fromShortText $ synopsis desc,
-- This checks if the library is buildable. However, since
-- desc is flattened, we might miss some flags. Perhaps use the
-- CondTree instead.
-- itemMaintainer = maintainer desc,
itemHasLibrary = hasLibs desc,
itemNumExecutables = length . filter (buildable . buildInfo) $ executables desc,
itemNumTests = length . filter (buildable . testBuildInfo) $ testSuites desc,
itemNumBenchmarks = length . filter (buildable . benchmarkBuildInfo) $ benchmarks desc
}
updateTagItem :: Set Tag -> PackageItem -> PackageItem
updateTagItem tags item =
item {
itemTags = tags
}
updateVoteItem :: Float -> PackageItem -> PackageItem
updateVoteItem score item =
item {
itemVotes = score,
itemHotness = fromIntegral (itemRevDepsCount item) * 2 + score + fromIntegral (itemDownloads item)
}
updateDeprecation :: Maybe [PackageName] -> PackageItem -> PackageItem
updateDeprecation pkgs item =
item {
itemDeprecated = pkgs
}
updateReferenceVersion :: PreferredInfo -> [Version] -> PackageItem -> PackageItem
updateReferenceVersion prefsinfo allVersions item =
item {
itemReferenceVersion =
case nonDeprecatedVersion of
[] -> ""
xs -> prettyShow $ maximum xs
}
where
nonDeprecatedVersion = filter (`notElem` deprecatedVersions prefsinfo) allVersions
updateReverseItem :: Int -> PackageItem -> PackageItem
updateReverseItem revDirectCount item =
item {
itemRevDepsCount = revDirectCount,
itemHotness = fromIntegral revDirectCount * 2 + itemVotes item + fromIntegral (itemDownloads item)
}
updateDownload :: Int -> PackageItem -> PackageItem
updateDownload count item =
item {
itemDownloads = count,
itemHotness = fromIntegral (itemRevDepsCount item) * 2 + itemVotes item + realToFrac count
}