-
Notifications
You must be signed in to change notification settings - Fork 158
/
deployments.go
130 lines (109 loc) · 3.58 KB
/
deployments.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
package empire
import (
"fmt"
"io"
"github.com/jinzhu/gorm"
"github.com/remind101/empire/pkg/jsonmessage"
"github.com/remind101/empire/twelvefactor"
"golang.org/x/net/context"
)
// deployerService is an implementation of the deployer interface that performs
// the core business logic to deploy.
type deployerService struct {
*Empire
}
// createRelease creates a new release that can be deployed
func (s *deployerService) createRelease(ctx context.Context, db *gorm.DB, ss twelvefactor.StatusStream, opts DeployOpts) (*Release, error) {
app, img := opts.App, opts.Image
// If no app is specified, attempt to find the app that relates to this
// images repository, or create it if not found.
if app == nil {
var err error
app, err = appsFindOrCreateByRepo(db, img.Repository)
if err != nil {
return nil, err
}
} else {
// If the app doesn't already have a repo attached to it, we'll attach
// this image's repo.
if err := appsEnsureRepo(db, app, img.Repository); err != nil {
return nil, err
}
}
// Grab the latest config.
config, err := s.configs.Config(db, app)
if err != nil {
return nil, err
}
// Create a new slug for the docker image.
slug, err := s.slugs.Create(ctx, db, img, opts.Output)
if err != nil {
return nil, err
}
// Create a new release for the Config
// and Slug.
desc := fmt.Sprintf("Deploy %s", img.String())
desc = appendMessageToDescription(desc, opts.User, opts.Message)
r, err := s.releases.Create(ctx, db, &Release{
App: app,
Config: config,
Slug: slug,
Description: desc,
})
return r, err
}
func (s *deployerService) createInTransaction(ctx context.Context, stream twelvefactor.StatusStream, opts DeployOpts) (*Release, error) {
tx := s.db.Begin()
r, err := s.createRelease(ctx, tx, stream, opts)
if err != nil {
tx.Rollback()
return r, err
}
return r, tx.Commit().Error
}
// Deploy is a thin wrapper around deploy to that adds the error to the
// jsonmessage stream.
func (s *deployerService) Deploy(ctx context.Context, opts DeployOpts) (*Release, error) {
w := opts.Output
var stream twelvefactor.StatusStream
if opts.Stream {
stream = w
}
r, err := s.createInTransaction(ctx, stream, opts)
if err != nil {
return r, w.Error(err)
}
if err := w.Status(fmt.Sprintf("Created new release v%d for %s", r.Version, r.App.Name)); err != nil {
return r, err
}
if err := s.releases.Release(ctx, r, stream); err != nil {
return r, w.Error(err)
}
return r, w.Status(fmt.Sprintf("Finished processing events for release v%d of %s", r.Version, r.App.Name))
}
// DeploymentStream provides a wrapper around an io.Writer for writing
// jsonmessage statuses, and implements the scheduler.StatusStream interface.
type DeploymentStream struct {
*jsonmessage.Stream
}
// NewDeploymentStream wraps the io.Writer as a DeploymentStream.
func NewDeploymentStream(w io.Writer) *DeploymentStream {
return &DeploymentStream{jsonmessage.NewStream(w)}
}
// Publish implements the scheduler.StatusStream interface.
func (w *DeploymentStream) Publish(status twelvefactor.Status) error {
return w.Status(status.Message)
}
// Status writes a simple status update to the jsonmessage stream.
func (w *DeploymentStream) Status(message string) error {
m := jsonmessage.JSONMessage{Status: fmt.Sprintf("Status: %s", message)}
return w.Encode(m)
}
// Error writes the error to the jsonmessage stream. The error that is provided
// is also returned, so that Error() can be used in return values.
func (w *DeploymentStream) Error(err error) error {
if encErr := w.Encode(jsonmessage.NewError(err)); encErr != nil {
return encErr
}
return err
}