5
5
6
6
package com .aws .greengrass .cli ;
7
7
8
+ import com .aws .greengrass .authorization .AuthorizationHandler ;
9
+ import com .aws .greengrass .authorization .Permission ;
10
+ import com .aws .greengrass .authorization .exceptions .AuthorizationException ;
8
11
import com .aws .greengrass .componentmanager .ComponentStore ;
9
12
import com .aws .greengrass .config .Node ;
10
13
import com .aws .greengrass .config .Topics ;
96
99
import static com .aws .greengrass .ipc .common .ExceptionUtil .translateExceptions ;
97
100
import static com .aws .greengrass .ipc .common .IPCErrorStrings .DEPLOYMENTS_QUEUE_FULL ;
98
101
import static com .aws .greengrass .ipc .common .IPCErrorStrings .DEPLOYMENTS_QUEUE_NOT_INITIALIZED ;
102
+ import static software .amazon .awssdk .aws .greengrass .GreengrassCoreIPCServiceModel .CREATE_DEBUG_PASSWORD ;
103
+ import static software .amazon .awssdk .aws .greengrass .GreengrassCoreIPCServiceModel .CREATE_LOCAL_DEPLOYMENT ;
104
+ import static software .amazon .awssdk .aws .greengrass .GreengrassCoreIPCServiceModel .GET_COMPONENT_DETAILS ;
105
+ import static software .amazon .awssdk .aws .greengrass .GreengrassCoreIPCServiceModel .GET_LOCAL_DEPLOYMENT_STATUS ;
106
+ import static software .amazon .awssdk .aws .greengrass .GreengrassCoreIPCServiceModel .LIST_COMPONENTS ;
107
+ import static software .amazon .awssdk .aws .greengrass .GreengrassCoreIPCServiceModel .LIST_LOCAL_DEPLOYMENTS ;
108
+ import static software .amazon .awssdk .aws .greengrass .GreengrassCoreIPCServiceModel .RESTART_COMPONENT ;
109
+ import static software .amazon .awssdk .aws .greengrass .GreengrassCoreIPCServiceModel .STOP_COMPONENT ;
99
110
100
111
@ SuppressWarnings ("PMD.CouplingBetweenObjects" )
101
112
public class CLIEventStreamAgent {
@@ -119,6 +130,9 @@ public class CLIEventStreamAgent {
119
130
@ Setter (AccessLevel .PACKAGE )
120
131
private DeploymentQueue deploymentQueue ;
121
132
133
+ @ Inject
134
+ private AuthorizationHandler authHandler ;
135
+
122
136
private final SecureRandom random = new SecureRandom ();
123
137
124
138
public GetComponentDetailsHandler getGetComponentDetailsHandler (OperationContinuationHandlerContext context ) {
@@ -190,8 +204,12 @@ protected void onStreamClosed() {
190
204
@ SuppressWarnings ("PMD.PreserveStackTrace" )
191
205
public GetComponentDetailsResponse handleRequest (GetComponentDetailsRequest request ) {
192
206
return translateExceptions (() -> {
193
- authorizeRequest (componentName );
194
207
validateGetComponentDetailsRequest (request );
208
+ authorizeRequest (Permission .builder ()
209
+ .principal (componentName )
210
+ .resource (request .getComponentName ())
211
+ .operation (GET_COMPONENT_DETAILS )
212
+ .build ());
195
213
String componentName = request .getComponentName ();
196
214
GreengrassService service ;
197
215
try {
@@ -219,11 +237,23 @@ private void validateGetComponentDetailsRequest(GetComponentDetailsRequest reque
219
237
}
220
238
}
221
239
222
- private void authorizeRequest (String componentName ) {
223
- if (Utils .isEmpty (componentName ) || !componentName .startsWith (GREENGRASS_CLI_CLIENT_ID_PREFIX ) && !CLI_SERVICE
224
- .equals (componentName )) {
240
+ private void authorizeRequest (Permission permission ) {
241
+ if (Utils .isEmpty (permission .getPrincipal ())) {
225
242
throw new UnauthorizedError ("Component is not authorized to call CLI APIs" );
226
243
}
244
+ // Allow CLI and CLI clients to call anything
245
+ if (permission .getPrincipal ().startsWith (GREENGRASS_CLI_CLIENT_ID_PREFIX )
246
+ || CLI_SERVICE .equals (permission .getPrincipal ())) {
247
+ return ;
248
+ }
249
+ // Check for authorization for all other components
250
+ try {
251
+ authHandler .isAuthorized (CLI_SERVICE , permission );
252
+ } catch (AuthorizationException e ) {
253
+ logger .atWarn ().kv ("error" , e .getMessage ()).kv ("componentName" , permission .getPrincipal ())
254
+ .log ("Not Authorized" );
255
+ throw new UnauthorizedError (e .getMessage ());
256
+ }
227
257
}
228
258
229
259
private ComponentDetails getComponentDetails (GreengrassService service ) {
@@ -259,7 +289,11 @@ protected void onStreamClosed() {
259
289
@ Override
260
290
public ListComponentsResponse handleRequest (ListComponentsRequest request ) {
261
291
return translateExceptions (() -> {
262
- authorizeRequest (componentName );
292
+ authorizeRequest (Permission .builder ()
293
+ .principal (componentName )
294
+ .resource (AuthorizationHandler .ANY_REGEX )
295
+ .operation (LIST_COMPONENTS )
296
+ .build ());
263
297
Collection <GreengrassService > services = kernel .orderedDependencies ();
264
298
List <ComponentDetails > listOfComponents =
265
299
services .stream ().filter (service -> service != kernel .getMain ())
@@ -295,8 +329,12 @@ protected void onStreamClosed() {
295
329
@ SuppressWarnings ("PMD.PreserveStackTrace" )
296
330
public RestartComponentResponse handleRequest (RestartComponentRequest request ) {
297
331
return translateExceptions (() -> {
298
- authorizeRequest (componentName );
299
332
validateRestartComponentRequest (request );
333
+ authorizeRequest (Permission .builder ()
334
+ .principal (componentName )
335
+ .resource (request .getComponentName ())
336
+ .operation (RESTART_COMPONENT )
337
+ .build ());
300
338
String componentName = request .getComponentName ();
301
339
RestartComponentResponse response = new RestartComponentResponse ();
302
340
response .setRestartStatus (RequestStatus .SUCCEEDED );
@@ -346,8 +384,12 @@ protected void onStreamClosed() {
346
384
@ SuppressWarnings ("PMD.PreserveStackTrace" )
347
385
public StopComponentResponse handleRequest (StopComponentRequest request ) {
348
386
return translateExceptions (() -> {
349
- authorizeRequest (componentName );
350
387
validateStopComponentRequest (request );
388
+ authorizeRequest (Permission .builder ()
389
+ .principal (componentName )
390
+ .resource (request .getComponentName ())
391
+ .operation (STOP_COMPONENT )
392
+ .build ());
351
393
String componentName = request .getComponentName ();
352
394
try {
353
395
GreengrassService service = kernel .locate (componentName );
@@ -396,7 +438,11 @@ protected void onStreamClosed() {
396
438
@ SuppressWarnings ({"PMD.PreserveStackTrace" , "PMD.AvoidCatchingGenericException" })
397
439
public CreateLocalDeploymentResponse handleRequest (CreateLocalDeploymentRequest request ) {
398
440
return translateExceptions (() -> {
399
- authorizeRequest (componentName );
441
+ authorizeRequest (Permission .builder ()
442
+ .principal (componentName )
443
+ .resource (AuthorizationHandler .ANY_REGEX )
444
+ .operation (CREATE_LOCAL_DEPLOYMENT )
445
+ .build ());
400
446
String deploymentId = UUID .randomUUID ().toString ();
401
447
//All inputs are valid. If all inputs are empty, then user might just want to retrigger the deployment
402
448
// with new recipes set using the updateRecipesAndArtifacts API.
@@ -492,8 +538,12 @@ protected void onStreamClosed() {
492
538
@ Override
493
539
public GetLocalDeploymentStatusResponse handleRequest (GetLocalDeploymentStatusRequest request ) {
494
540
return translateExceptions (() -> {
495
- authorizeRequest (componentName );
496
541
validateGetLocalDeploymentStatusRequest (request );
542
+ authorizeRequest (Permission .builder ()
543
+ .principal (componentName )
544
+ .resource (request .getDeploymentId ())
545
+ .operation (GET_LOCAL_DEPLOYMENT_STATUS )
546
+ .build ());
497
547
Topics localDeployments = cliServiceConfig .findTopics (PERSISTENT_LOCAL_DEPLOYMENTS );
498
548
if (localDeployments == null || localDeployments .findTopics (request .getDeploymentId ()) == null ) {
499
549
ResourceNotFoundError rnf = new ResourceNotFoundError ();
@@ -558,7 +608,11 @@ public List<Node> sortDeploymentsByTime(Iterator<Node> deploymentsIterator, Comp
558
608
@ Override
559
609
public ListLocalDeploymentsResponse handleRequest (ListLocalDeploymentsRequest request ) {
560
610
return translateExceptions (() -> {
561
- authorizeRequest (componentName );
611
+ authorizeRequest (Permission .builder ()
612
+ .principal (componentName )
613
+ .resource (AuthorizationHandler .ANY_REGEX )
614
+ .operation (LIST_LOCAL_DEPLOYMENTS )
615
+ .build ());
562
616
List <LocalDeployment > persistedDeployments = new ArrayList <>();
563
617
Topics localDeployments = cliServiceConfig .findTopics (PERSISTENT_LOCAL_DEPLOYMENTS );
564
618
List <Topics > deploymentsByTime = sortDeploymentsByTime (localDeployments .iterator (),
@@ -616,7 +670,11 @@ protected void onStreamClosed() {
616
670
@ Override
617
671
public CreateDebugPasswordResponse handleRequest (CreateDebugPasswordRequest request ) {
618
672
return translateExceptions (() -> {
619
- authorizeRequest (componentName );
673
+ authorizeRequest (Permission .builder ()
674
+ .principal (componentName )
675
+ .resource (AuthorizationHandler .ANY_REGEX )
676
+ .operation (CREATE_DEBUG_PASSWORD )
677
+ .build ());
620
678
CreateDebugPasswordResponse response = new CreateDebugPasswordResponse ();
621
679
622
680
String password = generatePassword (DEBUG_PASSWORD_LENGTH_REQUIREMENT );
0 commit comments