2022-09-02 15:58:49 +08:00
|
|
|
// Copyright 2022 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 template
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"net/url"
|
|
|
|
"regexp"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
|
2022-10-12 07:18:26 +02:00
|
|
|
"code.gitea.io/gitea/modules/container"
|
2022-09-02 15:58:49 +08:00
|
|
|
api "code.gitea.io/gitea/modules/structs"
|
|
|
|
|
|
|
|
"gitea.com/go-chi/binding"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Validate checks whether an IssueTemplate is considered valid, and returns the first error
|
|
|
|
func Validate(template *api.IssueTemplate) error {
|
|
|
|
if err := validateMetadata(template); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if template.Type() == api.IssueTemplateTypeYaml {
|
|
|
|
if err := validateYaml(template); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func validateMetadata(template *api.IssueTemplate) error {
|
|
|
|
if strings.TrimSpace(template.Name) == "" {
|
|
|
|
return fmt.Errorf("'name' is required")
|
|
|
|
}
|
|
|
|
if strings.TrimSpace(template.About) == "" {
|
|
|
|
return fmt.Errorf("'about' is required")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func validateYaml(template *api.IssueTemplate) error {
|
|
|
|
if len(template.Fields) == 0 {
|
|
|
|
return fmt.Errorf("'body' is required")
|
|
|
|
}
|
2022-10-12 07:18:26 +02:00
|
|
|
ids := make(container.Set[string])
|
2022-09-02 15:58:49 +08:00
|
|
|
for idx, field := range template.Fields {
|
|
|
|
if err := validateID(field, idx, ids); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := validateLabel(field, idx); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
position := newErrorPosition(idx, field.Type)
|
|
|
|
switch field.Type {
|
|
|
|
case api.IssueFormFieldTypeMarkdown:
|
|
|
|
if err := validateStringItem(position, field.Attributes, true, "value"); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
case api.IssueFormFieldTypeTextarea:
|
|
|
|
if err := validateStringItem(position, field.Attributes, false,
|
|
|
|
"description",
|
|
|
|
"placeholder",
|
|
|
|
"value",
|
|
|
|
"render",
|
|
|
|
); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
case api.IssueFormFieldTypeInput:
|
|
|
|
if err := validateStringItem(position, field.Attributes, false,
|
|
|
|
"description",
|
|
|
|
"placeholder",
|
|
|
|
"value",
|
|
|
|
); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := validateBoolItem(position, field.Validations, "is_number"); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := validateStringItem(position, field.Validations, false, "regex"); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
case api.IssueFormFieldTypeDropdown:
|
|
|
|
if err := validateStringItem(position, field.Attributes, false, "description"); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := validateBoolItem(position, field.Attributes, "multiple"); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := validateOptions(field, idx); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
case api.IssueFormFieldTypeCheckboxes:
|
|
|
|
if err := validateStringItem(position, field.Attributes, false, "description"); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := validateOptions(field, idx); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
return position.Errorf("unknown type")
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := validateRequired(field, idx); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func validateLabel(field *api.IssueFormField, idx int) error {
|
|
|
|
if field.Type == api.IssueFormFieldTypeMarkdown {
|
|
|
|
// The label is not required for a markdown field
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return validateStringItem(newErrorPosition(idx, field.Type), field.Attributes, true, "label")
|
|
|
|
}
|
|
|
|
|
|
|
|
func validateRequired(field *api.IssueFormField, idx int) error {
|
|
|
|
if field.Type == api.IssueFormFieldTypeMarkdown || field.Type == api.IssueFormFieldTypeCheckboxes {
|
|
|
|
// The label is not required for a markdown or checkboxes field
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return validateBoolItem(newErrorPosition(idx, field.Type), field.Validations, "required")
|
|
|
|
}
|
|
|
|
|
2022-10-12 07:18:26 +02:00
|
|
|
func validateID(field *api.IssueFormField, idx int, ids container.Set[string]) error {
|
2022-09-02 15:58:49 +08:00
|
|
|
if field.Type == api.IssueFormFieldTypeMarkdown {
|
|
|
|
// The ID is not required for a markdown field
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
position := newErrorPosition(idx, field.Type)
|
|
|
|
if field.ID == "" {
|
|
|
|
// If the ID is empty in yaml, template.Unmarshal will auto autofill it, so it cannot be empty
|
|
|
|
return position.Errorf("'id' is required")
|
|
|
|
}
|
|
|
|
if binding.AlphaDashPattern.MatchString(field.ID) {
|
|
|
|
return position.Errorf("'id' should contain only alphanumeric, '-' and '_'")
|
|
|
|
}
|
2022-10-12 07:18:26 +02:00
|
|
|
if !ids.Add(field.ID) {
|
2022-09-02 15:58:49 +08:00
|
|
|
return position.Errorf("'id' should be unique")
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func validateOptions(field *api.IssueFormField, idx int) error {
|
|
|
|
if field.Type != api.IssueFormFieldTypeDropdown && field.Type != api.IssueFormFieldTypeCheckboxes {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
position := newErrorPosition(idx, field.Type)
|
|
|
|
|
|
|
|
options, ok := field.Attributes["options"].([]interface{})
|
|
|
|
if !ok || len(options) == 0 {
|
|
|
|
return position.Errorf("'options' is required and should be a array")
|
|
|
|
}
|
|
|
|
|
|
|
|
for optIdx, option := range options {
|
|
|
|
position := newErrorPosition(idx, field.Type, optIdx)
|
|
|
|
switch field.Type {
|
|
|
|
case api.IssueFormFieldTypeDropdown:
|
|
|
|
if _, ok := option.(string); !ok {
|
|
|
|
return position.Errorf("should be a string")
|
|
|
|
}
|
|
|
|
case api.IssueFormFieldTypeCheckboxes:
|
2022-11-20 18:44:20 +08:00
|
|
|
opt, ok := option.(map[string]interface{})
|
2022-09-02 15:58:49 +08:00
|
|
|
if !ok {
|
|
|
|
return position.Errorf("should be a dictionary")
|
|
|
|
}
|
|
|
|
if label, ok := opt["label"].(string); !ok || label == "" {
|
|
|
|
return position.Errorf("'label' is required and should be a string")
|
|
|
|
}
|
|
|
|
|
|
|
|
if required, ok := opt["required"]; ok {
|
|
|
|
if _, ok := required.(bool); !ok {
|
|
|
|
return position.Errorf("'required' should be a bool")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func validateStringItem(position errorPosition, m map[string]interface{}, required bool, names ...string) error {
|
|
|
|
for _, name := range names {
|
|
|
|
v, ok := m[name]
|
|
|
|
if !ok {
|
|
|
|
if required {
|
|
|
|
return position.Errorf("'%s' is required", name)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
attr, ok := v.(string)
|
|
|
|
if !ok {
|
|
|
|
return position.Errorf("'%s' should be a string", name)
|
|
|
|
}
|
|
|
|
if strings.TrimSpace(attr) == "" && required {
|
|
|
|
return position.Errorf("'%s' is required", name)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func validateBoolItem(position errorPosition, m map[string]interface{}, names ...string) error {
|
|
|
|
for _, name := range names {
|
|
|
|
v, ok := m[name]
|
|
|
|
if !ok {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
if _, ok := v.(bool); !ok {
|
|
|
|
return position.Errorf("'%s' should be a bool", name)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
type errorPosition string
|
|
|
|
|
|
|
|
func (p errorPosition) Errorf(format string, a ...interface{}) error {
|
|
|
|
return fmt.Errorf(string(p)+": "+format, a...)
|
|
|
|
}
|
|
|
|
|
|
|
|
func newErrorPosition(fieldIdx int, fieldType api.IssueFormFieldType, optionIndex ...int) errorPosition {
|
|
|
|
ret := fmt.Sprintf("body[%d](%s)", fieldIdx, fieldType)
|
|
|
|
if len(optionIndex) > 0 {
|
|
|
|
ret += fmt.Sprintf(", option[%d]", optionIndex[0])
|
|
|
|
}
|
|
|
|
return errorPosition(ret)
|
|
|
|
}
|
|
|
|
|
|
|
|
// RenderToMarkdown renders template to markdown with specified values
|
|
|
|
func RenderToMarkdown(template *api.IssueTemplate, values url.Values) string {
|
|
|
|
builder := &strings.Builder{}
|
|
|
|
|
|
|
|
for _, field := range template.Fields {
|
|
|
|
f := &valuedField{
|
|
|
|
IssueFormField: field,
|
|
|
|
Values: values,
|
|
|
|
}
|
|
|
|
if f.ID == "" {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
f.WriteTo(builder)
|
|
|
|
}
|
|
|
|
|
|
|
|
return builder.String()
|
|
|
|
}
|
|
|
|
|
|
|
|
type valuedField struct {
|
|
|
|
*api.IssueFormField
|
|
|
|
url.Values
|
|
|
|
}
|
|
|
|
|
|
|
|
func (f *valuedField) WriteTo(builder *strings.Builder) {
|
|
|
|
if f.Type == api.IssueFormFieldTypeMarkdown {
|
|
|
|
// markdown blocks do not appear in output
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
// write label
|
|
|
|
_, _ = fmt.Fprintf(builder, "### %s\n\n", f.Label())
|
|
|
|
|
|
|
|
blankPlaceholder := "_No response_\n"
|
|
|
|
|
|
|
|
// write body
|
|
|
|
switch f.Type {
|
|
|
|
case api.IssueFormFieldTypeCheckboxes:
|
|
|
|
for _, option := range f.Options() {
|
|
|
|
checked := " "
|
|
|
|
if option.IsChecked() {
|
|
|
|
checked = "x"
|
|
|
|
}
|
|
|
|
_, _ = fmt.Fprintf(builder, "- [%s] %s\n", checked, option.Label())
|
|
|
|
}
|
|
|
|
case api.IssueFormFieldTypeDropdown:
|
|
|
|
var checkeds []string
|
|
|
|
for _, option := range f.Options() {
|
|
|
|
if option.IsChecked() {
|
|
|
|
checkeds = append(checkeds, option.Label())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if len(checkeds) > 0 {
|
|
|
|
_, _ = fmt.Fprintf(builder, "%s\n", strings.Join(checkeds, ", "))
|
|
|
|
} else {
|
|
|
|
_, _ = fmt.Fprint(builder, blankPlaceholder)
|
|
|
|
}
|
|
|
|
case api.IssueFormFieldTypeInput:
|
|
|
|
if value := f.Value(); value == "" {
|
|
|
|
_, _ = fmt.Fprint(builder, blankPlaceholder)
|
|
|
|
} else {
|
|
|
|
_, _ = fmt.Fprintf(builder, "%s\n", value)
|
|
|
|
}
|
|
|
|
case api.IssueFormFieldTypeTextarea:
|
|
|
|
if value := f.Value(); value == "" {
|
|
|
|
_, _ = fmt.Fprint(builder, blankPlaceholder)
|
|
|
|
} else if render := f.Render(); render != "" {
|
|
|
|
quotes := minQuotes(value)
|
|
|
|
_, _ = fmt.Fprintf(builder, "%s%s\n%s\n%s\n", quotes, f.Render(), value, quotes)
|
|
|
|
} else {
|
|
|
|
_, _ = fmt.Fprintf(builder, "%s\n", value)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_, _ = fmt.Fprintln(builder)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (f *valuedField) Label() string {
|
|
|
|
if label, ok := f.Attributes["label"].(string); ok {
|
|
|
|
return label
|
|
|
|
}
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
func (f *valuedField) Render() string {
|
|
|
|
if render, ok := f.Attributes["render"].(string); ok {
|
|
|
|
return render
|
|
|
|
}
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
func (f *valuedField) Value() string {
|
|
|
|
return strings.TrimSpace(f.Get(fmt.Sprintf("form-field-" + f.ID)))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (f *valuedField) Options() []*valuedOption {
|
|
|
|
if options, ok := f.Attributes["options"].([]interface{}); ok {
|
|
|
|
ret := make([]*valuedOption, 0, len(options))
|
|
|
|
for i, option := range options {
|
|
|
|
ret = append(ret, &valuedOption{
|
|
|
|
index: i,
|
|
|
|
data: option,
|
|
|
|
field: f,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
return ret
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
type valuedOption struct {
|
|
|
|
index int
|
|
|
|
data interface{}
|
|
|
|
field *valuedField
|
|
|
|
}
|
|
|
|
|
|
|
|
func (o *valuedOption) Label() string {
|
|
|
|
switch o.field.Type {
|
|
|
|
case api.IssueFormFieldTypeDropdown:
|
|
|
|
if label, ok := o.data.(string); ok {
|
|
|
|
return label
|
|
|
|
}
|
|
|
|
case api.IssueFormFieldTypeCheckboxes:
|
2022-11-20 18:44:20 +08:00
|
|
|
if vs, ok := o.data.(map[string]interface{}); ok {
|
2022-09-02 15:58:49 +08:00
|
|
|
if v, ok := vs["label"].(string); ok {
|
|
|
|
return v
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
func (o *valuedOption) IsChecked() bool {
|
|
|
|
switch o.field.Type {
|
|
|
|
case api.IssueFormFieldTypeDropdown:
|
|
|
|
checks := strings.Split(o.field.Get(fmt.Sprintf("form-field-%s", o.field.ID)), ",")
|
|
|
|
idx := strconv.Itoa(o.index)
|
|
|
|
for _, v := range checks {
|
|
|
|
if v == idx {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
case api.IssueFormFieldTypeCheckboxes:
|
|
|
|
return o.field.Get(fmt.Sprintf("form-field-%s-%d", o.field.ID, o.index)) == "on"
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
var minQuotesRegex = regexp.MustCompilePOSIX("^`{3,}")
|
|
|
|
|
|
|
|
// minQuotes return 3 or more back-quotes.
|
|
|
|
// If n back-quotes exists, use n+1 back-quotes to quote.
|
|
|
|
func minQuotes(value string) string {
|
|
|
|
ret := "```"
|
|
|
|
for _, v := range minQuotesRegex.FindAllString(value, -1) {
|
|
|
|
if len(v) >= len(ret) {
|
|
|
|
ret = v + "`"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ret
|
|
|
|
}
|