forked from knative/func
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathinstances.go
118 lines (106 loc) · 3.49 KB
/
instances.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
package function
import (
"context"
"errors"
"fmt"
)
const (
EnvironmentLocal = "local"
EnvironmentRemote = "remote"
)
var (
ErrNotInitialized = errors.New("function is not initialized")
ErrNotRunning = errors.New("function not running")
ErrRootRequired = errors.New("function root path is required")
ErrEnvironmentNotFound = errors.New("environment not found")
)
// Instances manager
//
// Instances are point-in-time snapshots of a function's runtime state in
// a given environment. By default 'local' and 'remote' environmnts are
// available when a function is run locally and deployed (respectively).
type Instances struct {
client *Client
}
// newInstances creates a new manager of instances.
func newInstances(client *Client) *Instances {
return &Instances{client: client}
}
// Get the instance data for a function in the named environment.
// For convenient access to the default 'local' and 'remote' environment
// see the Local and Remote methods, respectively.
// Instance returned is populated with a point-in-time snapshot of the
// function state in the named environment.
func (s *Instances) Get(ctx context.Context, f Function, environment string) (Instance, error) {
switch environment {
case EnvironmentLocal:
return s.Local(ctx, f)
case EnvironmentRemote:
return s.Remote(ctx, f.Name, f.Root)
default:
// Future versions will support additional ad-hoc named environments, such
// as for testing. Local and remote remaining the base cases.
return Instance{}, ErrEnvironmentNotFound
}
}
// Local instance details for the function
// If the function is not running locally the error returned is ErrNotRunning
func (s *Instances) Local(ctx context.Context, f Function) (Instance, error) {
var i Instance
// To create a local instance the function must have a root path defined
// which contains an initialized function and be running.
if f.Root == "" {
return i, ErrRootRequired
}
if !f.Initialized() {
return i, ErrNotInitialized
}
ports := jobPorts(f)
if len(ports) == 0 {
return i, ErrNotRunning
}
route := fmt.Sprintf("http://localhost:%s/", ports[0])
return Instance{
Route: route,
Routes: []string{route},
Name: f.Name,
}, nil
}
// Remote instance details for the function
//
// Since this is specific to the implicitly available 'remote' environment, the
// request can be completed with either a name or the local source. Therefore
// either name or root path can be passed. If name is not passed, the function
// at root is loaded and its name used for describing the remote instance.
// Name takes precedence.
func (s *Instances) Remote(ctx context.Context, name, root string) (Instance, error) {
var (
f Function
err error
)
// Error if name and root disagree
// If both a name and root were passed but the function at the root either
// does not exist or does not match the name, fail fast.
// The purpose of this method's signature is to allow passing either name or
// root, but doing so requires that we manually validate.
if name != "" && root != "" {
f, err = NewFunction(root)
if err != nil {
return Instance{}, err
}
if name != f.Name {
return Instance{}, errors.New(
"name passed does not match name of the function at root. " +
"Try passing either name or root rather than both.")
}
}
// Name takes precedence if provided
if name != "" {
f = Function{Name: name}
} else {
if f, err = NewFunction(root); err != nil {
return Instance{}, err
}
}
return s.client.describer.Describe(ctx, f.Name)
}