Платформа ЦРНП "Мирокод" для разработки проектов
https://git.mirocod.ru
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
215 lines
5.5 KiB
215 lines
5.5 KiB
// Copyright 2020 The Gitea Authors. All rights reserved. |
|
// Use of this source code is governed by a MIT-style |
|
// license that can be found in the LICENSE file. |
|
|
|
package repo |
|
|
|
import ( |
|
"math" |
|
"strings" |
|
|
|
"code.gitea.io/gitea/models/db" |
|
"code.gitea.io/gitea/modules/timeutil" |
|
|
|
"github.com/go-enry/go-enry/v2" |
|
) |
|
|
|
// LanguageStat describes language statistics of a repository |
|
type LanguageStat struct { |
|
ID int64 `xorm:"pk autoincr"` |
|
RepoID int64 `xorm:"UNIQUE(s) INDEX NOT NULL"` |
|
CommitID string |
|
IsPrimary bool |
|
Language string `xorm:"VARCHAR(50) UNIQUE(s) INDEX NOT NULL"` |
|
Percentage float32 `xorm:"-"` |
|
Size int64 `xorm:"NOT NULL DEFAULT 0"` |
|
Color string `xorm:"-"` |
|
CreatedUnix timeutil.TimeStamp `xorm:"INDEX CREATED"` |
|
} |
|
|
|
func init() { |
|
db.RegisterModel(new(LanguageStat)) |
|
} |
|
|
|
// LanguageStatList defines a list of language statistics |
|
type LanguageStatList []*LanguageStat |
|
|
|
// LoadAttributes loads attributes |
|
func (stats LanguageStatList) LoadAttributes() { |
|
for i := range stats { |
|
stats[i].Color = enry.GetColor(stats[i].Language) |
|
} |
|
} |
|
|
|
func (stats LanguageStatList) getLanguagePercentages() map[string]float32 { |
|
langPerc := make(map[string]float32) |
|
var otherPerc float32 = 100 |
|
var total int64 |
|
|
|
for _, stat := range stats { |
|
total += stat.Size |
|
} |
|
if total > 0 { |
|
for _, stat := range stats { |
|
perc := float32(math.Round(float64(stat.Size)/float64(total)*1000) / 10) |
|
if perc <= 0.1 { |
|
continue |
|
} |
|
otherPerc -= perc |
|
langPerc[stat.Language] = perc |
|
} |
|
otherPerc = float32(math.Round(float64(otherPerc)*10) / 10) |
|
} |
|
if otherPerc > 0 { |
|
langPerc["other"] = otherPerc |
|
} |
|
return langPerc |
|
} |
|
|
|
func getLanguageStats(e db.Engine, repo *Repository) (LanguageStatList, error) { |
|
stats := make(LanguageStatList, 0, 6) |
|
if err := e.Where("`repo_id` = ?", repo.ID).Desc("`size`").Find(&stats); err != nil { |
|
return nil, err |
|
} |
|
return stats, nil |
|
} |
|
|
|
// GetLanguageStats returns the language statistics for a repository |
|
func GetLanguageStats(repo *Repository) (LanguageStatList, error) { |
|
return getLanguageStats(db.GetEngine(db.DefaultContext), repo) |
|
} |
|
|
|
// GetTopLanguageStats returns the top language statistics for a repository |
|
func GetTopLanguageStats(repo *Repository, limit int) (LanguageStatList, error) { |
|
stats, err := getLanguageStats(db.GetEngine(db.DefaultContext), repo) |
|
if err != nil { |
|
return nil, err |
|
} |
|
perc := stats.getLanguagePercentages() |
|
topstats := make(LanguageStatList, 0, limit) |
|
var other float32 |
|
for i := range stats { |
|
if _, ok := perc[stats[i].Language]; !ok { |
|
continue |
|
} |
|
if stats[i].Language == "other" || len(topstats) >= limit { |
|
other += perc[stats[i].Language] |
|
continue |
|
} |
|
stats[i].Percentage = perc[stats[i].Language] |
|
topstats = append(topstats, stats[i]) |
|
} |
|
if other > 0 { |
|
topstats = append(topstats, &LanguageStat{ |
|
RepoID: repo.ID, |
|
Language: "other", |
|
Color: "#cccccc", |
|
Percentage: float32(math.Round(float64(other)*10) / 10), |
|
}) |
|
} |
|
topstats.LoadAttributes() |
|
return topstats, nil |
|
} |
|
|
|
// UpdateLanguageStats updates the language statistics for repository |
|
func UpdateLanguageStats(repo *Repository, commitID string, stats map[string]int64) error { |
|
ctx, committer, err := db.TxContext() |
|
if err != nil { |
|
return err |
|
} |
|
defer committer.Close() |
|
sess := db.GetEngine(ctx) |
|
|
|
oldstats, err := getLanguageStats(sess, repo) |
|
if err != nil { |
|
return err |
|
} |
|
var topLang string |
|
var s int64 |
|
for lang, size := range stats { |
|
if size > s { |
|
s = size |
|
topLang = strings.ToLower(lang) |
|
} |
|
} |
|
|
|
for lang, size := range stats { |
|
upd := false |
|
llang := strings.ToLower(lang) |
|
for _, s := range oldstats { |
|
// Update already existing language |
|
if strings.ToLower(s.Language) == llang { |
|
s.CommitID = commitID |
|
s.IsPrimary = llang == topLang |
|
s.Size = size |
|
if _, err := sess.ID(s.ID).Cols("`commit_id`", "`size`", "`is_primary`").Update(s); err != nil { |
|
return err |
|
} |
|
upd = true |
|
break |
|
} |
|
} |
|
// Insert new language |
|
if !upd { |
|
if _, err := sess.Insert(&LanguageStat{ |
|
RepoID: repo.ID, |
|
CommitID: commitID, |
|
IsPrimary: llang == topLang, |
|
Language: lang, |
|
Size: size, |
|
}); err != nil { |
|
return err |
|
} |
|
} |
|
} |
|
// Delete old languages |
|
statsToDelete := make([]int64, 0, len(oldstats)) |
|
for _, s := range oldstats { |
|
if s.CommitID != commitID { |
|
statsToDelete = append(statsToDelete, s.ID) |
|
} |
|
} |
|
if len(statsToDelete) > 0 { |
|
if _, err := sess.In("`id`", statsToDelete).Delete(&LanguageStat{}); err != nil { |
|
return err |
|
} |
|
} |
|
|
|
// Update indexer status |
|
if err = updateIndexerStatus(sess, repo, RepoIndexerTypeStats, commitID); err != nil { |
|
return err |
|
} |
|
|
|
return committer.Commit() |
|
} |
|
|
|
// CopyLanguageStat Copy originalRepo language stat information to destRepo (use for forked repo) |
|
func CopyLanguageStat(originalRepo, destRepo *Repository) error { |
|
ctx, committer, err := db.TxContext() |
|
if err != nil { |
|
return err |
|
} |
|
defer committer.Close() |
|
sess := db.GetEngine(ctx) |
|
|
|
RepoLang := make(LanguageStatList, 0, 6) |
|
if err := sess.Where("`repo_id` = ?", originalRepo.ID).Desc("`size`").Find(&RepoLang); err != nil { |
|
return err |
|
} |
|
if len(RepoLang) > 0 { |
|
for i := range RepoLang { |
|
RepoLang[i].ID = 0 |
|
RepoLang[i].RepoID = destRepo.ID |
|
RepoLang[i].CreatedUnix = timeutil.TimeStampNow() |
|
} |
|
// update destRepo's indexer status |
|
tmpCommitID := RepoLang[0].CommitID |
|
if err := updateIndexerStatus(sess, destRepo, RepoIndexerTypeStats, tmpCommitID); err != nil { |
|
return err |
|
} |
|
if _, err := sess.Insert(&RepoLang); err != nil { |
|
return err |
|
} |
|
} |
|
return committer.Commit() |
|
}
|
|
|