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.

repo_dashbord.go 13 kB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384
  1. package repo
  2. import (
  3. "fmt"
  4. "net/http"
  5. "strconv"
  6. "time"
  7. "code.gitea.io/gitea/models"
  8. "code.gitea.io/gitea/modules/log"
  9. "code.gitea.io/gitea/modules/context"
  10. "code.gitea.io/gitea/modules/setting"
  11. )
  12. const DEFAULT_PAGE_SIZE = 10
  13. const DATE_FORMAT = "2006-01-02"
  14. type ProjectsPeriodData struct {
  15. RecordBeginTime string `json:"recordBeginTime"`
  16. LastUpdatedTime string `json:"lastUpdatedTime"`
  17. PageSize int `json:"pageSize"`
  18. TotalPage int `json:"totalPage"`
  19. PageRecords []*models.RepoStatistic `json:"pageRecords"`
  20. }
  21. type UserInfo struct {
  22. User string `json:"user"`
  23. Mode int `json:"mode"`
  24. PR int64 `json:"pr"`
  25. Commit int `json:"commit"`
  26. }
  27. type ProjectLatestData struct {
  28. RecordBeginTime string `json:"recordBeginTime"`
  29. LastUpdatedTime string `json:"lastUpdatedTime"`
  30. CreatTime string `json:"creatTime"`
  31. OpenI float64 `json:"openi"`
  32. Comment int64 `json:"comment"`
  33. View int64 `json:"view"`
  34. Download int64 `json:"download"`
  35. IssueClosedRatio float32 `json:"issueClosedRatio"`
  36. Impact float64 `json:"impact"`
  37. Completeness float64 `json:"completeness"`
  38. Liveness float64 `json:"liveness"`
  39. ProjectHealth float64 `json:"projectHealth"`
  40. TeamHealth float64 `json:"teamHealth"`
  41. Growth float64 `json:"growth"`
  42. Description string `json:"description"`
  43. Top10 []UserInfo `json:"top10"`
  44. }
  45. func RestoreForkNumber(ctx *context.Context) {
  46. repos, err := models.GetAllRepositories()
  47. if err != nil {
  48. log.Error("GetAllRepositories failed: %v", err.Error())
  49. return
  50. }
  51. for _, repo := range repos {
  52. models.RestoreRepoStatFork(int64(repo.NumForks), repo.ID)
  53. }
  54. ctx.JSON(http.StatusOK, struct{}{})
  55. }
  56. func GetAllProjectsPeriodStatistics(ctx *context.Context) {
  57. recordBeginTime, err := getRecordBeginTime()
  58. if err != nil {
  59. log.Error("Can not get record begin time", err)
  60. ctx.Error(http.StatusBadRequest, ctx.Tr("repo.record_begintime_get_err"))
  61. return
  62. }
  63. beginTime, endTime, err := getTimePeroid(ctx, recordBeginTime)
  64. if err != nil {
  65. log.Error("Parameter is wrong", err)
  66. ctx.Error(http.StatusBadRequest, ctx.Tr("repo.parameter_is_wrong"))
  67. return
  68. }
  69. q := ctx.QueryTrim("q")
  70. page := ctx.QueryInt("page")
  71. if page <= 0 {
  72. page = 1
  73. }
  74. pageSize := ctx.QueryInt("pagesize")
  75. if pageSize <= 0 {
  76. pageSize = DEFAULT_PAGE_SIZE
  77. }
  78. orderBy := getOrderBy(ctx)
  79. latestUpdatedTime, latestDate, err := models.GetRepoStatLastUpdatedTime()
  80. if err != nil {
  81. log.Error("Can not query the last updated time.", err)
  82. ctx.Error(http.StatusBadRequest, ctx.Tr("repo.last_update_time_error"))
  83. return
  84. }
  85. countSql := generateCountSql(beginTime, endTime, latestDate, q)
  86. total, err := models.CountRepoStatByRawSql(countSql)
  87. if err != nil {
  88. log.Error("Can not query total count.", err)
  89. ctx.Error(http.StatusBadRequest, ctx.Tr("repo.total_count_get_error"))
  90. return
  91. }
  92. projectsPeriodData := ProjectsPeriodData{
  93. RecordBeginTime: recordBeginTime.Format(DATE_FORMAT),
  94. PageSize: pageSize,
  95. TotalPage: getTotalPage(total, pageSize),
  96. LastUpdatedTime: latestUpdatedTime,
  97. PageRecords: models.GetRepoStatisticByRawSql(generatePageSql(beginTime, endTime, latestDate, q, orderBy, page, pageSize)),
  98. }
  99. ctx.JSON(http.StatusOK, projectsPeriodData)
  100. }
  101. func GetProjectLatestStatistics(ctx *context.Context) {
  102. repoId := ctx.Params(":id")
  103. recordBeginTime, err := getRecordBeginTime()
  104. if err != nil {
  105. log.Error("Can not get record begin time", err)
  106. ctx.Error(http.StatusBadRequest, ctx.Tr("repo.record_begintime_get_err"))
  107. return
  108. }
  109. latestUpdatedTime, latestDate, err := models.GetRepoStatLastUpdatedTime(repoId)
  110. repoIdInt, _ := strconv.ParseInt(repoId, 10, 64)
  111. repoStat, err := models.GetRepoStatisticByDateAndRepoId(latestDate, repoIdInt)
  112. if err != nil {
  113. log.Error("Can not get the repo statistics "+repoId, err)
  114. ctx.Error(http.StatusBadRequest, ctx.Tr("repo.get_repo_stat_error"))
  115. return
  116. }
  117. repository, err := models.GetRepositoryByID(repoIdInt)
  118. if err != nil {
  119. log.Error("Can not get the repo info "+repoId, err)
  120. ctx.Error(http.StatusBadRequest, ctx.Tr("repo.get_repo_info_error"))
  121. return
  122. }
  123. projectLatestData := ProjectLatestData{
  124. RecordBeginTime: recordBeginTime.Format(DATE_FORMAT),
  125. CreatTime: time.Unix(int64(repository.CreatedUnix), 0).Format(DATE_FORMAT),
  126. LastUpdatedTime: latestUpdatedTime,
  127. OpenI: repoStat.RadarTotal,
  128. Comment: repoStat.NumComments,
  129. View: repoStat.NumVisits,
  130. Download: repoStat.NumDownloads,
  131. IssueClosedRatio: repoStat.IssueFixedRate,
  132. Impact: repoStat.Impact,
  133. Completeness: repoStat.Completeness,
  134. Liveness: repoStat.Liveness,
  135. ProjectHealth: repoStat.ProjectHealth,
  136. TeamHealth: repoStat.TeamHealth,
  137. Growth: repoStat.Growth,
  138. Description: repository.Description,
  139. }
  140. contributors, err := models.GetTop10Contributor(repository.RepoPath())
  141. if err != nil {
  142. log.Error("can not get contributors", err)
  143. }
  144. users := make([]UserInfo, 0)
  145. for _, contributor := range contributors {
  146. mode := repository.GetCollaboratorMode(contributor.UserId)
  147. if mode == -1 {
  148. if contributor.IsAdmin {
  149. mode = int(models.AccessModeAdmin)
  150. }
  151. if contributor.UserId == repository.OwnerID {
  152. mode = int(models.AccessModeOwner)
  153. }
  154. }
  155. pr := models.GetPullCountByUserAndRepoId(repoIdInt, contributor.UserId)
  156. userInfo := UserInfo{
  157. User: contributor.Committer,
  158. Commit: contributor.CommitCnt,
  159. Mode: mode,
  160. PR: pr,
  161. }
  162. users = append(users, userInfo)
  163. }
  164. projectLatestData.Top10 = users
  165. ctx.JSON(http.StatusOK, projectLatestData)
  166. }
  167. func GetProjectPeriodStatistics(ctx *context.Context) {
  168. repoId := ctx.Params(":id")
  169. recordBeginTime, err := getRecordBeginTime()
  170. if err != nil {
  171. log.Error("Can not get record begin time", err)
  172. ctx.Error(http.StatusBadRequest, ctx.Tr("repo.record_begintime_get_err"))
  173. return
  174. }
  175. repoIdInt, _ := strconv.ParseInt(repoId, 10, 64)
  176. if err != nil {
  177. log.Error("Can not get record begin time", err)
  178. ctx.Error(http.StatusBadRequest, ctx.Tr("repo.record_begintime_get_err"))
  179. return
  180. }
  181. beginTime, endTime, err := getTimePeroid(ctx, recordBeginTime)
  182. isOpenI := ctx.QueryBool("openi")
  183. var repositorys []*models.RepoStatistic
  184. if isOpenI {
  185. repositorys = models.GetRepoStatisticByRawSql(generateRadarSql(beginTime, endTime, repoIdInt))
  186. } else {
  187. repositorys = models.GetRepoStatisticByRawSql(generateTargetSql(beginTime, endTime, repoIdInt))
  188. }
  189. ctx.JSON(http.StatusOK, repositorys)
  190. }
  191. func generateRadarSql(beginTime time.Time, endTime time.Time, repoId int64) string {
  192. sql := "SELECT date, impact, completeness, liveness, project_health, team_health, growth, radar_total FROM repo_statistic" +
  193. " where repo_id=" + strconv.FormatInt(repoId, 10) + " and created_unix >=" + strconv.FormatInt(beginTime.Unix(), 10) +
  194. " and created_unix<" + strconv.FormatInt(endTime.Unix(), 10)
  195. return sql
  196. }
  197. func generateTargetSql(beginTime time.Time, endTime time.Time, repoId int64) string {
  198. sql := "SELECT date, num_visits,num_downloads,num_commits FROM repo_statistic" +
  199. " where repo_id=" + strconv.FormatInt(repoId, 10) + " and created_unix >=" + strconv.FormatInt(beginTime.Unix(), 10) +
  200. " and created_unix<" + strconv.FormatInt(endTime.Unix(), 10)
  201. return sql
  202. }
  203. func generateCountSql(beginTime time.Time, endTime time.Time, yesterday string, q string) string {
  204. countSql := "SELECT count(*) FROM " +
  205. "(SELECT repo_id FROM repo_statistic where created_unix >=" + strconv.FormatInt(beginTime.Unix(), 10) +
  206. " and created_unix<" + strconv.FormatInt(endTime.Unix(), 10) + " group by repo_id) A," +
  207. "(SELECT repo_id,name,is_private,radar_total from public.repo_statistic where date='" + yesterday + "') B" +
  208. " where A.repo_id=B.repo_id"
  209. if q != "" {
  210. countSql = countSql + " and B.name like '%" + q + "%'"
  211. }
  212. return countSql
  213. }
  214. func generatePageSql(beginTime time.Time, endTime time.Time, yesterday string, q string, orderBy string, page int, pageSize int) string {
  215. countSql := "SELECT A.repo_id,name,is_private,radar_total,num_watches,num_visits,num_downloads,num_pulls,num_commits,num_stars,num_forks,num_issues,num_closed_issues,num_contributor FROM " +
  216. "(SELECT repo_id,sum(num_watches_added) as num_watches,sum(num_visits) as num_visits, sum(num_downloads_added) as num_downloads,sum(num_pulls_added) as num_pulls,sum(num_commits_added) as num_commits,sum(num_stars_added) as num_stars,sum(num_forks_added) num_forks,sum(num_issues_added) as num_issues,sum(num_closed_issues_added) as num_closed_issues,sum(num_contributor_added) as num_contributor " +
  217. " FROM repo_statistic where created_unix >=" + strconv.FormatInt(beginTime.Unix(), 10) +
  218. " and created_unix<" + strconv.FormatInt(endTime.Unix(), 10) + " group by repo_id) A," +
  219. "(SELECT repo_id,name,is_private,radar_total from public.repo_statistic where date='" + yesterday + "') B" +
  220. " where A.repo_id=B.repo_id"
  221. if q != "" {
  222. countSql = countSql + " and B.name like '%" + q + "%'"
  223. }
  224. countSql = countSql + " order by " + orderBy + " desc,A.repo_id" + " limit " + strconv.Itoa(pageSize) + " offset " + strconv.Itoa((page-1)*pageSize)
  225. return countSql
  226. }
  227. func getOrderBy(ctx *context.Context) string {
  228. orderBy := ""
  229. switch ctx.Query("sort") {
  230. case "openi":
  231. orderBy = "B.radar_total"
  232. case "view":
  233. orderBy = "A.num_visits"
  234. case "download":
  235. orderBy = "A.num_downloads"
  236. case "pr":
  237. orderBy = "A.num_pulls"
  238. case "commit":
  239. orderBy = "A.num_commits"
  240. case "watch":
  241. orderBy = "A.num_watches"
  242. case "star":
  243. orderBy = "A.num_stars"
  244. case "fork":
  245. orderBy = "A.num_forks"
  246. case "issue":
  247. orderBy = "A.num_issues"
  248. case "issue_closed":
  249. orderBy = "A.num_closed_issues"
  250. case "contributor":
  251. orderBy = "A.num_contributor"
  252. default:
  253. orderBy = "B.radar_total"
  254. }
  255. return orderBy
  256. }
  257. func getTimePeroid(ctx *context.Context, recordBeginTime time.Time) (time.Time, time.Time, error) {
  258. queryType := ctx.QueryTrim("type")
  259. now := time.Now()
  260. recordBeginTimeTemp := recordBeginTime.AddDate(0, 0, 1)
  261. beginTimeStr := ctx.QueryTrim("beginTime")
  262. endTimeStr := ctx.QueryTrim("endTime")
  263. var beginTime time.Time
  264. var endTime time.Time
  265. if queryType != "" {
  266. if queryType == "all" {
  267. beginTime = recordBeginTimeTemp
  268. endTime = now
  269. } else if queryType == "yesterday" {
  270. endTime = now
  271. beginTime = time.Date(endTime.Year(), endTime.Month(), endTime.Day(), 0, 0, 0, 0, now.Location())
  272. } else if queryType == "current_week" {
  273. beginTime = now.AddDate(0, 0, -int(time.Now().Weekday())+1)
  274. beginTime = time.Date(beginTime.Year(), beginTime.Month(), beginTime.Day(), 0, 0, 0, 0, now.Location())
  275. endTime = now
  276. } else if queryType == "current_month" {
  277. endTime = now
  278. beginTime = time.Date(endTime.Year(), endTime.Month(), 2, 0, 0, 0, 0, now.Location())
  279. } else if queryType == "monthly" {
  280. endTime = now
  281. beginTime = now.AddDate(0, -1, 1)
  282. beginTime = time.Date(beginTime.Year(), beginTime.Month(), beginTime.Day(), 0, 0, 0, 0, now.Location())
  283. } else if queryType == "current_year" {
  284. endTime = now
  285. beginTime = time.Date(endTime.Year(), 1, 2, 0, 0, 0, 0, now.Location())
  286. } else if queryType == "last_month" {
  287. lastMonthTime := now.AddDate(0, -1, 0)
  288. beginTime = time.Date(lastMonthTime.Year(), lastMonthTime.Month(), 2, 0, 0, 0, 0, now.Location())
  289. endTime = time.Date(now.Year(), now.Month(), 2, 0, 0, 0, 0, now.Location())
  290. } else {
  291. return now, now, fmt.Errorf("The value of type parameter is wrong.")
  292. }
  293. } else {
  294. if beginTimeStr == "" || endTimeStr == "" {
  295. //如果查询类型和开始时间结束时间都未设置,按queryType=all处理
  296. beginTime = recordBeginTimeTemp
  297. endTime = now
  298. } else {
  299. beginTime, err := time.Parse("2006-01-02", beginTimeStr)
  300. if err != nil {
  301. return now, now, err
  302. }
  303. endTime, err := time.Parse("2006-01-02", endTimeStr)
  304. if err != nil {
  305. return now, now, err
  306. }
  307. beginTime = beginTime.AddDate(0, 0, 1)
  308. endTime = endTime.AddDate(0, 0, 1)
  309. }
  310. }
  311. if beginTime.Before(recordBeginTimeTemp) {
  312. beginTime = recordBeginTimeTemp
  313. }
  314. return beginTime, endTime, nil
  315. }
  316. func getRecordBeginTime() (time.Time, error) {
  317. return time.Parse(DATE_FORMAT, setting.RadarMap.RecordBeginTime)
  318. }
  319. func getTotalPage(total int64, pageSize int) int {
  320. another := 0
  321. if int(total)%pageSize != 0 {
  322. another = 1
  323. }
  324. return int(total)/pageSize + another
  325. }