-
Notifications
You must be signed in to change notification settings - Fork 127
/
Copy pathpackage.go
185 lines (162 loc) · 7.03 KB
/
package.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
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
package client
import (
"context"
"io"
"net/url"
"github.com/cloudfoundry/go-cfclient/v3/internal/path"
"github.com/cloudfoundry/go-cfclient/v3/resource"
)
type PackageClient commonClient
// PackageListOptions list filters
type PackageListOptions struct {
*ListOptions
GUIDs Filter `qs:"guids"` // list of package guids to filter by
States Filter `qs:"states"` // list of package states to filter by
Types Filter `qs:"types"` // list of package types to filter by, docker or bits
AppGUIDs Filter `qs:"app_guids"` // list of app guids to filter by
SpaceGUIDs Filter `qs:"space_guids"` // list of space guids to filter by
OrganizationGUIDs Filter `qs:"organization_guids"` // list of organization guids to filter by
}
// NewPackageListOptions creates new options to pass to list
func NewPackageListOptions() *PackageListOptions {
return &PackageListOptions{
ListOptions: NewListOptions(),
}
}
func (o PackageListOptions) ToQueryString() (url.Values, error) {
return o.ListOptions.ToQueryString(o)
}
// Copy the bits of a source package to a target package
func (c *PackageClient) Copy(ctx context.Context, srcPackageGUID string, destAppGUID string) (*resource.Package, error) {
var d resource.Package
r := resource.NewPackageCopy(destAppGUID)
_, err := c.client.post(ctx, path.Format("/v3/packages?source_guid=%s", srcPackageGUID), r, &d)
if err != nil {
return nil, err
}
return &d, nil
}
// Create a new package
func (c *PackageClient) Create(ctx context.Context, r *resource.PackageCreate) (*resource.Package, error) {
var p resource.Package
_, err := c.client.post(ctx, "/v3/packages", r, &p)
if err != nil {
return nil, err
}
return &p, nil
}
// Delete the specified package asynchronously and return a jobGUID
func (c *PackageClient) Delete(ctx context.Context, guid string) (string, error) {
return c.client.delete(ctx, path.Format("/v3/packages/%s", guid))
}
// Download the bits of an existing package
// It is the caller's responsibility to close the io.ReadCloser
func (c *PackageClient) Download(ctx context.Context, guid string) (io.ReadCloser, error) {
// This is the initial request, which will redirect to the blobstore location.
// The client will not automatically follow this redirect and uses a secondary
// unauthenticated client to download the bits
// https://v3-apidocs.cloudfoundry.org/version/3.128.0/index.html#download-package-bits
return c.client.download(ctx, path.Format("/v3/packages/%s/download", guid))
}
// First returns the first package matching the options or an error when less than 1 match
func (c *PackageClient) First(ctx context.Context, opts *PackageListOptions) (*resource.Package, error) {
return First[*PackageListOptions, *resource.Package](opts, func(opts *PackageListOptions) ([]*resource.Package, *Pager, error) {
return c.List(ctx, opts)
})
}
// FirstForApp returns the first package matching the options and app or an error when less than 1 match
func (c *PackageClient) FirstForApp(ctx context.Context, appGUID string, opts *PackageListOptions) (*resource.Package, error) {
return First[*PackageListOptions, *resource.Package](opts, func(opts *PackageListOptions) ([]*resource.Package, *Pager, error) {
return c.ListForApp(ctx, appGUID, opts)
})
}
// Get the specified build
func (c *PackageClient) Get(ctx context.Context, guid string) (*resource.Package, error) {
var p resource.Package
err := c.client.get(ctx, path.Format("/v3/packages/%s", guid), &p)
if err != nil {
return nil, err
}
return &p, nil
}
// List pages all the packages the user has access to
func (c *PackageClient) List(ctx context.Context, opts *PackageListOptions) ([]*resource.Package, *Pager, error) {
if opts == nil {
opts = NewPackageListOptions()
}
var res resource.PackageList
if err := c.client.list(ctx, "/v3/packages", opts.ToQueryString, &res); err != nil {
return nil, nil, err
}
pager := NewPager(res.Pagination)
return res.Resources, pager, nil
}
// ListAll retrieves all the packages the user has access to
func (c *PackageClient) ListAll(ctx context.Context, opts *PackageListOptions) ([]*resource.Package, error) {
if opts == nil {
opts = NewPackageListOptions()
}
return AutoPage[*PackageListOptions, *resource.Package](opts, func(opts *PackageListOptions) ([]*resource.Package, *Pager, error) {
return c.List(ctx, opts)
})
}
// ListForApp pages all the packages the user has access to
func (c *PackageClient) ListForApp(ctx context.Context, appGUID string, opts *PackageListOptions) ([]*resource.Package, *Pager, error) {
if opts == nil {
opts = NewPackageListOptions()
}
var res resource.PackageList
err := c.client.list(ctx, "/v3/apps/"+appGUID+"/packages", opts.ToQueryString, &res)
if err != nil {
return nil, nil, err
}
pager := NewPager(res.Pagination)
return res.Resources, pager, nil
}
// ListForAppAll retrieves all the packages the user has access to
func (c *PackageClient) ListForAppAll(ctx context.Context, appGUID string, opts *PackageListOptions) ([]*resource.Package, error) {
if opts == nil {
opts = NewPackageListOptions()
}
return AutoPage[*PackageListOptions, *resource.Package](opts, func(opts *PackageListOptions) ([]*resource.Package, *Pager, error) {
return c.ListForApp(ctx, appGUID, opts)
})
}
// PollReady waits until the package is ready, fails, or times out
func (c *PackageClient) PollReady(ctx context.Context, guid string, opts *PollingOptions) error {
return PollForStateOrTimeout(func() (string, string, error) {
pkg, err := c.Get(ctx, guid)
if pkg != nil {
return string(pkg.State), "", err
}
return "", "", err
}, string(resource.PackageStateReady), opts)
}
// Single returns a single package matching the options or an error if not exactly 1 match
func (c *PackageClient) Single(ctx context.Context, opts *PackageListOptions) (*resource.Package, error) {
return Single[*PackageListOptions, *resource.Package](opts, func(opts *PackageListOptions) ([]*resource.Package, *Pager, error) {
return c.List(ctx, opts)
})
}
// SingleForApp returns a single package matching the options for the app or an error if not exactly 1 match
func (c *PackageClient) SingleForApp(ctx context.Context, appGUID string, opts *PackageListOptions) (*resource.Package, error) {
return Single[*PackageListOptions, *resource.Package](opts, func(opts *PackageListOptions) ([]*resource.Package, *Pager, error) {
return c.ListForApp(ctx, appGUID, opts)
})
}
// Update the specified attributes of the package
func (c *PackageClient) Update(ctx context.Context, guid string, r *resource.PackageUpdate) (*resource.Package, error) {
var p resource.Package
_, err := c.client.patch(ctx, path.Format("/v3/packages/%s", guid), r, &p)
if err != nil {
return nil, err
}
return &p, nil
}
// Upload an app's zip file contents
func (c *PackageClient) Upload(ctx context.Context, guid string, zipFile io.Reader) (*resource.Package, error) {
p := path.Format("/v3/packages/%s/upload", guid)
var pkg resource.Package
_, err := c.client.postFileUpload(ctx, p, "bits", "package.zip", zipFile, &pkg)
return &pkg, err
}