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 36 kB

3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
3 years ago
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984
  1. package repo
  2. import (
  3. "fmt"
  4. "net/http"
  5. "net/url"
  6. "strconv"
  7. "strings"
  8. "time"
  9. "github.com/360EntSecGroup-Skylar/excelize/v2"
  10. "code.gitea.io/gitea/models"
  11. "code.gitea.io/gitea/modules/log"
  12. "code.gitea.io/gitea/modules/repository"
  13. "code.gitea.io/gitea/modules/context"
  14. "code.gitea.io/gitea/modules/setting"
  15. )
  16. const DEFAULT_PAGE_SIZE = 10
  17. const DATE_FORMAT = "2006-01-02"
  18. const MONTH_FORMAT = "2006-01"
  19. const EXCEL_DATE_FORMAT = "20060102"
  20. const CREATE_TIME_FORMAT = "2006/01/02 15:04:05"
  21. const UPDATE_TIME_FORMAT = "2006-01-02 15:04:05"
  22. type ProjectsPeriodData struct {
  23. RecordBeginTime string `json:"recordBeginTime"`
  24. LastUpdatedTime string `json:"lastUpdatedTime"`
  25. PageSize int `json:"pageSize"`
  26. TotalPage int `json:"totalPage"`
  27. TotalCount int64 `json:"totalCount"`
  28. PageRecords []*models.RepoStatistic `json:"pageRecords"`
  29. }
  30. type UserInfo struct {
  31. User string `json:"user"`
  32. Mode int `json:"mode"`
  33. PR int64 `json:"pr"`
  34. Commit int `json:"commit"`
  35. RelAvatarLink string `json:"relAvatarLink"`
  36. Email string `json:"email"`
  37. }
  38. type ProjectLatestData struct {
  39. RecordBeginTime string `json:"recordBeginTime"`
  40. LastUpdatedTime string `json:"lastUpdatedTime"`
  41. CreatTime string `json:"creatTime"`
  42. OpenI float64 `json:"openi"`
  43. Comment int64 `json:"comment"`
  44. View int64 `json:"view"`
  45. Download int64 `json:"download"`
  46. IssueClosedRatio float32 `json:"issueClosedRatio"`
  47. Impact float64 `json:"impact"`
  48. Completeness float64 `json:"completeness"`
  49. Liveness float64 `json:"liveness"`
  50. ProjectHealth float64 `json:"projectHealth"`
  51. TeamHealth float64 `json:"teamHealth"`
  52. Growth float64 `json:"growth"`
  53. Description string `json:"description"`
  54. Top10 []UserInfo `json:"top10"`
  55. }
  56. type ProjectSummaryBaseData struct {
  57. NumReposAdd int64 `json:"numReposAdd"`
  58. NumRepoPublicAdd int64 `json:"numRepoPublicAdd"`
  59. NumRepoPrivateAdd int64 `json:"numRepoPrivateAdd"`
  60. NumRepoForkAdd int64 `json:"numRepoForkAdd"`
  61. NumRepoMirrorAdd int64 `json:"numRepoMirrorAdd"`
  62. NumRepoSelfAdd int64 `json:"numRepoSelfAdd"`
  63. NumRepos int64 `json:"numRepos"`
  64. CreatTime string `json:"creatTime"`
  65. }
  66. type ProjectSummaryData struct {
  67. ProjectSummaryBaseData
  68. NumRepoPublic int64 `json:"numRepoPublic"`
  69. NumRepoPrivate int64 `json:"numRepoPrivate"`
  70. NumRepoFork int64 `json:"numRepoFork"`
  71. NumRepoMirror int64 `json:"numRepoMirror"`
  72. NumRepoSelf int64 `json:"numRepoSelf"`
  73. NumRepoOrgAdd int64 `json:"numRepoOrgAdd"`
  74. NumRepoNotOrgAdd int64 `json:"numRepoNotOrgAdd"`
  75. NumRepoOrg int64 `json:"numRepoOrg"`
  76. NumRepoNotOrg int64 `json:"numRepoNotOrg"`
  77. }
  78. type ProjectSummaryPeriodData struct {
  79. RecordBeginTime string `json:"recordBeginTime"`
  80. TotalCount int64 `json:"totalCount"`
  81. PageRecords []*ProjectSummaryBaseData `json:"pageRecords"`
  82. }
  83. func RestoreForkNumber(ctx *context.Context) {
  84. repos, err := models.GetAllRepositories()
  85. if err != nil {
  86. log.Error("GetAllRepositories failed: %v", err.Error())
  87. return
  88. }
  89. for _, repo := range repos {
  90. models.RestoreRepoStatFork(int64(repo.NumForks), repo.ID)
  91. }
  92. ctx.JSON(http.StatusOK, struct{}{})
  93. }
  94. func GetLatestProjectsSummaryData(ctx *context.Context) {
  95. stat, err := models.GetLatest2SummaryStatistic()
  96. data := ProjectSummaryData{}
  97. if err == nil && len(stat) > 0 {
  98. data.NumRepos = stat[0].NumRepos
  99. data.NumRepoOrg = stat[0].NumRepoOrg
  100. data.NumRepoNotOrg = stat[0].NumRepos - stat[0].NumRepoOrg
  101. data.NumRepoFork = stat[0].NumRepoFork
  102. data.NumRepoMirror = stat[0].NumRepoMirror
  103. data.NumRepoSelf = stat[0].NumRepoSelf
  104. data.NumRepoPrivate = stat[0].NumRepoPrivate
  105. data.NumRepoPublic = stat[0].NumRepoPublic
  106. data.CreatTime = stat[0].CreatedUnix.Format(UPDATE_TIME_FORMAT)
  107. if len(stat) == 2 {
  108. data.NumReposAdd = stat[0].NumRepos - stat[1].NumRepos
  109. data.NumRepoOrgAdd = stat[0].NumRepoOrg - stat[1].NumRepoOrg
  110. data.NumRepoNotOrgAdd = (stat[0].NumRepos - stat[0].NumRepoOrg) - (stat[1].NumRepos - stat[1].NumRepoOrg)
  111. data.NumRepoForkAdd = stat[0].NumRepoFork - stat[1].NumRepoFork
  112. data.NumRepoMirrorAdd = stat[0].NumRepoMirror - stat[1].NumRepoMirror
  113. data.NumRepoSelfAdd = stat[0].NumRepoSelf - stat[1].NumRepoSelf
  114. data.NumRepoPrivateAdd = stat[0].NumRepoPrivate - stat[1].NumRepoPrivate
  115. data.NumRepoPublicAdd = stat[0].NumRepoPublic - stat[1].NumRepoPublic
  116. }
  117. }
  118. ctx.JSON(200, data)
  119. }
  120. func GetProjectsSummaryData(ctx *context.Context) {
  121. var datas = make([]*ProjectSummaryBaseData, 0)
  122. recordBeginTime, err := getRecordBeginTime()
  123. if err != nil {
  124. log.Error("Can not get record begin time", err)
  125. ctx.Error(http.StatusBadRequest, ctx.Tr("repo.record_begintime_get_err"))
  126. return
  127. }
  128. beginTime, endTime, err := getTimePeroid(ctx, recordBeginTime)
  129. beginTime = beginTime.AddDate(0, 0, -1)
  130. queryType := ctx.QueryTrim("type")
  131. var count int64
  132. if queryType == "all" || queryType == "current_year" {
  133. dates := getEndOfMonthDates(beginTime, endTime)
  134. count, _ = models.GetSummaryStatisticByDateCount(dates)
  135. stats, err := models.GetAllSummaryStatisticByDates(dates)
  136. if err != nil {
  137. log.Warn("can not get summary data", err)
  138. } else {
  139. for i, v := range stats {
  140. if i == 0 {
  141. continue
  142. }
  143. data := ProjectSummaryBaseData{}
  144. setStatisticsData(&data, v, stats[i-1])
  145. createTime, _ := time.Parse(DATE_FORMAT, v.Date)
  146. data.CreatTime = createTime.Format(MONTH_FORMAT)
  147. datas = append(datas, &data)
  148. }
  149. }
  150. } else {
  151. count, _ = models.GetSummaryStatisticByTimeCount(beginTime, endTime)
  152. stats, err := models.GetAllSummaryStatisticByTime(beginTime, endTime)
  153. if err != nil {
  154. log.Warn("can not get summary data", err)
  155. } else {
  156. for i, v := range stats {
  157. if i == 0 {
  158. continue
  159. }
  160. data := ProjectSummaryBaseData{}
  161. setStatisticsData(&data, v, stats[i-1])
  162. data.CreatTime = v.Date
  163. datas = append(datas, &data)
  164. }
  165. }
  166. }
  167. projectSummaryPeriodData := ProjectSummaryPeriodData{
  168. TotalCount: count - 1,
  169. RecordBeginTime: recordBeginTime.Format(DATE_FORMAT),
  170. PageRecords: reverse(datas),
  171. }
  172. ctx.JSON(200, projectSummaryPeriodData)
  173. }
  174. func reverse(datas []*ProjectSummaryBaseData ) []*ProjectSummaryBaseData {
  175. for i := 0; i < len(datas)/2; i++ {
  176. j := len(datas) - i - 1
  177. datas[i], datas[j] = datas[j], datas[i]
  178. }
  179. return datas
  180. }
  181. func setStatisticsData(data *ProjectSummaryBaseData, v *models.SummaryStatistic, stats *models.SummaryStatistic) {
  182. data.NumReposAdd = v.NumRepos - stats.NumRepos
  183. data.NumRepoPublicAdd = v.NumRepoPublic - stats.NumRepoPublic
  184. data.NumRepoPrivateAdd = v.NumRepoPrivate - stats.NumRepoPrivate
  185. data.NumRepoMirrorAdd = v.NumRepoMirror - stats.NumRepoMirror
  186. data.NumRepoForkAdd = v.NumRepoFork - stats.NumRepoFork
  187. data.NumRepoSelfAdd = v.NumRepoSelf - stats.NumRepoSelf
  188. data.NumRepos = v.NumRepos
  189. }
  190. func getEndOfMonthDates(beginTime time.Time, endTime time.Time) []string {
  191. var dates = []string{}
  192. date := endOfMonth(beginTime.AddDate(0, -1, 0))
  193. dates = append(dates, date.Format(DATE_FORMAT))
  194. tempDate := endOfMonth(beginTime)
  195. for {
  196. if tempDate.Before(endTime) {
  197. dates = append(dates, tempDate.Format(DATE_FORMAT))
  198. tempDate = endOfMonth(tempDate.AddDate(0, 0, 1))
  199. } else {
  200. break
  201. }
  202. }
  203. return dates
  204. }
  205. func endOfMonth(date time.Time) time.Time {
  206. return date.AddDate(0, 1, -date.Day())
  207. }
  208. func GetAllProjectsPeriodStatistics(ctx *context.Context) {
  209. recordBeginTime, err := getRecordBeginTime()
  210. if err != nil {
  211. log.Error("Can not get record begin time", err)
  212. ctx.Error(http.StatusBadRequest, ctx.Tr("repo.record_begintime_get_err"))
  213. return
  214. }
  215. beginTime, endTime, err := getTimePeroid(ctx, recordBeginTime)
  216. if err != nil {
  217. log.Error("Parameter is wrong", err)
  218. ctx.Error(http.StatusBadRequest, ctx.Tr("repo.parameter_is_wrong"))
  219. return
  220. }
  221. q := ctx.QueryTrim("q")
  222. page := ctx.QueryInt("page")
  223. if page <= 0 {
  224. page = 1
  225. }
  226. pageSize := ctx.QueryInt("pagesize")
  227. if pageSize <= 0 {
  228. pageSize = DEFAULT_PAGE_SIZE
  229. }
  230. orderBy := getOrderBy(ctx)
  231. latestUpdatedTime, latestDate, err := models.GetRepoStatLastUpdatedTime()
  232. if err != nil {
  233. log.Error("Can not query the last updated time.", err)
  234. ctx.Error(http.StatusBadRequest, ctx.Tr("repo.last_update_time_error"))
  235. return
  236. }
  237. countSql := generateCountSql(beginTime, endTime, latestDate, q)
  238. total, err := models.CountRepoStatByRawSql(countSql)
  239. if err != nil {
  240. log.Error("Can not query total count.", err)
  241. ctx.Error(http.StatusBadRequest, ctx.Tr("repo.total_count_get_error"))
  242. return
  243. }
  244. sql := generateSqlByType(ctx, beginTime, endTime, latestDate, q, orderBy, page, pageSize)
  245. projectsPeriodData := ProjectsPeriodData{
  246. RecordBeginTime: recordBeginTime.Format(DATE_FORMAT),
  247. PageSize: pageSize,
  248. TotalPage: getTotalPage(total, pageSize),
  249. TotalCount: total,
  250. LastUpdatedTime: latestUpdatedTime,
  251. PageRecords: models.GetRepoStatisticByRawSql(sql),
  252. }
  253. ctx.JSON(http.StatusOK, projectsPeriodData)
  254. }
  255. func generateSqlByType(ctx *context.Context, beginTime time.Time, endTime time.Time, latestDate string, q string, orderBy string, page int, pageSize int) string {
  256. sql := ""
  257. if ctx.QueryTrim("type") == "all" {
  258. sql = generateTypeAllSql(beginTime, endTime, latestDate, q, orderBy, page, pageSize)
  259. } else {
  260. sql = generatePageSql(beginTime, endTime, latestDate, q, orderBy, page, pageSize)
  261. }
  262. return sql
  263. }
  264. func ServeAllProjectsPeriodStatisticsFile(ctx *context.Context) {
  265. recordBeginTime, err := getRecordBeginTime()
  266. if err != nil {
  267. log.Error("Can not get record begin time", err)
  268. ctx.Error(http.StatusBadRequest, ctx.Tr("repo.record_begintime_get_err"))
  269. return
  270. }
  271. beginTime, endTime, err := getTimePeroid(ctx, recordBeginTime)
  272. if err != nil {
  273. log.Error("Parameter is wrong", err)
  274. ctx.Error(http.StatusBadRequest, ctx.Tr("repo.parameter_is_wrong"))
  275. return
  276. }
  277. q := ctx.QueryTrim("q")
  278. page := ctx.QueryInt("page")
  279. if page <= 0 {
  280. page = 1
  281. }
  282. pageSize := 1000
  283. orderBy := getOrderBy(ctx)
  284. _, latestDate, err := models.GetRepoStatLastUpdatedTime()
  285. if err != nil {
  286. log.Error("Can not query the last updated time.", err)
  287. ctx.Error(http.StatusBadRequest, ctx.Tr("repo.last_update_time_error"))
  288. return
  289. }
  290. countSql := generateCountSql(beginTime, endTime, latestDate, q)
  291. total, err := models.CountRepoStatByRawSql(countSql)
  292. if err != nil {
  293. log.Error("Can not query total count.", err)
  294. ctx.Error(http.StatusBadRequest, ctx.Tr("repo.total_count_get_error"))
  295. return
  296. }
  297. var projectAnalysis = ctx.Tr("repo.repo_stat_inspect")
  298. fileName := getFileName(ctx, beginTime, endTime, projectAnalysis)
  299. totalPage := getTotalPage(total, pageSize)
  300. f := excelize.NewFile()
  301. index := f.NewSheet(projectAnalysis)
  302. f.DeleteSheet("Sheet1")
  303. for k, v := range allProjectsPeroidHeader(ctx) {
  304. f.SetCellValue(projectAnalysis, k, v)
  305. }
  306. var row = 2
  307. for i := 0; i <= totalPage; i++ {
  308. pageRecords := models.GetRepoStatisticByRawSql(generateSqlByType(ctx, beginTime, endTime, latestDate, q, orderBy, i+1, pageSize))
  309. for _, record := range pageRecords {
  310. for k, v := range allProjectsPeroidValues(row, record, ctx) {
  311. f.SetCellValue(projectAnalysis, k, v)
  312. }
  313. row++
  314. }
  315. }
  316. f.SetActiveSheet(index)
  317. ctx.Resp.Header().Set("Content-Disposition", "attachment; filename="+url.QueryEscape(fileName))
  318. ctx.Resp.Header().Set("Content-Type", "application/octet-stream")
  319. f.WriteTo(ctx.Resp)
  320. }
  321. func GetProjectsSummaryDataFile(ctx *context.Context) {
  322. recordBeginTime, err := getRecordBeginTime()
  323. if err != nil {
  324. log.Error("Can not get record begin time", err)
  325. ctx.Error(http.StatusBadRequest, ctx.Tr("repo.record_begintime_get_err"))
  326. return
  327. }
  328. beginTime, endTime, err := getTimePeroid(ctx, recordBeginTime)
  329. beginTime = beginTime.AddDate(0, 0, -1)
  330. if err != nil {
  331. log.Error("Parameter is wrong", err)
  332. ctx.Error(http.StatusBadRequest, ctx.Tr("repo.parameter_is_wrong"))
  333. return
  334. }
  335. page := ctx.QueryInt("page")
  336. if page <= 0 {
  337. page = 1
  338. }
  339. pageSize := 100
  340. if err != nil {
  341. log.Error("Can not query the last updated time.", err)
  342. ctx.Error(http.StatusBadRequest, ctx.Tr("repo.last_update_time_error"))
  343. return
  344. }
  345. var projectAnalysis = ctx.Tr("repo.repo_stat_develop")
  346. fileName := getSummaryFileName(ctx, beginTime, endTime, projectAnalysis)
  347. f := excelize.NewFile()
  348. index := f.NewSheet(projectAnalysis)
  349. f.DeleteSheet("Sheet1")
  350. for k, v := range allProjectsPeriodSummaryHeader(ctx) {
  351. f.SetCellValue(projectAnalysis, k, v)
  352. }
  353. var total int64
  354. queryType := ctx.QueryTrim("type")
  355. var datas = make([]*ProjectSummaryBaseData, 0)
  356. if queryType == "all" || queryType == "current_year" {
  357. dates := getEndOfMonthDates(beginTime, endTime)
  358. total, _ = models.GetSummaryStatisticByDateCount(dates)
  359. totalPage := getTotalPage(total, pageSize)
  360. for i := 0; i < totalPage; i++ {
  361. stats, err := models.GetSummaryStatisticByDates(dates, i+1, pageSize)
  362. if err != nil {
  363. log.Warn("can not get summary data", err)
  364. } else {
  365. for j, v := range stats {
  366. if j == 0 {
  367. continue
  368. }
  369. data := ProjectSummaryBaseData{}
  370. setStatisticsData(&data, v, stats[j-1])
  371. createTime, _ := time.Parse(DATE_FORMAT, v.Date)
  372. data.CreatTime = createTime.Format(MONTH_FORMAT)
  373. datas = append(datas, &data)
  374. }
  375. }
  376. }
  377. } else {
  378. total, _ = models.GetSummaryStatisticByTimeCount(beginTime, endTime)
  379. totalPage := getTotalPage(total, pageSize)
  380. for i := 0; i < totalPage; i++ {
  381. stats, err := models.GetSummaryStatisticByTime(beginTime, endTime, i+1, pageSize)
  382. if err != nil {
  383. log.Warn("can not get summary data", err)
  384. } else {
  385. for j, v := range stats {
  386. if j == 0 {
  387. continue
  388. }
  389. data := ProjectSummaryBaseData{}
  390. setStatisticsData(&data, v, stats[j-1])
  391. data.CreatTime = v.Date
  392. datas = append(datas, &data)
  393. }
  394. }
  395. }
  396. }
  397. row := 2
  398. datas = reverse(datas)
  399. for _, data := range datas {
  400. for k, v := range allProjectsPeriodSummaryValues(row, data, ctx) {
  401. f.SetCellValue(projectAnalysis, k, v)
  402. }
  403. row++
  404. }
  405. f.SetActiveSheet(index)
  406. ctx.Resp.Header().Set("Content-Disposition", "attachment; filename="+url.QueryEscape(fileName))
  407. ctx.Resp.Header().Set("Content-Type", "application/octet-stream")
  408. f.WriteTo(ctx.Resp)
  409. }
  410. func ServeAllProjectsOpenIStatisticsFile(ctx *context.Context) {
  411. page := ctx.QueryInt("page")
  412. if page <= 0 {
  413. page = 1
  414. }
  415. pageSize := 1000
  416. _, latestDate, err := models.GetRepoStatLastUpdatedTime()
  417. if err != nil {
  418. log.Error("Can not query the last updated time.", err)
  419. ctx.Error(http.StatusBadRequest, ctx.Tr("repo.last_update_time_error"))
  420. return
  421. }
  422. date := ctx.QueryTrim("date")
  423. if date == "" {
  424. date = latestDate
  425. }
  426. countSql := generateOpenICountSql(date)
  427. total, err := models.CountRepoStatByRawSql(countSql)
  428. if err != nil {
  429. log.Error("Can not query total count.", err)
  430. ctx.Error(http.StatusBadRequest, ctx.Tr("repo.total_count_get_error"))
  431. return
  432. }
  433. var projectAnalysis = ctx.Tr("repo.repo_stat_inspect")
  434. fileName := "项目分析_OPENI_" + date + ".xlsx"
  435. totalPage := getTotalPage(total, pageSize)
  436. f := excelize.NewFile()
  437. index := f.NewSheet(projectAnalysis)
  438. f.DeleteSheet("Sheet1")
  439. for k, v := range allProjectsOpenIHeader() {
  440. f.SetCellValue(projectAnalysis, k, v)
  441. }
  442. var row = 2
  443. for i := 0; i <= totalPage; i++ {
  444. pageRecords := models.GetRepoStatisticByRawSql(generateTypeAllOpenISql(date, i+1, pageSize))
  445. for _, record := range pageRecords {
  446. for k, v := range allProjectsOpenIValues(row, record, ctx) {
  447. f.SetCellValue(projectAnalysis, k, v)
  448. }
  449. row++
  450. }
  451. }
  452. f.SetActiveSheet(index)
  453. ctx.Resp.Header().Set("Content-Disposition", "attachment; filename="+url.QueryEscape(fileName))
  454. ctx.Resp.Header().Set("Content-Type", "application/octet-stream")
  455. f.WriteTo(ctx.Resp)
  456. }
  457. func getFileName(ctx *context.Context, beginTime time.Time, endTime time.Time, projectAnalysis string) string {
  458. baseName := projectAnalysis + "_"
  459. if ctx.QueryTrim("q") != "" {
  460. baseName = baseName + ctx.QueryTrim("q") + "_"
  461. }
  462. if ctx.QueryTrim("type") == "all" {
  463. baseName = baseName + ctx.Tr("repo.all")
  464. } else {
  465. baseName = baseName + beginTime.AddDate(0, 0, -1).Format(EXCEL_DATE_FORMAT) + "_" + endTime.AddDate(0, 0, -1).Format(EXCEL_DATE_FORMAT)
  466. }
  467. frontName := baseName + ".xlsx"
  468. return frontName
  469. }
  470. func getSummaryFileName(ctx *context.Context, beginTime time.Time, endTime time.Time, projectAnalysis string) string {
  471. baseName := projectAnalysis + "_"
  472. if ctx.QueryTrim("type") == "all" {
  473. baseName = baseName + ctx.Tr("repo.all")
  474. } else if ctx.QueryTrim("type") == "current_year" {
  475. baseName = baseName + ctx.Tr("repo.current_year")
  476. } else {
  477. baseName = baseName + beginTime.Format(EXCEL_DATE_FORMAT) + "_" + endTime.AddDate(0, 0, -1).Format(EXCEL_DATE_FORMAT)
  478. }
  479. frontName := baseName + ".xlsx"
  480. return frontName
  481. }
  482. func allProjectsPeroidHeader(ctx *context.Context) map[string]string {
  483. return map[string]string{"A1": ctx.Tr("admin.repos.id"), "B1": ctx.Tr("admin.repos.projectName"), "C1": ctx.Tr("repo.owner"), "D1": ctx.Tr("admin.repos.isPrivate"), "E1": ctx.Tr("admin.repos.openi"), "F1": ctx.Tr("admin.repos.visit"), "G1": ctx.Tr("admin.repos.download"), "H1": ctx.Tr("admin.repos.pr"), "I1": ctx.Tr("admin.repos.commit"),
  484. "J1": ctx.Tr("admin.repos.watches"), "K1": ctx.Tr("admin.repos.stars"), "L1": ctx.Tr("admin.repos.forks"), "M1": ctx.Tr("admin.repos.issues"), "N1": ctx.Tr("admin.repos.closedIssues"), "O1": ctx.Tr("admin.repos.contributor"), "P1": ctx.Tr("admin.repos.isFork"), "Q1": ctx.Tr("admin.repos.isMirror"), "R1": ctx.Tr("admin.repos.create")}
  485. }
  486. func allProjectsPeriodSummaryHeader(ctx *context.Context) map[string]string {
  487. return map[string]string{"A1": ctx.Tr("repo.date"), "B1": ctx.Tr("repo.repo_add"), "C1": ctx.Tr("repo.repo_total"), "D1": ctx.Tr("repo.repo_public_add"), "E1": ctx.Tr("repo.repo_private_add"), "F1": ctx.Tr("repo.repo_self_add"), "G1": ctx.Tr("repo.repo_fork_add"), "H1": ctx.Tr("repo.repo_mirror_add")}
  488. }
  489. func allProjectsPeriodSummaryValues(row int, rs *ProjectSummaryBaseData, ctx *context.Context) map[string]string {
  490. return map[string]string{getCellName("A", row): rs.CreatTime, getCellName("B", row): strconv.FormatInt(rs.NumReposAdd, 10), getCellName("C", row): strconv.FormatInt(rs.NumRepos, 10), getCellName("D", row): strconv.FormatInt(rs.NumRepoPublicAdd, 10), getCellName("E", row): strconv.FormatInt(rs.NumRepoPrivateAdd, 10),
  491. getCellName("F", row): strconv.FormatInt(rs.NumRepoSelfAdd, 10), getCellName("G", row): strconv.FormatInt(rs.NumRepoForkAdd, 10), getCellName("H", row): strconv.FormatInt(rs.NumRepoMirrorAdd, 10),
  492. }
  493. }
  494. func allProjectsPeroidValues(row int, rs *models.RepoStatistic, ctx *context.Context) map[string]string {
  495. return map[string]string{getCellName("A", row): strconv.FormatInt(rs.RepoID, 10), getCellName("B", row): rs.DisplayName(), getCellName("C", row): rs.OwnerName, getCellName("D", row): getBoolDisplay(rs.IsPrivate, ctx), getCellName("E", row): strconv.FormatFloat(rs.RadarTotal, 'f', 2, 64),
  496. getCellName("F", row): strconv.FormatInt(rs.NumVisits, 10), getCellName("G", row): strconv.FormatInt(rs.NumDownloads, 10), getCellName("H", row): strconv.FormatInt(rs.NumPulls, 10), getCellName("I", row): strconv.FormatInt(rs.NumCommits, 10),
  497. getCellName("J", row): strconv.FormatInt(rs.NumWatches, 10), getCellName("K", row): strconv.FormatInt(rs.NumStars, 10), getCellName("L", row): strconv.FormatInt(rs.NumForks, 10), getCellName("M", row): strconv.FormatInt(rs.NumIssues, 10),
  498. getCellName("N", row): strconv.FormatInt(rs.NumClosedIssues, 10), getCellName("O", row): strconv.FormatInt(rs.NumContributor, 10), getCellName("P", row): getBoolDisplay(rs.IsFork, ctx), getCellName("Q", row): getBoolDisplay(rs.IsMirror, ctx), getCellName("R", row): time.Unix(int64(rs.RepoCreatedUnix), 0).Format(CREATE_TIME_FORMAT),
  499. }
  500. }
  501. func allProjectsOpenIHeader() map[string]string {
  502. return map[string]string{"A1": "ID", "B1": "项目名称", "C1": "拥有者", "D1": "私有", "E1": "OpenI指数",
  503. "F1": "影响力", "G1": "成熟度", "H1": "活跃度", "I1": "项目健康度", "J1": "团队健康度", "K1": "项目发展趋势",
  504. "L1": "关注数", "M1": "点赞数", "N1": "派生数", "O1": "代码下载量", "P1": "评论数", "Q1": "浏览量", "R1": "已解决任务数", "S1": "版本发布数量", "T1": "有效开发年龄",
  505. "U1": "数据集", "V1": "模型数", "W1": "百科页面数量", "X1": "提交数", "Y1": "任务数", "Z1": "PR数", "AA1": "版本发布数量", "AB1": "任务完成比例", "AC1": "贡献者数", "AD1": "关键贡献者数",
  506. "AE1": "新人增长量", "AF1": "代码规模增长量", "AG1": "任务增长量", "AH1": "新人增长量", "AI1": "提交增长量", "AJ1": "评论增长量", "AK1": "迁移", "AL1": "镜像", "AM1": "项目创建时间",
  507. }
  508. }
  509. func allProjectsOpenIValues(row int, rs *models.RepoStatistic, ctx *context.Context) map[string]string {
  510. return map[string]string{getCellName("A", row): strconv.FormatInt(rs.RepoID, 10), getCellName("B", row): rs.DisplayName(), getCellName("C", row): rs.OwnerName, getCellName("D", row): getBoolDisplay(rs.IsPrivate, ctx), getCellName("E", row): strconv.FormatFloat(rs.RadarTotal, 'f', 2, 64),
  511. getCellName("F", row): strconv.FormatFloat(rs.Impact, 'f', 2, 64), getCellName("G", row): strconv.FormatFloat(rs.Completeness, 'f', 2, 64), getCellName("H", row): strconv.FormatFloat(rs.Liveness, 'f', 2, 64), getCellName("I", row): strconv.FormatFloat(rs.ProjectHealth, 'f', 2, 64), getCellName("J", row): strconv.FormatFloat(rs.TeamHealth, 'f', 2, 64), getCellName("K", row): strconv.FormatFloat(rs.Growth, 'f', 2, 64),
  512. getCellName("L", row): strconv.FormatInt(rs.NumWatches, 10), getCellName("M", row): strconv.FormatInt(rs.NumStars, 10), getCellName("N", row): strconv.FormatInt(rs.NumForks, 10), getCellName("O", row): strconv.FormatInt(rs.NumDownloads, 10),
  513. getCellName("P", row): strconv.FormatInt(rs.NumComments, 10), getCellName("Q", row): strconv.FormatInt(rs.NumVisits, 10), getCellName("R", row): strconv.FormatInt(rs.NumClosedIssues, 10), getCellName("S", row): strconv.FormatInt(rs.NumVersions, 10),
  514. getCellName("T", row): strconv.FormatInt(rs.NumDevMonths, 10), getCellName("U", row): strconv.FormatInt(rs.DatasetSize, 10), getCellName("V", row): strconv.FormatInt(rs.NumModels, 10), getCellName("W", row): strconv.FormatInt(rs.NumWikiViews, 10),
  515. getCellName("X", row): strconv.FormatInt(rs.NumCommits, 10), getCellName("Y", row): strconv.FormatInt(rs.NumIssues, 10), getCellName("Z", row): strconv.FormatInt(rs.NumPulls, 10), getCellName("AA", row): strconv.FormatInt(rs.NumVersions, 10),
  516. getCellName("AB", row): strconv.FormatFloat(float64(rs.IssueFixedRate), 'f', 2, 64), getCellName("AC", row): strconv.FormatInt(rs.NumContributor, 10), getCellName("AD", row): strconv.FormatInt(rs.NumKeyContributor, 10), getCellName("AE", row): strconv.FormatInt(rs.NumContributorsGrowth, 10),
  517. getCellName("AF", row): strconv.FormatInt(rs.NumCommitLinesGrowth, 10), getCellName("AG", row): strconv.FormatInt(rs.NumIssuesGrowth, 10), getCellName("AH", row): strconv.FormatInt(rs.NumContributorsGrowth, 10), getCellName("AI", row): strconv.FormatInt(rs.NumCommitsGrowth, 10), getCellName("AJ", row): strconv.FormatInt(rs.NumCommentsGrowth, 10), getCellName("AK", row): getBoolDisplay(rs.IsFork, ctx), getCellName("AL", row): getBoolDisplay(rs.IsMirror, ctx), getCellName("AM", row): time.Unix(int64(rs.RepoCreatedUnix), 0).Format(CREATE_TIME_FORMAT),
  518. }
  519. }
  520. func getCellName(col string, row int) string {
  521. return col + strconv.Itoa(row)
  522. }
  523. func getBoolDisplay(value bool, ctx *context.Context) string {
  524. if value {
  525. return ctx.Tr("admin.repos.yes")
  526. } else {
  527. return ctx.Tr("admin.repos.no")
  528. }
  529. }
  530. func GetProjectLatestStatistics(ctx *context.Context) {
  531. repoId := ctx.Params(":id")
  532. recordBeginTime, err := getRecordBeginTime()
  533. if err != nil {
  534. log.Error("Can not get record begin time", err)
  535. ctx.Error(http.StatusBadRequest, ctx.Tr("repo.record_begintime_get_err"))
  536. return
  537. }
  538. latestUpdatedTime, latestDate, err := models.GetRepoStatLastUpdatedTime(repoId)
  539. repoIdInt, _ := strconv.ParseInt(repoId, 10, 64)
  540. repoStat, err := models.GetRepoStatisticByDateAndRepoId(latestDate, repoIdInt)
  541. if err != nil {
  542. log.Error("Can not get the repo statistics "+repoId, err)
  543. ctx.Error(http.StatusBadRequest, ctx.Tr("repo.get_repo_stat_error"))
  544. return
  545. }
  546. repository, err := models.GetRepositoryByID(repoIdInt)
  547. if err != nil {
  548. log.Error("Can not get the repo info "+repoId, err)
  549. ctx.Error(http.StatusBadRequest, ctx.Tr("repo.get_repo_info_error"))
  550. return
  551. }
  552. projectLatestData := ProjectLatestData{
  553. RecordBeginTime: recordBeginTime.Format(DATE_FORMAT),
  554. CreatTime: time.Unix(int64(repository.CreatedUnix), 0).Format(DATE_FORMAT),
  555. LastUpdatedTime: latestUpdatedTime,
  556. OpenI: repoStat.RadarTotal,
  557. Comment: repoStat.NumComments,
  558. View: repoStat.NumVisits,
  559. Download: repoStat.NumDownloads,
  560. IssueClosedRatio: repoStat.IssueFixedRate,
  561. Impact: repoStat.Impact,
  562. Completeness: repoStat.Completeness,
  563. Liveness: repoStat.Liveness,
  564. ProjectHealth: repoStat.ProjectHealth,
  565. TeamHealth: repoStat.TeamHealth,
  566. Growth: repoStat.Growth,
  567. Description: repository.Description,
  568. }
  569. contributors, err := models.GetTop10Contributor(repository.RepoPath())
  570. if err != nil {
  571. log.Error("can not get contributors", err)
  572. }
  573. users := make([]UserInfo, 0)
  574. for _, contributor := range contributors {
  575. mode := repository.GetCollaboratorMode(contributor.UserId)
  576. if mode == -1 {
  577. if contributor.IsAdmin {
  578. mode = int(models.AccessModeAdmin)
  579. }
  580. if contributor.UserId == repository.OwnerID {
  581. mode = int(models.AccessModeOwner)
  582. }
  583. }
  584. pr := models.GetPullCountByUserAndRepoId(repoIdInt, contributor.UserId)
  585. userInfo := UserInfo{
  586. User: contributor.Committer,
  587. Commit: contributor.CommitCnt,
  588. Mode: mode,
  589. PR: pr,
  590. RelAvatarLink: contributor.RelAvatarLink,
  591. Email: contributor.Email,
  592. }
  593. users = append(users, userInfo)
  594. }
  595. projectLatestData.Top10 = users
  596. ctx.JSON(http.StatusOK, projectLatestData)
  597. }
  598. func GetProjectPeriodStatistics(ctx *context.Context) {
  599. repoId := ctx.Params(":id")
  600. recordBeginTime, err := getRecordBeginTime()
  601. if err != nil {
  602. log.Error("Can not get record begin time", err)
  603. ctx.Error(http.StatusBadRequest, ctx.Tr("repo.record_begintime_get_err"))
  604. return
  605. }
  606. repoIdInt, _ := strconv.ParseInt(repoId, 10, 64)
  607. if err != nil {
  608. log.Error("Can not get record begin time", err)
  609. ctx.Error(http.StatusBadRequest, ctx.Tr("repo.record_begintime_get_err"))
  610. return
  611. }
  612. beginTime, endTime, err := getTimePeroid(ctx, recordBeginTime)
  613. isOpenI := ctx.QueryBool("openi")
  614. var repositorys []*models.RepoStatistic
  615. if isOpenI {
  616. repositorys = models.GetRepoStatisticByRawSql(generateRadarSql(beginTime, endTime, repoIdInt))
  617. } else {
  618. repositorys = models.GetRepoStatisticByRawSql(generateTargetSql(beginTime, endTime, repoIdInt))
  619. }
  620. ctx.JSON(http.StatusOK, repositorys)
  621. }
  622. func generateRadarSql(beginTime time.Time, endTime time.Time, repoId int64) string {
  623. sql := "SELECT date, impact, completeness, liveness, project_health, team_health, growth, radar_total FROM repo_statistic" +
  624. " where repo_id=" + strconv.FormatInt(repoId, 10) + " and created_unix >=" + strconv.FormatInt(beginTime.Unix(), 10) +
  625. " and created_unix<" + strconv.FormatInt(endTime.Unix(), 10) + " order by created_unix"
  626. return sql
  627. }
  628. func generateTargetSql(beginTime time.Time, endTime time.Time, repoId int64) string {
  629. sql := "SELECT date, num_visits,num_downloads_added as num_downloads,num_commits_added as num_commits FROM repo_statistic" +
  630. " where repo_id=" + strconv.FormatInt(repoId, 10) + " and created_unix >=" + strconv.FormatInt(beginTime.Unix(), 10) +
  631. " and created_unix<" + strconv.FormatInt(endTime.Unix(), 10) + " order by created_unix desc"
  632. return sql
  633. }
  634. func generateCountSql(beginTime time.Time, endTime time.Time, latestDate string, q string) string {
  635. countSql := "SELECT count(*) FROM " +
  636. "(SELECT repo_id FROM repo_statistic where created_unix >=" + strconv.FormatInt(beginTime.Unix(), 10) +
  637. " and created_unix<" + strconv.FormatInt(endTime.Unix(), 10) + " group by repo_id) A," +
  638. "(SELECT repo_id,name,alias,is_private,radar_total from public.repo_statistic where date='" + latestDate + "') B" +
  639. " where A.repo_id=B.repo_id"
  640. if q != "" {
  641. countSql = countSql + " and LOWER(B.alias) like '%" + strings.ToLower(q) + "%'"
  642. }
  643. return countSql
  644. }
  645. func generateOpenICountSql(latestDate string) string {
  646. countSql := "SELECT count(*) FROM " +
  647. "public.repo_statistic where date='" + latestDate + "'"
  648. return countSql
  649. }
  650. func generateTypeAllSql(beginTime time.Time, endTime time.Time, latestDate string, q string, orderBy string, page int, pageSize int) string {
  651. sql := "SELECT A.repo_id,name,alias,owner_name,is_private,is_mirror,is_fork,repo_created_unix,radar_total,num_watches,num_visits,num_downloads,num_pulls,num_commits,num_stars,num_forks,num_issues,num_closed_issues,num_contributor FROM " +
  652. "(SELECT repo_id,sum(num_visits) as num_visits " +
  653. " FROM repo_statistic where created_unix >=" + strconv.FormatInt(beginTime.Unix(), 10) +
  654. " and created_unix<" + strconv.FormatInt(endTime.Unix(), 10) + " group by repo_id) A," +
  655. "(SELECT repo_id,name,alias,owner_name,is_private,is_mirror,is_fork,repo_created_unix,radar_total,num_watches,num_downloads,num_pulls,num_commits,num_stars,num_forks,num_issues,num_closed_issues,num_contributor from public.repo_statistic where date='" + latestDate + "') B" +
  656. " where A.repo_id=B.repo_id"
  657. if q != "" {
  658. sql = sql + " and LOWER(alias) like '%" + strings.ToLower(q) + "%'"
  659. }
  660. sql = sql + " order by " + orderBy + " desc,repo_id" + " limit " + strconv.Itoa(pageSize) + " offset " + strconv.Itoa((page-1)*pageSize)
  661. return sql
  662. }
  663. func generateTypeAllOpenISql(latestDate string, page int, pageSize int) string {
  664. sql := "SELECT id, repo_id, date, num_watches, num_stars, num_forks, num_downloads, num_comments, num_visits, num_closed_issues, num_versions, num_dev_months, repo_size, dataset_size, num_models, num_wiki_views, num_commits, num_issues, num_pulls, issue_fixed_rate, num_contributor, num_key_contributor, num_contributors_growth, num_commits_growth, num_commit_lines_growth, num_issues_growth, num_comments_growth, impact, completeness, liveness, project_health, team_health, growth, radar_total, name,alias, is_private,is_mirror,is_fork,repo_created_unix, owner_name FROM " +
  665. " public.repo_statistic where date='" + latestDate + "'"
  666. sql = sql + " order by radar_total desc,repo_id" + " limit " + strconv.Itoa(pageSize) + " offset " + strconv.Itoa((page-1)*pageSize)
  667. return sql
  668. }
  669. func generatePageSql(beginTime time.Time, endTime time.Time, latestDate string, q string, orderBy string, page int, pageSize int) string {
  670. sql := "SELECT A.repo_id,name,alias,owner_name,is_private,is_mirror,is_fork,repo_created_unix,radar_total,num_watches,num_visits,num_downloads,num_pulls,num_commits,num_stars,num_forks,num_issues,num_closed_issues,num_contributor FROM " +
  671. "(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 " +
  672. " FROM repo_statistic where created_unix >=" + strconv.FormatInt(beginTime.Unix(), 10) +
  673. " and created_unix<" + strconv.FormatInt(endTime.Unix(), 10) + " group by repo_id) A," +
  674. "(SELECT repo_id,name,alias,owner_name,is_private,is_mirror,is_fork,repo_created_unix,radar_total from public.repo_statistic where date='" + latestDate + "') B" +
  675. " where A.repo_id=B.repo_id"
  676. if q != "" {
  677. sql = sql + " and LOWER(B.alias) like '%" + strings.ToLower(q) + "%'"
  678. }
  679. sql = sql + " order by " + orderBy + " desc,A.repo_id" + " limit " + strconv.Itoa(pageSize) + " offset " + strconv.Itoa((page-1)*pageSize)
  680. return sql
  681. }
  682. func getOrderBy(ctx *context.Context) string {
  683. orderBy := ""
  684. switch ctx.Query("sort") {
  685. case "openi":
  686. orderBy = "B.radar_total"
  687. case "view":
  688. orderBy = "A.num_visits"
  689. case "download":
  690. orderBy = "A.num_downloads"
  691. case "pr":
  692. orderBy = "A.num_pulls"
  693. case "commit":
  694. orderBy = "A.num_commits"
  695. case "watch":
  696. orderBy = "A.num_watches"
  697. case "star":
  698. orderBy = "A.num_stars"
  699. case "fork":
  700. orderBy = "A.num_forks"
  701. case "issue":
  702. orderBy = "A.num_issues"
  703. case "issue_closed":
  704. orderBy = "A.num_closed_issues"
  705. case "contributor":
  706. orderBy = "A.num_contributor"
  707. default:
  708. orderBy = "B.radar_total"
  709. }
  710. return orderBy
  711. }
  712. func getTimePeroid(ctx *context.Context, recordBeginTime time.Time) (time.Time, time.Time, error) {
  713. queryType := ctx.QueryTrim("type")
  714. now := time.Now()
  715. recordBeginTimeTemp := recordBeginTime.AddDate(0, 0, 1)
  716. beginTimeStr := ctx.QueryTrim("beginTime")
  717. endTimeStr := ctx.QueryTrim("endTime")
  718. var beginTime time.Time
  719. var endTime time.Time
  720. var err error
  721. if queryType != "" {
  722. if queryType == "all" {
  723. beginTime = recordBeginTimeTemp
  724. endTime = now
  725. } else if queryType == "yesterday" {
  726. endTime = now
  727. beginTime = time.Date(endTime.Year(), endTime.Month(), endTime.Day(), 0, 0, 0, 0, now.Location())
  728. } else if queryType == "current_week" {
  729. beginTime = now.AddDate(0, 0, -int(time.Now().Weekday())+2) //begin from monday
  730. beginTime = time.Date(beginTime.Year(), beginTime.Month(), beginTime.Day(), 0, 0, 0, 0, now.Location())
  731. endTime = now
  732. } else if queryType == "current_month" {
  733. endTime = now
  734. beginTime = time.Date(endTime.Year(), endTime.Month(), 2, 0, 0, 0, 0, now.Location())
  735. } else if queryType == "monthly" {
  736. endTime = now
  737. beginTime = now.AddDate(0, -1, 1)
  738. beginTime = time.Date(beginTime.Year(), beginTime.Month(), beginTime.Day(), 0, 0, 0, 0, now.Location())
  739. } else if queryType == "current_year" {
  740. endTime = now
  741. beginTime = time.Date(endTime.Year(), 1, 2, 0, 0, 0, 0, now.Location())
  742. } else if queryType == "last_month" {
  743. lastMonthTime := now.AddDate(0, -1, 0)
  744. beginTime = time.Date(lastMonthTime.Year(), lastMonthTime.Month(), 2, 0, 0, 0, 0, now.Location())
  745. endTime = time.Date(now.Year(), now.Month(), 2, 0, 0, 0, 0, now.Location())
  746. } else {
  747. return now, now, fmt.Errorf("The value of type parameter is wrong.")
  748. }
  749. } else {
  750. if beginTimeStr == "" || endTimeStr == "" {
  751. //如果查询类型和开始时间结束时间都未设置,按queryType=all处理
  752. beginTime = recordBeginTimeTemp
  753. endTime = now
  754. } else {
  755. beginTime, err = time.ParseInLocation("2006-01-02", beginTimeStr, time.Local)
  756. if err != nil {
  757. return now, now, err
  758. }
  759. endTime, err = time.ParseInLocation("2006-01-02", endTimeStr, time.Local)
  760. if err != nil {
  761. return now, now, err
  762. }
  763. beginTime = beginTime.AddDate(0, 0, 1)
  764. endTime = endTime.AddDate(0, 0, 1)
  765. }
  766. }
  767. if beginTime.Before(recordBeginTimeTemp) {
  768. beginTime = recordBeginTimeTemp
  769. }
  770. return beginTime, endTime, nil
  771. }
  772. func getRecordBeginTime() (time.Time, error) {
  773. return time.ParseInLocation(DATE_FORMAT, setting.RadarMap.RecordBeginTime, time.Local)
  774. }
  775. func getTotalPage(total int64, pageSize int) int {
  776. another := 0
  777. if int(total)%pageSize != 0 {
  778. another = 1
  779. }
  780. return int(total)/pageSize + another
  781. }
  782. func ProjectNumVisit(ctx *context.APIContext) {
  783. var (
  784. err error
  785. )
  786. var userName = ctx.Query("user")
  787. var projectName = ctx.Query("project")
  788. var beginTime = ctx.Query("begintime")
  789. var endTime = ctx.Query("endtime")
  790. var ProjectNumVisits int
  791. ProjectNumVisits, err = repository.AppointProjectView(userName, projectName, beginTime, endTime) //访问量
  792. if err != nil {
  793. ctx.NotFound(err)
  794. }
  795. log.Info("ProjectNumVisits is:", ProjectNumVisits)
  796. ctx.JSON(http.StatusOK, map[string]interface{}{
  797. "ProjectNumVisits": ProjectNumVisits,
  798. "StatusOK": 0,
  799. })
  800. }