Skip to content

Commit afb90e9

Browse files
jeremydSasso
authored andcommitted
Adding deployALL buttons to admin panel (#162)
1 parent 829e707 commit afb90e9

File tree

8 files changed

+137
-5
lines changed

8 files changed

+137
-5
lines changed

dashboard/src/actions/index.js

+40
Original file line numberDiff line numberDiff line change
@@ -844,3 +844,43 @@ export const cancelRelease = (project_slug, payload) => {
844844
}
845845
}
846846
}
847+
848+
export const PROJECT_DEPLOY_ALL_REQUEST = 'PROJECT_DEPLOY_ALL_REQUEST'
849+
export const PROJECT_DEPLOY_ALL_SUCCESS = 'PROJECT_DEPLOY_ALL_SUCCESS'
850+
export const PROJECT_DEPLOY_ALL_FAILURE = 'PROJECT_DEPLOY_ALL_FAILURE'
851+
852+
export const projectDeployAll = () => {
853+
return {
854+
[CALL_API]: {
855+
endpoint: `${API_ROOT}/admin/deployallprojects`,
856+
method: 'PUT',
857+
body: '',
858+
headers: {
859+
'Accept': 'application/json',
860+
'Content-Type': 'application/json',
861+
'Authorization': `Bearer ${authToken()}`
862+
},
863+
types: [ PROJECT_DEPLOY_ALL_REQUEST, PROJECT_DEPLOY_ALL_SUCCESS, PROJECT_DEPLOY_ALL_FAILURE ]
864+
}
865+
}
866+
}
867+
868+
export const SERVICE_DEPLOY_ALL_REQUEST = 'SERVICE_DEPLOY_ALL_REQUEST'
869+
export const SERVICE_DEPLOY_ALL_SUCCESS = 'SERVICE_DEPLOY_ALL_SUCCESS'
870+
export const SERVICE_DEPLOY_ALL_FAILURE = 'SERVICE_DEPLOY_ALL_FAILURE'
871+
872+
export const loadBalancerDeployAll = () => {
873+
return {
874+
[CALL_API]: {
875+
endpoint: `${API_ROOT}/admin/deployloadbalancers`,
876+
method: 'PUT',
877+
body: '',
878+
headers: {
879+
'Accept': 'application/json',
880+
'Content-Type': 'application/json',
881+
'Authorization': `Bearer ${authToken()}`
882+
},
883+
types: [ SERVICE_DEPLOY_ALL_REQUEST, SERVICE_DEPLOY_ALL_SUCCESS, SERVICE_DEPLOY_ALL_FAILURE ]
884+
}
885+
}
886+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,38 @@
1+
import React, { Component } from 'react'
2+
import { connect } from 'react-redux'
3+
import ButtonConfirmAction from '../ButtonConfirmAction'
4+
import { projectDeployAll, loadBalancerDeployAll } from '../../actions'
5+
6+
class AdminActionsList extends Component {
7+
render() {
8+
return(
9+
<div>
10+
<div className="hr-divider m-t-md m-b">
11+
<h3 className="hr-divider-content hr-divider-heading">Admin Actions</h3>
12+
</div>
13+
<ButtonConfirmAction btnLabel="Deploy All Projects" btnIconClass="fa fa-rocket" size="lg" btnClass="btn btn-warning btn-lg float-xs-center" onConfirm={() => this.onDeployAllProjectsClick()}>
14+
Are you sure you want to deploy <b>every</b> project?
15+
</ButtonConfirmAction>
16+
<ButtonConfirmAction btnLabel="Update All Services" btnIconClass="fa fa-rocket" size="lg" btnClass="btn btn-danger btn-lg float-xs-center" onConfirm={() => this.onDeployAllLoadBalancersClick()}>
17+
Are you sure you want to deploy <b>every</b> load balancer?
18+
</ButtonConfirmAction>
19+
20+
</div>
21+
)
22+
}
23+
24+
onDeployAllProjectsClick() {
25+
this.props.projectDeployAll()
26+
}
27+
onDeployAllLoadBalancersClick() {
28+
this.props.loadBalancerDeployAll()
29+
}
30+
}
31+
32+
const mapStateToProps = (_state, _ownProps) => ({
33+
})
34+
35+
export default connect(mapStateToProps, {
36+
projectDeployAll,
37+
loadBalancerDeployAll,
38+
})(AdminActionsList)

dashboard/src/components/admin/index.js

+2
Original file line numberDiff line numberDiff line change
@@ -1,12 +1,14 @@
11
import React, { Component } from 'react'
22
import { connect } from 'react-redux'
33
import ServiceSpecList from './ServiceSpecList'
4+
import AdminActionsList from './AdminActionsList'
45

56
class Admin extends Component {
67
render() {
78
return (
89
<div>
910
<ServiceSpecList specs={this.props.serviceSpecs}/>
11+
<AdminActionsList/>
1012
</div>
1113
)
1214
}

server/configs/.gitignore

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
kubeconfigs

server/plugins/codeflow/api_admin.go

+49
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,7 @@ package codeflow
33
import (
44
"log"
55
"net/http"
6+
"time"
67

78
"gopkg.in/mgo.v2/bson"
89

@@ -21,11 +22,59 @@ func (x *Admin) Register(api *rest.Api) []*rest.Route {
2122
rest.Put(x.Path+"/serviceSpecs", x.updateServiceSpec),
2223
rest.Delete(x.Path+"/serviceSpecs/#slug", x.deleteServiceSpec),
2324
rest.Get(x.Path+"/serviceSpecs/#slug/services", x.serviceSpecServices),
25+
rest.Put(x.Path+"/deployallprojects", x.deployAllProjects),
26+
rest.Put(x.Path+"/deployloadbalancers", x.deployAllLoadBalancers),
2427
)
2528
log.Printf("Started the codeflow admin handler on %s\n", x.Path)
2629
return routes
2730
}
2831

32+
// Deploys all projects (does not touch services)
33+
func (x *Admin) deployAllProjects(w rest.ResponseWriter, r *rest.Request) {
34+
projects := []Project{}
35+
m := bson.M{"deleted": false}
36+
results := db.Collection("projects").Find(m)
37+
results.Query.Sort("-$natural").All(&projects)
38+
log.Printf("Deploying %d projects", len(projects))
39+
for _, p := range projects {
40+
log.Printf("Deploying %s...", p.Slug)
41+
release := Release{}
42+
if err := GetCurrentRelease(p.Id, &release); err != nil {
43+
log.Printf("GetCurrentRelease::Error: %s; Aborting release for %s", err, p.Slug)
44+
} else {
45+
log.Printf("Deploying %s release", release.Id)
46+
err := CreateDeploy(&release)
47+
if err != nil {
48+
log.Printf("ERROR '%s' creating deployment for %s", err, p.Slug)
49+
}
50+
}
51+
}
52+
}
53+
54+
// Deploys all loadBalancers
55+
func (x *Admin) deployAllLoadBalancers(w rest.ResponseWriter, r *rest.Request) {
56+
projects := []Project{}
57+
m := bson.M{"deleted": false}
58+
results := db.Collection("projects").Find(m)
59+
results.Query.Sort("-$natural").All(&projects)
60+
for _, p := range projects {
61+
log.Printf("Finding services for '%s' project...", p.Slug)
62+
l := bson.M{"projectId": p.Id}
63+
lbResults := db.Collection("extensions").Find(l)
64+
loadBalancer := &LoadBalancer{}
65+
for lbResults.Next(loadBalancer) {
66+
log.Printf("Deploying loadbalancer %s for %s", loadBalancer.Name, p.Slug)
67+
err := ExtensionUpdated(loadBalancer)
68+
if err != nil {
69+
log.Printf("ERROR '%s' updating loadBalancer %s", err, loadBalancer.Name)
70+
}
71+
}
72+
}
73+
// Throttle so that we can be "nice" to the amazon APIs
74+
log.Println("Sleeping 1 second")
75+
time.Sleep(time.Second * 1)
76+
}
77+
2978
func (x *Admin) deleteServiceSpec(w rest.ResponseWriter, r *rest.Request) {
3079
slug := r.PathParam("slug")
3180

server/plugins/codeflow/codeflow.go

+1-1
Original file line numberDiff line numberDiff line change
@@ -312,7 +312,7 @@ func (x *Codeflow) Process(e agent.Event) error {
312312
}
313313
}
314314

315-
if err := db.Collection("features").FindOne(bson.M{"hash": payload.Hash}, &feature); err != nil {
315+
if err := db.Collection("features").FindOne(bson.M{"hash": payload.Hash, "projectId": project.Id}, &feature); err != nil {
316316
if _, ok := err.(*bongo.DocumentNotFoundError); ok {
317317
feature = Feature{
318318
ProjectId: project.Id,

server/plugins/kubedeploy/deployments.go

+4-2
Original file line numberDiff line numberDiff line change
@@ -262,12 +262,14 @@ func (x *KubeDeploy) doDeploy(e agent.Event) error {
262262

263263
config, err := clientcmd.BuildConfigFromFlags("", kubeconfig)
264264
if err != nil {
265-
panic(err.Error())
265+
log.Printf("ERROR '%s' while building kubernetes api client config. Aborting!", err)
266+
return nil
266267
}
267268

268269
clientset, err := kubernetes.NewForConfig(config)
269270
if err != nil {
270-
log.Println("Error getting cluster config.")
271+
log.Println("Error getting cluster config. Aborting!")
272+
return nil
271273
}
272274

273275
data := e.Payload.(plugins.DockerDeploy)

server/plugins/kubedeploy/loadbalancers.go

+2-2
Original file line numberDiff line numberDiff line change
@@ -216,8 +216,8 @@ func (x *KubeDeploy) doLoadBalancer(e agent.Event) error {
216216
var ELBDNS string
217217
if payload.Type == plugins.External || payload.Type == plugins.Office {
218218
log.Printf("Waiting for ELB address for %s", payload.Name)
219-
// Timeout waiting for ELB DNS name after 600 seconds
220-
timeout := 600
219+
// Timeout waiting for ELB DNS name after 900 seconds
220+
timeout := 900
221221
for {
222222
elbResult, elbErr := coreInterface.Services(namespace).Get(payload.Name, meta_v1.GetOptions{})
223223
if elbErr != nil {

0 commit comments

Comments
 (0)