/
gitprovider.go
122 lines (105 loc) · 5.01 KB
/
gitprovider.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
/*
Copyright 2020 The Flux CD contributors.
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
package gitprovider
import "context"
// ProviderID is a typed string for a given Git provider
// The provider constants are defined in their respective packages.
type ProviderID string
// InfoRequest is an interface which all {Object}Info objects that can be used as Create() or Reconcile()
// requests in the Client should implement. Most likely, the struct should also implement DefaultedInfoRequest,
// as most objects have optional, defaulted fields.
type InfoRequest interface {
// ValidateInfo validates the object at {Object}.Set() and POST-time, before defaulting.
// Set (non-nil) and required fields should be validated.
ValidateInfo() error
// Equals can be used to check if this *Info request (the desired state) matches the actual
// passed in as the argument.
Equals(actual InfoRequest) bool
}
// DefaultedInfoRequest is a superset of InfoRequest, also including a Default() function that can
// modify the underlying object, adding default values. ValidateAndDefaultInfo() can be used
// to first validate, and then default.
type DefaultedInfoRequest interface {
// DefaultedInfoRequest is a superset of InfoRequest
InfoRequest
// Default will be run after validation, setting optional pointer fields to their
// default values before doing the POST request. This function MUST be registered with
// the base struct as a pointer receiver.
Default()
}
// Updatable is an interface which all objects that can be updated
// using the Client implement.
type Updatable interface {
// Update will apply the desired state in this object to the server.
// Only set fields will be respected (i.e. PATCH behaviour).
// In order to apply changes to this object, use the .Set({Resource}Info) error
// function, or cast .APIObject() to a pointer to the provider-specific type
// and set custom fields there.
//
// ErrNotFound is returned if the resource does not exist.
//
// The internal API object will be overridden with the received server data.
Update(ctx context.Context) error
}
// Deletable is an interface which all objects that can be deleted
// using the Client implement.
type Deletable interface {
// Delete deletes the current resource irreversibly.
//
// ErrNotFound is returned if the resource doesn't exist anymore.
Delete(ctx context.Context) error
}
// Reconcilable is an interface which all objects that can be reconciled
// using the Client implement.
type Reconcilable interface {
// Reconcile makes sure the desired state in this object (called "req" here) becomes
// the actual state in the backing Git provider.
//
// If req doesn't exist under the hood, it is created (actionTaken == true).
// If req doesn't equal the actual state, the resource will be updated (actionTaken == true).
// If req is already the actual state, this is a no-op (actionTaken == false).
//
// The internal API object will be overridden with the received server data if actionTaken == true.
Reconcile(ctx context.Context) (actionTaken bool, err error)
}
// Object is the interface all types should implement.
type Object interface {
// APIObject returns the underlying value that was returned from the server.
// This is always a pointer to a struct.
APIObject() interface{}
}
// OrganizationBound describes an object that is bound to a given organization, e.g. a team.
type OrganizationBound interface {
// Organization returns the OrganizationRef associated with this object.
Organization() OrganizationRef
}
// RepositoryBound describes an object that is bound to a given repository, e.g. a deploy key.
type RepositoryBound interface {
// Repository returns the RepositoryRef associated with this object.
Repository() RepositoryRef
}
// ValidateAndDefaultInfo can be used in client Create() and Reconcile() functions, where the
// request object, which implements InfoRequest, shall be first validated, and then defaulted.
// Defaulting happens at Create(), because we want to consistently apply this library's defaults
// across providers. Defaulting also happens at Reconcile(), because as the object has been created
// with defaults, the actual state fetched from the server will contain those defaults, and would
// result in a diff between the (possibly non-defaulted) request and actual state.
// TODO: Unit and integration test this.
// TODO: Document in Create() and Reconcile() that req is modified (?) and should not be used anymore.
func ValidateAndDefaultInfo(info DefaultedInfoRequest) error {
if err := info.ValidateInfo(); err != nil {
return err
}
info.Default()
return nil
}