2023-03-01 18:44:23 -05:00
|
|
|
// Copyright 2023 The Gitea Authors. All rights reserved.
|
|
|
|
// SPDX-License-Identifier: MIT
|
|
|
|
|
|
|
|
package label
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"code.gitea.io/gitea/modules/options"
|
|
|
|
|
|
|
|
"gopkg.in/yaml.v3"
|
|
|
|
)
|
|
|
|
|
|
|
|
type labelFile struct {
|
|
|
|
Labels []*Label `yaml:"labels"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// ErrTemplateLoad represents a "ErrTemplateLoad" kind of error.
|
|
|
|
type ErrTemplateLoad struct {
|
|
|
|
TemplateFile string
|
|
|
|
OriginalError error
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsErrTemplateLoad checks if an error is a ErrTemplateLoad.
|
|
|
|
func IsErrTemplateLoad(err error) bool {
|
|
|
|
_, ok := err.(ErrTemplateLoad)
|
|
|
|
return ok
|
|
|
|
}
|
|
|
|
|
|
|
|
func (err ErrTemplateLoad) Error() string {
|
2023-04-10 03:44:02 -05:00
|
|
|
return fmt.Sprintf("failed to load label template file %q: %v", err.TemplateFile, err.OriginalError)
|
2023-03-01 18:44:23 -05:00
|
|
|
}
|
|
|
|
|
2023-04-10 03:44:02 -05:00
|
|
|
// LoadTemplateFile loads the label template file by given file name, returns a slice of Label structs.
|
|
|
|
func LoadTemplateFile(fileName string) ([]*Label, error) {
|
|
|
|
data, err := options.Labels(fileName)
|
2023-03-01 18:44:23 -05:00
|
|
|
if err != nil {
|
2023-04-10 03:44:02 -05:00
|
|
|
return nil, ErrTemplateLoad{fileName, fmt.Errorf("LoadTemplateFile: %w", err)}
|
2023-03-01 18:44:23 -05:00
|
|
|
}
|
|
|
|
|
2023-04-10 03:44:02 -05:00
|
|
|
if strings.HasSuffix(fileName, ".yaml") || strings.HasSuffix(fileName, ".yml") {
|
|
|
|
return parseYamlFormat(fileName, data)
|
|
|
|
}
|
|
|
|
return parseLegacyFormat(fileName, data)
|
2023-03-01 18:44:23 -05:00
|
|
|
}
|
|
|
|
|
2023-04-10 03:44:02 -05:00
|
|
|
func parseYamlFormat(fileName string, data []byte) ([]*Label, error) {
|
2023-03-01 18:44:23 -05:00
|
|
|
lf := &labelFile{}
|
|
|
|
|
|
|
|
if err := yaml.Unmarshal(data, lf); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Validate label data and fix colors
|
|
|
|
for _, l := range lf.Labels {
|
|
|
|
l.Color = strings.TrimSpace(l.Color)
|
|
|
|
if len(l.Name) == 0 || len(l.Color) == 0 {
|
2023-04-10 03:44:02 -05:00
|
|
|
return nil, ErrTemplateLoad{fileName, errors.New("label name and color are required fields")}
|
2023-03-01 18:44:23 -05:00
|
|
|
}
|
|
|
|
color, err := NormalizeColor(l.Color)
|
|
|
|
if err != nil {
|
2023-04-10 03:44:02 -05:00
|
|
|
return nil, ErrTemplateLoad{fileName, fmt.Errorf("bad HTML color code '%s' in label: %s", l.Color, l.Name)}
|
2023-03-01 18:44:23 -05:00
|
|
|
}
|
|
|
|
l.Color = color
|
|
|
|
}
|
|
|
|
|
|
|
|
return lf.Labels, nil
|
|
|
|
}
|
|
|
|
|
2023-04-10 03:44:02 -05:00
|
|
|
func parseLegacyFormat(fileName string, data []byte) ([]*Label, error) {
|
2023-03-01 18:44:23 -05:00
|
|
|
lines := strings.Split(string(data), "\n")
|
|
|
|
list := make([]*Label, 0, len(lines))
|
|
|
|
for i := 0; i < len(lines); i++ {
|
|
|
|
line := strings.TrimSpace(lines[i])
|
|
|
|
if len(line) == 0 {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
parts, description, _ := strings.Cut(line, ";")
|
|
|
|
|
2023-04-10 03:44:02 -05:00
|
|
|
color, labelName, ok := strings.Cut(parts, " ")
|
2023-03-01 18:44:23 -05:00
|
|
|
if !ok {
|
2023-04-10 03:44:02 -05:00
|
|
|
return nil, ErrTemplateLoad{fileName, fmt.Errorf("line is malformed: %s", line)}
|
2023-03-01 18:44:23 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
color, err := NormalizeColor(color)
|
|
|
|
if err != nil {
|
2023-04-10 03:44:02 -05:00
|
|
|
return nil, ErrTemplateLoad{fileName, fmt.Errorf("bad HTML color code '%s' in line: %s", color, line)}
|
2023-03-01 18:44:23 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
list = append(list, &Label{
|
2023-04-10 03:44:02 -05:00
|
|
|
Name: strings.TrimSpace(labelName),
|
2023-03-01 18:44:23 -05:00
|
|
|
Color: color,
|
|
|
|
Description: strings.TrimSpace(description),
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
return list, nil
|
|
|
|
}
|
|
|
|
|
2023-04-10 03:44:02 -05:00
|
|
|
// LoadTemplateDescription loads the labels from a template file, returns a description string by joining each Label.Name with comma
|
|
|
|
func LoadTemplateDescription(fileName string) (string, error) {
|
2023-03-01 18:44:23 -05:00
|
|
|
var buf strings.Builder
|
2023-04-10 03:44:02 -05:00
|
|
|
list, err := LoadTemplateFile(fileName)
|
2023-03-01 18:44:23 -05:00
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
for i := 0; i < len(list); i++ {
|
|
|
|
if i > 0 {
|
|
|
|
buf.WriteString(", ")
|
|
|
|
}
|
|
|
|
buf.WriteString(list[i].Name)
|
|
|
|
}
|
|
|
|
return buf.String(), nil
|
|
|
|
}
|