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.

package_property.go 4.0KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121
  1. // Copyright 2022 The Gitea Authors. All rights reserved.
  2. // SPDX-License-Identifier: MIT
  3. package packages
  4. import (
  5. "context"
  6. "code.gitea.io/gitea/models/db"
  7. "xorm.io/builder"
  8. )
  9. func init() {
  10. db.RegisterModel(new(PackageProperty))
  11. }
  12. type PropertyType int64
  13. const (
  14. // PropertyTypeVersion means the reference is a package version
  15. PropertyTypeVersion PropertyType = iota // 0
  16. // PropertyTypeFile means the reference is a package file
  17. PropertyTypeFile // 1
  18. // PropertyTypePackage means the reference is a package
  19. PropertyTypePackage // 2
  20. )
  21. // PackageProperty represents a property of a package, version or file
  22. type PackageProperty struct {
  23. ID int64 `xorm:"pk autoincr"`
  24. RefType PropertyType `xorm:"INDEX NOT NULL"`
  25. RefID int64 `xorm:"INDEX NOT NULL"`
  26. Name string `xorm:"INDEX NOT NULL"`
  27. Value string `xorm:"TEXT NOT NULL"`
  28. }
  29. // InsertProperty creates a property
  30. func InsertProperty(ctx context.Context, refType PropertyType, refID int64, name, value string) (*PackageProperty, error) {
  31. pp := &PackageProperty{
  32. RefType: refType,
  33. RefID: refID,
  34. Name: name,
  35. Value: value,
  36. }
  37. _, err := db.GetEngine(ctx).Insert(pp)
  38. return pp, err
  39. }
  40. // GetProperties gets all properties
  41. func GetProperties(ctx context.Context, refType PropertyType, refID int64) ([]*PackageProperty, error) {
  42. pps := make([]*PackageProperty, 0, 10)
  43. return pps, db.GetEngine(ctx).Where("ref_type = ? AND ref_id = ?", refType, refID).Find(&pps)
  44. }
  45. // GetPropertiesByName gets all properties with a specific name
  46. func GetPropertiesByName(ctx context.Context, refType PropertyType, refID int64, name string) ([]*PackageProperty, error) {
  47. pps := make([]*PackageProperty, 0, 10)
  48. return pps, db.GetEngine(ctx).Where("ref_type = ? AND ref_id = ? AND name = ?", refType, refID, name).Find(&pps)
  49. }
  50. // UpdateProperty updates a property
  51. func UpdateProperty(ctx context.Context, pp *PackageProperty) error {
  52. _, err := db.GetEngine(ctx).ID(pp.ID).Update(pp)
  53. return err
  54. }
  55. // DeleteAllProperties deletes all properties of a ref
  56. func DeleteAllProperties(ctx context.Context, refType PropertyType, refID int64) error {
  57. _, err := db.GetEngine(ctx).Where("ref_type = ? AND ref_id = ?", refType, refID).Delete(&PackageProperty{})
  58. return err
  59. }
  60. // DeletePropertyByID deletes a property
  61. func DeletePropertyByID(ctx context.Context, propertyID int64) error {
  62. _, err := db.GetEngine(ctx).ID(propertyID).Delete(&PackageProperty{})
  63. return err
  64. }
  65. // DeletePropertyByName deletes properties by name
  66. func DeletePropertyByName(ctx context.Context, refType PropertyType, refID int64, name string) error {
  67. _, err := db.GetEngine(ctx).Where("ref_type = ? AND ref_id = ? AND name = ?", refType, refID, name).Delete(&PackageProperty{})
  68. return err
  69. }
  70. type DistinctPropertyDependency struct {
  71. Name string
  72. Value string
  73. }
  74. // GetDistinctPropertyValues returns all distinct property values for a given type.
  75. // Optional: Search only in dependence of another property.
  76. func GetDistinctPropertyValues(ctx context.Context, packageType Type, ownerID int64, refType PropertyType, propertyName string, dep *DistinctPropertyDependency) ([]string, error) {
  77. var cond builder.Cond = builder.Eq{
  78. "package_property.ref_type": refType,
  79. "package_property.name": propertyName,
  80. "package.type": packageType,
  81. "package.owner_id": ownerID,
  82. }
  83. if dep != nil {
  84. innerCond := builder.
  85. Expr("pp.ref_id = package_property.ref_id").
  86. And(builder.Eq{
  87. "pp.ref_type": refType,
  88. "pp.name": dep.Name,
  89. "pp.value": dep.Value,
  90. })
  91. cond = cond.And(builder.Exists(builder.Select("pp.ref_id").From("package_property pp").Where(innerCond)))
  92. }
  93. values := make([]string, 0, 5)
  94. return values, db.GetEngine(ctx).
  95. Table("package_property").
  96. Distinct("package_property.value").
  97. Join("INNER", "package_file", "package_file.id = package_property.ref_id").
  98. Join("INNER", "package_version", "package_version.id = package_file.version_id").
  99. Join("INNER", "package", "package.id = package_version.package_id").
  100. Where(cond).
  101. Find(&values)
  102. }