1
|
|
|
<?php |
2
|
|
|
|
3
|
|
|
|
4
|
|
|
class DeployPlanDispatcher extends Dispatcher { |
5
|
|
|
|
6
|
|
|
const ACTION_PLAN = 'plan'; |
7
|
|
|
|
8
|
|
|
/** |
9
|
|
|
* @var array |
10
|
|
|
*/ |
11
|
|
|
private static $action_types = [ |
12
|
|
|
self::ACTION_PLAN |
13
|
|
|
]; |
14
|
|
|
|
15
|
|
|
/** |
16
|
|
|
* @var array |
17
|
|
|
*/ |
18
|
|
|
public static $allowed_actions = [ |
19
|
|
|
'gitupdate', |
20
|
|
|
'gitrefs', |
21
|
|
|
'deploysummary' |
22
|
|
|
]; |
23
|
|
|
|
24
|
|
|
/** |
25
|
|
|
* @var \DNProject |
26
|
|
|
*/ |
27
|
|
|
protected $project = null; |
28
|
|
|
|
29
|
|
|
/** |
30
|
|
|
* @var \DNEnvironment |
31
|
|
|
*/ |
32
|
|
|
protected $environment = null; |
33
|
|
|
|
34
|
|
|
public function init() { |
35
|
|
|
parent::init(); |
36
|
|
|
|
37
|
|
|
$this->project = $this->getCurrentProject(); |
38
|
|
|
|
39
|
|
|
if(!$this->project) { |
40
|
|
|
return $this->project404Response(); |
41
|
|
|
} |
42
|
|
|
|
43
|
|
|
// Performs canView permission check by limiting visible projects |
44
|
|
|
$this->environment = $this->getCurrentEnvironment($this->project); |
45
|
|
|
if(!$this->environment) { |
46
|
|
|
return $this->environment404Response(); |
47
|
|
|
} |
48
|
|
|
} |
49
|
|
|
|
50
|
|
|
/** |
51
|
|
|
* @return string |
52
|
|
|
*/ |
53
|
|
|
public function Link() { |
54
|
|
|
return \Controller::join_links($this->environment->Link(), self::ACTION_PLAN); |
55
|
|
|
} |
56
|
|
|
|
57
|
|
|
/** |
58
|
|
|
* |
59
|
|
|
* @param \SS_HTTPRequest $request |
60
|
|
|
* |
61
|
|
|
* @return \HTMLText|\SS_HTTPResponse |
62
|
|
|
*/ |
63
|
|
|
public function index(\SS_HTTPRequest $request) { |
64
|
|
|
$this->setCurrentActionType(self::ACTION_PLAN); |
65
|
|
|
return $this->customise([ |
66
|
|
|
'Environment' => $this->environment |
67
|
|
|
])->renderWith(['Plan', 'DNRoot']); |
68
|
|
|
} |
69
|
|
|
|
70
|
|
|
/** |
71
|
|
|
* @param SS_HTTPRequest $request |
72
|
|
|
* @return SS_HTTPResponse |
73
|
|
|
*/ |
74
|
|
|
public function gitupdate(SS_HTTPRequest $request) { |
75
|
|
|
switch($request->httpMethod()) { |
76
|
|
|
case 'POST': |
77
|
|
|
return $this->createFetch(); |
78
|
|
|
case 'GET': |
79
|
|
|
return $this->getFetch($this->getRequest()->param('ID')); |
80
|
|
|
default: |
81
|
|
|
return $this->getAPIResponse('Method not allowed, requires POST or GET/{id}', 405); |
82
|
|
|
} |
83
|
|
|
} |
84
|
|
|
|
85
|
|
|
/** |
86
|
|
|
* @param SS_HTTPRequest $request |
87
|
|
|
* |
88
|
|
|
* @return string |
89
|
|
|
*/ |
90
|
|
|
public function gitrefs(\SS_HTTPRequest $request) { |
91
|
|
|
|
92
|
|
|
$refs = []; |
93
|
|
|
$order = 0; |
94
|
|
|
$refs[] = [ |
95
|
|
|
'id' => ++$order, |
96
|
|
|
'label' => "Branch version", |
97
|
|
|
"description" => "Deploy the latest version of a branch", |
98
|
|
|
"list" => $this->getGitBranches($this->project) |
99
|
|
|
]; |
100
|
|
|
|
101
|
|
|
$refs[] = [ |
102
|
|
|
'id' => ++$order, |
103
|
|
|
'label' => "Tag version", |
104
|
|
|
"description" => "Deploy a tagged release", |
105
|
|
|
"list" => $this->getGitTags($this->project) |
106
|
|
|
]; |
107
|
|
|
|
108
|
|
|
// @todo: the original was a tree that was keyed by environment, the |
109
|
|
|
// front-end dropdown needs to be changed to support that. brrrr. |
110
|
|
|
$prevDeploys = []; |
111
|
|
|
foreach($this->getGitPrevDeploys($this->project) as $env) { |
112
|
|
|
foreach($env as $deploy) { |
113
|
|
|
$prevDeploys[] = $deploy; |
114
|
|
|
} |
115
|
|
|
} |
116
|
|
|
$refs[] = [ |
117
|
|
|
'id' => ++$order, |
118
|
|
|
'label' => "Redeploy a release that was previously deployed (to any environment", |
119
|
|
|
"description" => "Deploy a previous release", |
120
|
|
|
"list" => $prevDeploys |
121
|
|
|
]; |
122
|
|
|
|
123
|
|
|
$body = json_encode($refs, JSON_PRETTY_PRINT); |
124
|
|
|
$this->getResponse()->addHeader('Content-Type', 'application/json'); |
125
|
|
|
$this->getResponse()->setBody($body); |
126
|
|
|
return $body; |
127
|
|
|
} |
128
|
|
|
|
129
|
|
|
/** |
130
|
|
|
* Generate the data structure used by the frontend component. |
131
|
|
|
* |
132
|
|
|
* @param string $name of the component |
133
|
|
|
* |
134
|
|
|
* @return array |
135
|
|
|
*/ |
136
|
|
|
public function getModel($name) { |
137
|
|
|
return [ |
138
|
|
|
'APIEndpoint' => Director::absoluteBaseURL().$this->Link() |
139
|
|
|
]; |
140
|
|
|
} |
141
|
|
|
|
142
|
|
|
/** |
143
|
|
|
* @param int $ID |
144
|
|
|
* @return SS_HTTPResponse |
145
|
|
|
*/ |
146
|
|
|
protected function getFetch($ID) { |
147
|
|
|
$ping = DNGitFetch::get()->byID($ID); |
148
|
|
|
if(!$ping) { |
149
|
|
|
return $this->getAPIResponse('Fetch not found', 404); |
150
|
|
|
} |
151
|
|
|
$output = [ |
152
|
|
|
'id' => $ID, |
153
|
|
|
'status' => $ping->ResqueStatus(), |
154
|
|
|
'message' => array_filter(explode(PHP_EOL, $ping->LogContent())) |
155
|
|
|
]; |
156
|
|
|
|
157
|
|
|
return $this->getAPIResponse($output, 200); |
|
|
|
|
158
|
|
|
} |
159
|
|
|
|
160
|
|
|
/** |
161
|
|
|
* @return SS_HTTPResponse |
162
|
|
|
*/ |
163
|
|
View Code Duplication |
protected function createFetch() { |
|
|
|
|
164
|
|
|
/** @var DNGitFetch $fetch */ |
165
|
|
|
$fetch = DNGitFetch::create(); |
166
|
|
|
$fetch->ProjectID = $this->project->ID; |
167
|
|
|
$fetch->write(); |
168
|
|
|
$fetch->start(); |
169
|
|
|
|
170
|
|
|
$location = Director::absoluteBaseURL() . $this->Link() . '/gitupdate/' . $fetch->ID; |
171
|
|
|
$output = array( |
172
|
|
|
'message' => 'Fetch queued as job ' . $fetch->ResqueToken, |
173
|
|
|
'href' => $location, |
174
|
|
|
); |
175
|
|
|
|
176
|
|
|
$response = $this->getAPIResponse($output, 201); |
|
|
|
|
177
|
|
|
$response->addHeader('Location', $location); |
178
|
|
|
return $response; |
179
|
|
|
} |
180
|
|
|
|
181
|
|
|
/** |
182
|
|
|
* @param SS_HTTPRequest $request |
183
|
|
|
* |
184
|
|
|
* @return SS_HTTPResponse |
185
|
|
|
*/ |
186
|
|
|
public function deploysummary(SS_HTTPRequest $request) { |
187
|
|
|
if(!trim($request->getBody())) { |
188
|
|
|
return $this->getAPIResponse('no body was sent in the request', 400); |
189
|
|
|
} |
190
|
|
|
|
191
|
|
|
$jsonBody = json_decode($request->getBody(), true); |
192
|
|
|
if(empty($jsonBody)) { |
193
|
|
|
return $this->getAPIResponse('request did not contain a parsable JSON payload', 400); |
194
|
|
|
} |
195
|
|
|
|
196
|
|
|
$options = [ |
197
|
|
|
'sha' => $jsonBody['sha'] |
198
|
|
|
]; |
199
|
|
|
|
200
|
|
|
$strategy = $this->environment->Backend()->planDeploy($this->environment, $options); |
201
|
|
|
$data = $strategy->toArray(); |
202
|
|
|
|
203
|
|
|
$interface = $this->project->getRepositoryInterface(); |
204
|
|
|
if($this->canCompareCodeVersions($interface, $data['changes']['Code version'])) { |
|
|
|
|
205
|
|
|
$compareurl = sprintf( |
206
|
|
|
'%s/compare/%s...%s', |
207
|
|
|
$interface->URL, |
208
|
|
|
$data['changes']['Code version']['from'], |
209
|
|
|
$data['changes']['Code version']['to'] |
210
|
|
|
); |
211
|
|
|
$data['changes']['Code version']['compareUrl'] = $compareurl; |
212
|
|
|
} |
213
|
|
|
|
214
|
|
|
// Append json to response |
215
|
|
|
$token = SecurityToken::inst(); |
216
|
|
|
$data['SecurityID'] = $token->getValue(); |
217
|
|
|
|
218
|
|
|
$this->extend('updateDeploySummary', $data); |
219
|
|
|
|
220
|
|
|
return json_encode($data); |
221
|
|
|
} |
222
|
|
|
|
223
|
|
|
/** |
224
|
|
|
* @param $project |
225
|
|
|
* |
226
|
|
|
* @return array |
227
|
|
|
*/ |
228
|
|
|
protected function getGitBranches($project) { |
229
|
|
|
$branches = []; |
230
|
|
|
foreach($project->DNBranchList() as $branch) { |
231
|
|
|
$branches[] = [ |
232
|
|
|
'key' => $branch->SHA(), |
233
|
|
|
'value' => $branch->Name(), |
234
|
|
|
]; |
235
|
|
|
} |
236
|
|
|
return $branches; |
237
|
|
|
} |
238
|
|
|
|
239
|
|
|
/** |
240
|
|
|
* @param $project |
241
|
|
|
* |
242
|
|
|
* @return array |
243
|
|
|
*/ |
244
|
|
|
protected function getGitTags($project) { |
245
|
|
|
$tags = []; |
246
|
|
|
foreach($project->DNTagList()->setLimit(null) as $tag) { |
247
|
|
|
$tags[] = [ |
248
|
|
|
'key' => $tag->SHA(), |
249
|
|
|
'value' => $tag->Name(), |
250
|
|
|
]; |
251
|
|
|
} |
252
|
|
|
return $tags; |
253
|
|
|
} |
254
|
|
|
|
255
|
|
|
/** |
256
|
|
|
* @param $project |
257
|
|
|
* |
258
|
|
|
* @return array |
259
|
|
|
*/ |
260
|
|
|
protected function getGitPrevDeploys($project) { |
261
|
|
|
$redeploy = []; |
262
|
|
View Code Duplication |
foreach($project->DNEnvironmentList() as $dnEnvironment) { |
|
|
|
|
263
|
|
|
$envName = $dnEnvironment->Name; |
264
|
|
|
$perEnvDeploys = []; |
265
|
|
|
foreach($dnEnvironment->DeployHistory() as $deploy) { |
266
|
|
|
$sha = $deploy->SHA; |
267
|
|
|
|
268
|
|
|
// Check if exists to make sure the newest deployment date is used. |
269
|
|
|
if(!isset($perEnvDeploys[$sha])) { |
270
|
|
|
$pastValue = sprintf( |
271
|
|
|
"%s (deployed %s)", |
272
|
|
|
substr($sha, 0, 8), |
273
|
|
|
$deploy->obj('LastEdited')->Ago() |
274
|
|
|
); |
275
|
|
|
$perEnvDeploys[$sha] = [ |
276
|
|
|
'key' => $sha, |
277
|
|
|
'value' => $pastValue |
278
|
|
|
]; |
279
|
|
|
} |
280
|
|
|
} |
281
|
|
|
if(!empty($perEnvDeploys)) { |
282
|
|
|
$redeploy[$envName] = array_values($perEnvDeploys); |
283
|
|
|
} |
284
|
|
|
} |
285
|
|
|
return $redeploy; |
286
|
|
|
} |
287
|
|
|
|
288
|
|
|
/** |
289
|
|
|
* Return a simple response with a message |
290
|
|
|
* |
291
|
|
|
* @param string $message |
292
|
|
|
* @param int $statusCode |
293
|
|
|
* @return SS_HTTPResponse |
294
|
|
|
*/ |
295
|
|
|
protected function getAPIResponse($message, $statusCode) { |
296
|
|
|
$output = [ |
297
|
|
|
'message' => $message, |
298
|
|
|
'status_code' => $statusCode |
299
|
|
|
]; |
300
|
|
|
$body = json_encode($output, JSON_PRETTY_PRINT); |
301
|
|
|
$response = $this->getResponse(); |
302
|
|
|
$response->addHeader('Content-Type', 'application/json'); |
303
|
|
|
$response->setBody($body); |
304
|
|
|
$response->setStatusCode($statusCode); |
305
|
|
|
return $response; |
306
|
|
|
} |
307
|
|
|
|
308
|
|
|
/** |
309
|
|
|
* @param ArrayData $interface |
310
|
|
|
* @param array $codeVersion |
311
|
|
|
* |
312
|
|
|
* @return bool |
313
|
|
|
*/ |
314
|
|
|
protected function canCompareCodeVersions(\ArrayData $interface, $codeVersion) { |
315
|
|
|
if(empty($interface)) { |
316
|
|
|
return false; |
317
|
|
|
} |
318
|
|
|
if(empty($interface->URL)) { |
319
|
|
|
return false; |
320
|
|
|
} |
321
|
|
|
if(empty($codeVersion['from']) || empty($codeVersion['to'])) { |
322
|
|
|
return false; |
323
|
|
|
} |
324
|
|
|
if(strlen($codeVersion['from']) !== 40 || strlen($codeVersion['to']) !== 40) { |
325
|
|
|
return false; |
326
|
|
|
} |
327
|
|
|
return true; |
328
|
|
|
} |
329
|
|
|
} |
330
|
|
|
|
It seems like the type of the argument is not accepted by the function/method which you are calling.
In some cases, in particular if PHP’s automatic type-juggling kicks in this might be fine. In other cases, however this might be a bug.
We suggest to add an explicit type cast like in the following example: