8
8
"context"
9
9
"fmt"
10
10
"os"
11
- "path"
12
11
"path/filepath"
13
12
"sort"
14
13
"strings"
@@ -27,6 +26,11 @@ import (
27
26
asymkey_service "code.gitea.io/gitea/services/asymkey"
28
27
)
29
28
29
+ type OptionFile struct {
30
+ DisplayName string
31
+ Description string
32
+ }
33
+
30
34
var (
31
35
// Gitignores contains the gitiginore files
32
36
Gitignores []string
@@ -37,65 +41,73 @@ var (
37
41
// Readmes contains the readme files
38
42
Readmes []string
39
43
40
- // LabelTemplates contains the label template files and the list of labels for each file
41
- LabelTemplates map [string ]string
44
+ // LabelTemplateFiles contains the label template files, each item has its DisplayName and Description
45
+ LabelTemplateFiles []OptionFile
46
+ labelTemplateFileMap = map [string ]string {} // DisplayName => FileName mapping
42
47
)
43
48
49
+ type optionFileList struct {
50
+ all []string // all files provided by bindata & custom-path. Sorted.
51
+ custom []string // custom files provided by custom-path. Non-sorted, internal use only.
52
+ }
53
+
54
+ // mergeCustomLabelFiles merges the custom label files. Always use the file's main name (DisplayName) as the key to de-duplicate.
55
+ func mergeCustomLabelFiles (fl optionFileList ) []string {
56
+ exts := map [string ]int {"" : 0 , ".yml" : 1 , ".yaml" : 2 } // "yaml" file has the highest priority to be used.
57
+
58
+ m := map [string ]string {}
59
+ merge := func (list []string ) {
60
+ sort .Slice (list , func (i , j int ) bool { return exts [filepath .Ext (list [i ])] < exts [filepath .Ext (list [j ])] })
61
+ for _ , f := range list {
62
+ m [strings .TrimSuffix (f , filepath .Ext (f ))] = f
63
+ }
64
+ }
65
+ merge (fl .all )
66
+ merge (fl .custom )
67
+
68
+ files := make ([]string , 0 , len (m ))
69
+ for _ , f := range m {
70
+ files = append (files , f )
71
+ }
72
+ sort .Strings (files )
73
+ return files
74
+ }
75
+
44
76
// LoadRepoConfig loads the repository config
45
- func LoadRepoConfig () {
46
- // Load .gitignore and license files and readme templates.
47
- types := []string {"gitignore" , "license" , "readme" , "label" }
48
- typeFiles := make ([][]string , 4 )
77
+ func LoadRepoConfig () error {
78
+ types := []string {"gitignore" , "license" , "readme" , "label" } // option file directories
79
+ typeFiles := make ([]optionFileList , len (types ))
49
80
for i , t := range types {
50
- files , err := options . Dir ( t )
51
- if err != nil {
52
- log . Fatal ( "Failed to get %s files: %v " , t , err )
81
+ var err error
82
+ if typeFiles [ i ]. all , err = options . Dir ( t ); err != nil {
83
+ return fmt . Errorf ( "failed to list %s files: %w " , t , err )
53
84
}
54
- if t == "label" {
55
- for i , f := range files {
56
- ext := strings .ToLower (filepath .Ext (f ))
57
- if ext == ".yaml" || ext == ".yml" {
58
- files [i ] = f [:len (f )- len (ext )]
59
- }
85
+ sort .Strings (typeFiles [i ].all )
86
+ customPath := filepath .Join (setting .CustomPath , "options" , t )
87
+ if isDir , err := util .IsDir (customPath ); err != nil {
88
+ return fmt .Errorf ("failed to check custom %s dir: %w" , t , err )
89
+ } else if isDir {
90
+ if typeFiles [i ].custom , err = util .StatDir (customPath ); err != nil {
91
+ return fmt .Errorf ("failed to list custom %s files: %w" , t , err )
60
92
}
61
93
}
62
- customPath := path .Join (setting .CustomPath , "options" , t )
63
- isDir , err := util .IsDir (customPath )
64
- if err != nil {
65
- log .Fatal ("Failed to get custom %s files: %v" , t , err )
66
- }
67
- if isDir {
68
- customFiles , err := util .StatDir (customPath )
69
- if err != nil {
70
- log .Fatal ("Failed to get custom %s files: %v" , t , err )
71
- }
72
-
73
- for _ , f := range customFiles {
74
- if ! util .SliceContainsString (files , f , true ) {
75
- files = append (files , f )
76
- }
77
- }
78
- }
79
- typeFiles [i ] = files
80
94
}
81
95
82
- Gitignores = typeFiles [0 ]
83
- Licenses = typeFiles [1 ]
84
- Readmes = typeFiles [2 ]
85
- LabelTemplatesFiles := typeFiles [3 ]
86
- sort .Strings (Gitignores )
87
- sort .Strings (Licenses )
88
- sort .Strings (Readmes )
89
- sort .Strings (LabelTemplatesFiles )
96
+ Gitignores = typeFiles [0 ].all
97
+ Licenses = typeFiles [1 ].all
98
+ Readmes = typeFiles [2 ].all
90
99
91
100
// Load label templates
92
- LabelTemplates = make (map [string ]string )
93
- for _ , templateFile := range LabelTemplatesFiles {
94
- labels , err := label .LoadFormatted (templateFile )
101
+ LabelTemplateFiles = nil
102
+ labelTemplateFileMap = map [string ]string {}
103
+ for _ , file := range mergeCustomLabelFiles (typeFiles [3 ]) {
104
+ description , err := label .LoadTemplateDescription (file )
95
105
if err != nil {
96
- log . Error ( "Failed to load labels: %v " , err )
106
+ return fmt . Errorf ( "failed to load labels: %w " , err )
97
107
}
98
- LabelTemplates [templateFile ] = labels
108
+ displayName := strings .TrimSuffix (file , filepath .Ext (file ))
109
+ labelTemplateFileMap [displayName ] = file
110
+ LabelTemplateFiles = append (LabelTemplateFiles , OptionFile {DisplayName : displayName , Description : description })
99
111
}
100
112
101
113
// Filter out invalid names and promote preferred licenses.
@@ -111,6 +123,7 @@ func LoadRepoConfig() {
111
123
}
112
124
}
113
125
Licenses = sortedLicenses
126
+ return nil
114
127
}
115
128
116
129
func prepareRepoCommit (ctx context.Context , repo * repo_model.Repository , tmpDir , repoPath string , opts CreateRepoOptions ) error {
@@ -344,7 +357,7 @@ func initRepository(ctx context.Context, repoPath string, u *user_model.User, re
344
357
345
358
// InitializeLabels adds a label set to a repository using a template
346
359
func InitializeLabels (ctx context.Context , id int64 , labelTemplate string , isOrg bool ) error {
347
- list , err := label . GetTemplateFile (labelTemplate )
360
+ list , err := LoadTemplateLabelsByDisplayName (labelTemplate )
348
361
if err != nil {
349
362
return err
350
363
}
@@ -370,3 +383,11 @@ func InitializeLabels(ctx context.Context, id int64, labelTemplate string, isOrg
370
383
}
371
384
return nil
372
385
}
386
+
387
+ // LoadTemplateLabelsByDisplayName loads a label template by its display name
388
+ func LoadTemplateLabelsByDisplayName (displayName string ) ([]* label.Label , error ) {
389
+ if fileName , ok := labelTemplateFileMap [displayName ]; ok {
390
+ return label .LoadTemplateFile (fileName )
391
+ }
392
+ return nil , label.ErrTemplateLoad {TemplateFile : displayName , OriginalError : fmt .Errorf ("label template %q not found" , displayName )}
393
+ }
0 commit comments