You can not select more than 25 topics Topics must start with a chinese character,a letter or number, can include dashes ('-') and can be up to 35 characters long.

user_business_analysis.go 16 kB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474
  1. package models
  2. import (
  3. "fmt"
  4. "time"
  5. "code.gitea.io/gitea/modules/log"
  6. "code.gitea.io/gitea/modules/timeutil"
  7. )
  8. type UserBusinessAnalysis struct {
  9. ID int64 `xorm:"pk"`
  10. CountDate int64 `xorm:"pk"`
  11. //action :ActionMergePullRequest // 11
  12. CodeMergeCount int `xorm:"NOT NULL DEFAULT 0"`
  13. //action :ActionCommitRepo // 5
  14. CommitCount int `xorm:"NOT NULL DEFAULT 0"`
  15. //action :ActionCommentIssue // 10
  16. IssueCount int `xorm:"NOT NULL DEFAULT 0"`
  17. //comment table current date
  18. CommentCount int `xorm:"NOT NULL DEFAULT 0"`
  19. //watch table current date
  20. FocusRepoCount int `xorm:"NOT NULL DEFAULT 0"`
  21. //star table current date
  22. StarRepoCount int `xorm:"NOT NULL DEFAULT 0"`
  23. //follow table
  24. WatchedCount int `xorm:"NOT NULL DEFAULT 0"`
  25. // user table
  26. GiteaAgeMonth int `xorm:"NOT NULL DEFAULT 0"`
  27. //
  28. CommitCodeSize int `xorm:"NOT NULL DEFAULT 0"`
  29. //attachement table
  30. CommitDatasetSize int `xorm:"NOT NULL DEFAULT 0"`
  31. //0
  32. CommitModelCount int `xorm:"NOT NULL DEFAULT 0"`
  33. //issue, issueassignees
  34. SolveIssueCount int `xorm:"NOT NULL DEFAULT 0"`
  35. //baike
  36. EncyclopediasCount int `xorm:"NOT NULL DEFAULT 0"`
  37. //user
  38. RegistDate timeutil.TimeStamp `xorm:"NOT NULL"`
  39. //repo
  40. CreateRepoCount int `xorm:"NOT NULL DEFAULT 0"`
  41. //login count, from elk
  42. LoginCount int `xorm:"NOT NULL DEFAULT 0"`
  43. //openi index
  44. OpenIIndex int `xorm:"NOT NULL DEFAULT 0"`
  45. //user
  46. Email string `xorm:"NOT NULL"`
  47. //user
  48. Name string `xorm:"NOT NULL"`
  49. }
  50. func QueryUserStaticData(startTime int64, endTime int64) []*UserBusinessAnalysis {
  51. log.Info("query startTime =" + fmt.Sprint(startTime) + " endTime=" + fmt.Sprint(endTime))
  52. statictisSess := xStatistic.NewSession()
  53. defer statictisSess.Close()
  54. statictisSess.Select("*").Table("user_business_analysis").Where(" count_date>=" + fmt.Sprint(startTime) + " and count_date<=" + fmt.Sprint(endTime)).OrderBy("count_date desc")
  55. userBusinessAnalysisList := make([]*UserBusinessAnalysis, 0)
  56. statictisSess.Find(&userBusinessAnalysisList)
  57. resultMap := make(map[int64]*UserBusinessAnalysis)
  58. log.Info("query result size=" + fmt.Sprint(len(userBusinessAnalysisList)))
  59. for _, userRecord := range userBusinessAnalysisList {
  60. if _, ok := resultMap[userRecord.ID]; !ok {
  61. resultMap[userRecord.ID] = userRecord
  62. } else {
  63. resultMap[userRecord.ID].CodeMergeCount += userRecord.CodeMergeCount
  64. resultMap[userRecord.ID].CommitCount += userRecord.CommitCount
  65. resultMap[userRecord.ID].IssueCount += userRecord.IssueCount
  66. resultMap[userRecord.ID].CommentCount += userRecord.CommentCount
  67. resultMap[userRecord.ID].FocusRepoCount += userRecord.FocusRepoCount
  68. resultMap[userRecord.ID].StarRepoCount += userRecord.StarRepoCount
  69. resultMap[userRecord.ID].WatchedCount += userRecord.WatchedCount
  70. resultMap[userRecord.ID].CommitCodeSize += userRecord.CommitCodeSize
  71. resultMap[userRecord.ID].CommitDatasetSize += userRecord.CommitDatasetSize
  72. resultMap[userRecord.ID].CommitModelCount += userRecord.CommitModelCount
  73. resultMap[userRecord.ID].SolveIssueCount += userRecord.SolveIssueCount
  74. resultMap[userRecord.ID].EncyclopediasCount += userRecord.EncyclopediasCount
  75. resultMap[userRecord.ID].CreateRepoCount += userRecord.CreateRepoCount
  76. resultMap[userRecord.ID].LoginCount += userRecord.LoginCount
  77. }
  78. }
  79. userBusinessAnalysisReturnList := make([]*UserBusinessAnalysis, len(resultMap))
  80. index := 0
  81. for _, v := range resultMap {
  82. userBusinessAnalysisReturnList[index] = v
  83. index += 1
  84. }
  85. log.Info("return size=" + fmt.Sprint(len(userBusinessAnalysisReturnList)))
  86. return userBusinessAnalysisReturnList
  87. }
  88. func CounDataByDate(wikiCountMap map[string]int, startTime time.Time, endTime time.Time) {
  89. log.Info("start to count other user info data")
  90. sess := x.NewSession()
  91. defer sess.Close()
  92. sess.Select("`user`.*").Table("user")
  93. userList := make([]*User, 0)
  94. sess.Find(&userList)
  95. currentTimeNow := time.Now()
  96. log.Info("current time:" + currentTimeNow.Format("2006-01-02 15:04:05"))
  97. //yesterday := currentTimeNow.AddDate(0, 0, -1)
  98. //startTime := time.Date(yesterday.Year(), yesterday.Month(), yesterday.Day(), 0, 0, 0, 0, yesterday.Location())
  99. start_unix := startTime.Unix()
  100. log.Info("DB query time:" + startTime.Format("2006-01-02 15:04:05"))
  101. //endTime := time.Date(currentTimeNow.Year(), currentTimeNow.Month(), currentTimeNow.Day(), 0, 0, 0, 0, currentTimeNow.Location())
  102. end_unix := endTime.Unix()
  103. CountDate := time.Date(currentTimeNow.Year(), currentTimeNow.Month(), currentTimeNow.Day(), 0, 1, 0, 0, currentTimeNow.Location())
  104. CodeMergeCountMap := queryPullRequest(start_unix, end_unix)
  105. CommitCountMap := queryAction(start_unix, end_unix, 5)
  106. IssueCountMap := queryAction(start_unix, end_unix, 10)
  107. CommentCountMap := queryComment(start_unix, end_unix)
  108. FocusRepoCountMap := queryWatch(start_unix, end_unix)
  109. StarRepoCountMap := queryStar(start_unix, end_unix)
  110. WatchedCountMap := queryFollow(start_unix, end_unix)
  111. CommitCodeSizeMap, err := GetAllUserKPIStats()
  112. if err != nil {
  113. log.Info("query commit code errr.")
  114. } else {
  115. log.Info("query commit code size, len=" + fmt.Sprint(len(CommitCodeSizeMap)))
  116. }
  117. CommitDatasetSizeMap := queryDatasetSize(start_unix, end_unix)
  118. SolveIssueCountMap := querySolveIssue(start_unix, end_unix)
  119. CreateRepoCountMap := queryUserCreateRepo(start_unix, end_unix)
  120. LoginCountMap := queryLoginCount(start_unix, end_unix)
  121. statictisSess := xStatistic.NewSession()
  122. defer statictisSess.Close()
  123. for i, userRecord := range userList {
  124. var dateRecord UserBusinessAnalysis
  125. dateRecord.ID = userRecord.ID
  126. log.Info("i=" + fmt.Sprint(i) + " userName=" + userRecord.Name)
  127. dateRecord.CountDate = CountDate.Unix()
  128. statictisSess.Delete(&dateRecord)
  129. dateRecord.Email = userRecord.Email
  130. dateRecord.RegistDate = userRecord.CreatedUnix
  131. dateRecord.Name = userRecord.Name
  132. dateRecord.GiteaAgeMonth = subMonth(currentTimeNow, userRecord.CreatedUnix.AsTime())
  133. if _, ok := CodeMergeCountMap[dateRecord.ID]; !ok {
  134. dateRecord.CodeMergeCount = 0
  135. } else {
  136. dateRecord.CodeMergeCount = CodeMergeCountMap[dateRecord.ID]
  137. }
  138. if _, ok := CommitCountMap[dateRecord.ID]; !ok {
  139. dateRecord.CommitCount = 0
  140. } else {
  141. dateRecord.CommitCount = CommitCountMap[dateRecord.ID]
  142. }
  143. if _, ok := IssueCountMap[dateRecord.ID]; !ok {
  144. dateRecord.IssueCount = 0
  145. } else {
  146. dateRecord.IssueCount = IssueCountMap[dateRecord.ID]
  147. }
  148. if _, ok := CommentCountMap[dateRecord.ID]; !ok {
  149. dateRecord.CommentCount = 0
  150. } else {
  151. dateRecord.CommentCount = CommentCountMap[dateRecord.ID]
  152. }
  153. if _, ok := FocusRepoCountMap[dateRecord.ID]; !ok {
  154. dateRecord.FocusRepoCount = 0
  155. } else {
  156. dateRecord.FocusRepoCount = FocusRepoCountMap[dateRecord.ID]
  157. }
  158. if _, ok := StarRepoCountMap[dateRecord.ID]; !ok {
  159. dateRecord.StarRepoCount = 0
  160. } else {
  161. dateRecord.StarRepoCount = StarRepoCountMap[dateRecord.ID]
  162. }
  163. if _, ok := WatchedCountMap[dateRecord.ID]; !ok {
  164. dateRecord.WatchedCount = 0
  165. } else {
  166. dateRecord.WatchedCount = WatchedCountMap[dateRecord.ID]
  167. }
  168. if _, ok := CommitCodeSizeMap[dateRecord.Email]; !ok {
  169. dateRecord.CommitCodeSize = 0
  170. } else {
  171. dateRecord.CommitCodeSize = int(CommitCodeSizeMap[dateRecord.Email].CommitLines)
  172. }
  173. if _, ok := CommitDatasetSizeMap[dateRecord.ID]; !ok {
  174. dateRecord.CommitDatasetSize = 0
  175. } else {
  176. dateRecord.CommitDatasetSize = CommitDatasetSizeMap[dateRecord.ID]
  177. }
  178. if _, ok := SolveIssueCountMap[dateRecord.ID]; !ok {
  179. dateRecord.SolveIssueCount = 0
  180. } else {
  181. dateRecord.SolveIssueCount = SolveIssueCountMap[dateRecord.ID]
  182. }
  183. if _, ok := wikiCountMap[dateRecord.Name]; !ok {
  184. dateRecord.EncyclopediasCount = 0
  185. } else {
  186. dateRecord.EncyclopediasCount = wikiCountMap[dateRecord.Name]
  187. }
  188. if _, ok := CreateRepoCountMap[dateRecord.ID]; !ok {
  189. dateRecord.CreateRepoCount = 0
  190. } else {
  191. dateRecord.CreateRepoCount = CreateRepoCountMap[dateRecord.ID]
  192. }
  193. if _, ok := LoginCountMap[dateRecord.ID]; !ok {
  194. dateRecord.LoginCount = 0
  195. } else {
  196. dateRecord.LoginCount = LoginCountMap[dateRecord.ID]
  197. }
  198. dateRecord.CommitModelCount = 0
  199. statictisSess.Insert(&dateRecord)
  200. }
  201. }
  202. func querySolveIssue(start_unix int64, end_unix int64) map[int64]int {
  203. //select issue_assignees.* from issue_assignees,issue where issue.is_closed=true and issue.id=issue_assignees.issue_id
  204. sess := x.NewSession()
  205. defer sess.Close()
  206. sess.Select("issue_assignees.*").Table("issue_assignees").
  207. Join("inner", "issue", "issue.id=issue_assignees.issue_id").
  208. Where("issue.is_closed=true and issue.closed_unix>=" + fmt.Sprint(start_unix) + " and issue.closed_unix<=" + fmt.Sprint(end_unix))
  209. issueAssigneesList := make([]*IssueAssignees, 0)
  210. sess.Find(&issueAssigneesList)
  211. resultMap := make(map[int64]int)
  212. log.Info("query IssueAssignees size=" + fmt.Sprint(len(issueAssigneesList)))
  213. for _, issueAssigneesRecord := range issueAssigneesList {
  214. if _, ok := resultMap[issueAssigneesRecord.AssigneeID]; !ok {
  215. resultMap[issueAssigneesRecord.AssigneeID] = 1
  216. } else {
  217. resultMap[issueAssigneesRecord.AssigneeID] += 1
  218. }
  219. }
  220. return resultMap
  221. }
  222. func queryPullRequest(start_unix int64, end_unix int64) map[int64]int {
  223. sess := x.NewSession()
  224. defer sess.Close()
  225. sess.Select("issue.*").Table("issue").
  226. Join("inner", "pull_request", "issue.id=pull_request.issue_id").
  227. Where("pull_request.merged_unix>=" + fmt.Sprint(start_unix) + " and pull_request.merged_unix<=" + fmt.Sprint(end_unix))
  228. issueList := make([]*Issue, 0)
  229. sess.Find(&issueList)
  230. resultMap := make(map[int64]int)
  231. log.Info("query issue(PR) size=" + fmt.Sprint(len(issueList)))
  232. for _, issueRecord := range issueList {
  233. if _, ok := resultMap[issueRecord.PosterID]; !ok {
  234. resultMap[issueRecord.PosterID] = 1
  235. } else {
  236. resultMap[issueRecord.PosterID] += 1
  237. }
  238. }
  239. return resultMap
  240. }
  241. func queryAction(start_unix int64, end_unix int64, actionType int64) map[int64]int {
  242. sess := x.NewSession()
  243. defer sess.Close()
  244. sess.Select("id,user_id,op_type,act_user_id").Table("action").Where("op_type=" + fmt.Sprint(actionType) + " and created_unix>=" + fmt.Sprint(start_unix) + " and created_unix<=" + fmt.Sprint(end_unix))
  245. actionList := make([]*Action, 0)
  246. sess.Find(&actionList)
  247. resultMap := make(map[int64]int)
  248. log.Info("query action size=" + fmt.Sprint(len(actionList)))
  249. for _, actionRecord := range actionList {
  250. if _, ok := resultMap[actionRecord.UserID]; !ok {
  251. resultMap[actionRecord.UserID] = 1
  252. } else {
  253. resultMap[actionRecord.UserID] += 1
  254. }
  255. }
  256. return resultMap
  257. }
  258. func queryComment(start_unix int64, end_unix int64) map[int64]int {
  259. sess := x.NewSession()
  260. defer sess.Close()
  261. sess.Select("id,type,poster_id").Table("comment").Where(" created_unix>=" + fmt.Sprint(start_unix) + " and created_unix<=" + fmt.Sprint(end_unix))
  262. commentList := make([]*Comment, 0)
  263. sess.Find(&commentList)
  264. resultMap := make(map[int64]int)
  265. log.Info("query Comment size=" + fmt.Sprint(len(commentList)))
  266. for _, commentRecord := range commentList {
  267. if _, ok := resultMap[commentRecord.PosterID]; !ok {
  268. resultMap[commentRecord.PosterID] = 1
  269. } else {
  270. resultMap[commentRecord.PosterID] += 1
  271. }
  272. }
  273. return resultMap
  274. }
  275. func queryWatch(start_unix int64, end_unix int64) map[int64]int {
  276. sess := x.NewSession()
  277. defer sess.Close()
  278. sess.Select("id,user_id,repo_id").Table("watch").Where(" created_unix>=" + fmt.Sprint(start_unix) + " and created_unix<=" + fmt.Sprint(end_unix))
  279. watchList := make([]*Watch, 0)
  280. sess.Find(&watchList)
  281. resultMap := make(map[int64]int)
  282. log.Info("query Watch size=" + fmt.Sprint(len(watchList)))
  283. for _, watchRecord := range watchList {
  284. if _, ok := resultMap[watchRecord.UserID]; !ok {
  285. resultMap[watchRecord.UserID] = 1
  286. } else {
  287. resultMap[watchRecord.UserID] += 1
  288. }
  289. }
  290. return resultMap
  291. }
  292. func queryStar(start_unix int64, end_unix int64) map[int64]int {
  293. sess := x.NewSession()
  294. defer sess.Close()
  295. sess.Select("id,uid,repo_id").Table("star").Where(" created_unix>=" + fmt.Sprint(start_unix) + " and created_unix<=" + fmt.Sprint(end_unix))
  296. starList := make([]*Star, 0)
  297. sess.Find(&starList)
  298. resultMap := make(map[int64]int)
  299. log.Info("query Star size=" + fmt.Sprint(len(starList)))
  300. for _, starRecord := range starList {
  301. if _, ok := resultMap[starRecord.UID]; !ok {
  302. resultMap[starRecord.UID] = 1
  303. } else {
  304. resultMap[starRecord.UID] += 1
  305. }
  306. }
  307. return resultMap
  308. }
  309. func queryFollow(start_unix int64, end_unix int64) map[int64]int {
  310. sess := x.NewSession()
  311. defer sess.Close()
  312. sess.Select("id,user_id,follow_id").Table("follow").Where(" created_unix>=" + fmt.Sprint(start_unix) + " and created_unix<=" + fmt.Sprint(end_unix))
  313. followList := make([]*Follow, 0)
  314. sess.Find(&followList)
  315. resultMap := make(map[int64]int)
  316. log.Info("query Follow size=" + fmt.Sprint(len(followList)))
  317. for _, followRecord := range followList {
  318. if _, ok := resultMap[followRecord.UserID]; !ok {
  319. resultMap[followRecord.UserID] = 1
  320. } else {
  321. resultMap[followRecord.UserID] += 1
  322. }
  323. }
  324. return resultMap
  325. }
  326. func queryDatasetSize(start_unix int64, end_unix int64) map[int64]int {
  327. sess := x.NewSession()
  328. defer sess.Close()
  329. sess.Select("id,uploader_id,size").Table("attachment").Where(" created_unix>=" + fmt.Sprint(start_unix) + " and created_unix<=" + fmt.Sprint(end_unix))
  330. attachmentList := make([]*Attachment, 0)
  331. sess.Find(&attachmentList)
  332. resultMap := make(map[int64]int)
  333. log.Info("query Attachment size=" + fmt.Sprint(len(attachmentList)))
  334. for _, attachRecord := range attachmentList {
  335. if _, ok := resultMap[attachRecord.UploaderID]; !ok {
  336. resultMap[attachRecord.UploaderID] = int(attachRecord.Size / (1024 * 1024)) //MB
  337. } else {
  338. resultMap[attachRecord.UploaderID] += int(attachRecord.Size / (1024 * 1024)) //MB
  339. }
  340. }
  341. return resultMap
  342. }
  343. func queryUserCreateRepo(start_unix int64, end_unix int64) map[int64]int {
  344. sess := x.NewSession()
  345. defer sess.Close()
  346. sess.Select("id,owner_id,name").Table("repository").Where("is_fork=false and created_unix>=" + fmt.Sprint(start_unix) + " and created_unix<=" + fmt.Sprint(end_unix))
  347. repoList := make([]*Repository, 0)
  348. sess.Find(&repoList)
  349. resultMap := make(map[int64]int)
  350. log.Info("query Repository size=" + fmt.Sprint(len(repoList)))
  351. for _, repoRecord := range repoList {
  352. if _, ok := resultMap[repoRecord.OwnerID]; !ok {
  353. resultMap[repoRecord.OwnerID] = 1
  354. } else {
  355. resultMap[repoRecord.OwnerID] += 1
  356. }
  357. }
  358. return resultMap
  359. }
  360. func queryLoginCount(start_unix int64, end_unix int64) map[int64]int {
  361. statictisSess := xStatistic.NewSession()
  362. defer statictisSess.Close()
  363. statictisSess.Select("id,u_id").Table("user_login_log").Where("created_unix>=" + fmt.Sprint(start_unix) + " and created_unix<=" + fmt.Sprint(end_unix))
  364. userLoginLogList := make([]*UserLoginLog, 0)
  365. statictisSess.Find(&userLoginLogList)
  366. resultMap := make(map[int64]int)
  367. log.Info("query user login size=" + fmt.Sprint(len(userLoginLogList)))
  368. for _, loginRecord := range userLoginLogList {
  369. if _, ok := resultMap[loginRecord.UId]; !ok {
  370. resultMap[loginRecord.UId] = 1
  371. } else {
  372. resultMap[loginRecord.UId] += 1
  373. }
  374. }
  375. return resultMap
  376. }
  377. func subMonth(t1, t2 time.Time) (month int) {
  378. y1 := t1.Year()
  379. y2 := t2.Year()
  380. m1 := int(t1.Month())
  381. m2 := int(t2.Month())
  382. d1 := t1.Day()
  383. d2 := t2.Day()
  384. yearInterval := y1 - y2
  385. // 如果 d1的 月-日 小于 d2的 月-日 那么 yearInterval-- 这样就得到了相差的年数
  386. if m1 < m2 || m1 == m2 && d1 < d2 {
  387. yearInterval--
  388. }
  389. // 获取月数差值
  390. monthInterval := (m1 + 12) - m2
  391. if d1 < d2 {
  392. monthInterval--
  393. }
  394. monthInterval %= 12
  395. month = yearInterval*12 + monthInterval
  396. return month
  397. }