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.

file_format.go 2.3KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112
  1. // Copyright 2022 The Gitea Authors. All rights reserved.
  2. // Use of this source code is governed by a MIT-style
  3. // license that can be found in the LICENSE file.
  4. package migration
  5. import (
  6. "fmt"
  7. "os"
  8. "strings"
  9. "code.gitea.io/gitea/modules/json"
  10. "code.gitea.io/gitea/modules/log"
  11. "github.com/santhosh-tekuri/jsonschema/v5"
  12. "gopkg.in/yaml.v2"
  13. )
  14. // Load project data from file, with optional validation
  15. func Load(filename string, data interface{}, validation bool) error {
  16. isJSON := strings.HasSuffix(filename, ".json")
  17. bs, err := os.ReadFile(filename)
  18. if err != nil {
  19. return err
  20. }
  21. if validation {
  22. err := validate(bs, data, isJSON)
  23. if err != nil {
  24. return err
  25. }
  26. }
  27. return unmarshal(bs, data, isJSON)
  28. }
  29. func unmarshal(bs []byte, data interface{}, isJSON bool) error {
  30. if isJSON {
  31. return json.Unmarshal(bs, data)
  32. }
  33. return yaml.Unmarshal(bs, data)
  34. }
  35. func getSchema(filename string) (*jsonschema.Schema, error) {
  36. c := jsonschema.NewCompiler()
  37. c.LoadURL = openSchema
  38. return c.Compile(filename)
  39. }
  40. func validate(bs []byte, datatype interface{}, isJSON bool) error {
  41. var v interface{}
  42. err := unmarshal(bs, &v, isJSON)
  43. if err != nil {
  44. return err
  45. }
  46. if !isJSON {
  47. v, err = toStringKeys(v)
  48. if err != nil {
  49. return err
  50. }
  51. }
  52. var schemaFilename string
  53. switch datatype := datatype.(type) {
  54. case *[]*Issue:
  55. schemaFilename = "issue.json"
  56. case *[]*Milestone:
  57. schemaFilename = "milestone.json"
  58. default:
  59. return fmt.Errorf("file_format:validate: %T has not a validation implemented", datatype)
  60. }
  61. sch, err := getSchema(schemaFilename)
  62. if err != nil {
  63. return err
  64. }
  65. err = sch.Validate(v)
  66. if err != nil {
  67. log.Error("migration validation with %s failed for\n%s", schemaFilename, string(bs))
  68. }
  69. return err
  70. }
  71. func toStringKeys(val interface{}) (interface{}, error) {
  72. var err error
  73. switch val := val.(type) {
  74. case map[interface{}]interface{}:
  75. m := make(map[string]interface{})
  76. for k, v := range val {
  77. k, ok := k.(string)
  78. if !ok {
  79. return nil, fmt.Errorf("found non-string key %T %s", k, k)
  80. }
  81. m[k], err = toStringKeys(v)
  82. if err != nil {
  83. return nil, err
  84. }
  85. }
  86. return m, nil
  87. case []interface{}:
  88. l := make([]interface{}, len(val))
  89. for i, v := range val {
  90. l[i], err = toStringKeys(v)
  91. if err != nil {
  92. return nil, err
  93. }
  94. }
  95. return l, nil
  96. default:
  97. return val, nil
  98. }
  99. }