Minior Change - Licence Statement
[so.git] / mso-api-handlers / mso-api-handler-infra / src / main / java / org / onap / so / apihandlerinfra / GlobalHealthcheckHandler.java
1 /*-
2  * ============LICENSE_START=======================================================
3  * ONAP - SO
4  * ================================================================================
5  * Copyright (C) 2017 AT&T Intellectual Property. All rights reserved.
6  * ================================================================================
7  * Modifications Copyright (c) 2019 Samsung
8  * ================================================================================
9  * Licensed under the Apache License, Version 2.0 (the "License");
10  * you may not use this file except in compliance with the License.
11  * You may obtain a copy of the License at
12  * 
13  *      http://www.apache.org/licenses/LICENSE-2.0
14  * 
15  * Unless required by applicable law or agreed to in writing, software
16  * distributed under the License is distributed on an "AS IS" BASIS,
17  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
18  * See the License for the specific language governing permissions and
19  * limitations under the License.
20  * ============LICENSE_END=========================================================
21  */
22
23 package org.onap.so.apihandlerinfra;
24
25
26 import java.net.URI;
27 import java.util.Collections;
28
29 import org.slf4j.Logger;
30 import org.slf4j.LoggerFactory;
31 import org.springframework.http.MediaType;
32 import org.springframework.http.ResponseEntity;
33
34 import javax.annotation.PostConstruct;
35 import javax.transaction.Transactional;
36 import javax.ws.rs.DefaultValue;
37 import javax.ws.rs.GET;
38 import javax.ws.rs.Path;
39 import javax.ws.rs.Produces;
40 import javax.ws.rs.QueryParam;
41 import javax.ws.rs.container.ContainerRequestContext;
42 import javax.ws.rs.core.Context;
43 import javax.ws.rs.core.Response;
44 import javax.ws.rs.core.UriBuilder;
45
46 import org.apache.http.HttpStatus;
47 import org.onap.so.logger.MessageEnum;
48 import org.onap.so.logger.MsoLogger;
49 import org.springframework.beans.factory.annotation.Autowired;
50 import org.springframework.core.env.Environment;
51 import org.springframework.http.HttpEntity;
52 import org.springframework.http.HttpHeaders;
53 import org.springframework.stereotype.Component;
54 import org.springframework.web.client.RestTemplate;
55 import org.springframework.http.HttpMethod;
56 import io.swagger.annotations.Api;
57 import io.swagger.annotations.ApiOperation;
58
59
60 @Component
61 @Path("/globalhealthcheck")
62 @Api(value="/globalhealthcheck",description="APIH Infra Global Health Check")
63 public class GlobalHealthcheckHandler {
64         private static Logger logger = LoggerFactory.getLogger(GlobalHealthcheckHandler.class);
65         private static final String CONTEXTPATH_PROPERTY = "management.context-path";
66         private static final String PROPERTY_DOMAIN = "mso.health.endpoints";
67         private static final String CATALOGDB_PROPERTY = PROPERTY_DOMAIN + ".catalogdb";
68         private static final String REQUESTDB_PROPERTY = PROPERTY_DOMAIN + ".requestdb";
69         private static final String SDNC_PROPERTY = PROPERTY_DOMAIN + ".sdnc";
70         private static final String OPENSTACK_PROPERTY = PROPERTY_DOMAIN + ".openstack";
71         private static final String BPMN_PROPERTY = PROPERTY_DOMAIN + ".bpmn";
72         private static final String ASDC_PROPERTY = PROPERTY_DOMAIN + ".asdc";
73         private static final String REQUESTDBATTSVC_PROPERTY = PROPERTY_DOMAIN + ".requestdbattsvc";
74         private static final String DEFAULT_PROPERTY_VALUE = "";
75         
76     // e.g. /manage
77     private String actuatorContextPath;
78         private String endpointCatalogdb;
79         private String endpointRequestdb;
80         private String endpointSdnc;
81         private String endpointOpenstack;
82         private String endpointBpmn;
83         private String endpointAsdc;
84         private String endpointRequestdbAttsvc;
85         
86         @Autowired
87         private Environment env;
88
89         @Autowired
90         private RestTemplate restTemplate;
91         private final String health = "/health";
92
93         
94         @PostConstruct
95         protected void init() {
96                 actuatorContextPath  = env.getProperty(CONTEXTPATH_PROPERTY, String.class, DEFAULT_PROPERTY_VALUE);
97                 endpointCatalogdb  = env.getProperty(CATALOGDB_PROPERTY, String.class, DEFAULT_PROPERTY_VALUE);
98                 endpointRequestdb  = env.getProperty(REQUESTDB_PROPERTY, String.class, DEFAULT_PROPERTY_VALUE);
99                 endpointSdnc  = env.getProperty(SDNC_PROPERTY, String.class, DEFAULT_PROPERTY_VALUE);           
100                 endpointOpenstack  = env.getProperty(OPENSTACK_PROPERTY, String.class, DEFAULT_PROPERTY_VALUE);         
101                 endpointBpmn  = env.getProperty(BPMN_PROPERTY, String.class, DEFAULT_PROPERTY_VALUE);           
102                 endpointAsdc  = env.getProperty(ASDC_PROPERTY, String.class, DEFAULT_PROPERTY_VALUE);           
103                 endpointRequestdbAttsvc  = env.getProperty(REQUESTDBATTSVC_PROPERTY, String.class, DEFAULT_PROPERTY_VALUE);             
104         }
105         
106     @GET
107     @Produces("application/json")
108         @ApiOperation(value="Performing global health check",response=Response.class)
109     @Transactional
110     public Response globalHealthcheck (@DefaultValue("true") @QueryParam("enableBpmn") boolean enableBpmn, 
111                                                                         @Context ContainerRequestContext requestContext) {
112         Response HEALTH_CHECK_RESPONSE = null;
113         // Build internal response object
114         HealthcheckResponse rsp = new HealthcheckResponse();
115         
116         try{
117             long startTime = System.currentTimeMillis ();
118             MsoLogger.setServiceName ("GlobalHealthcheck");
119             // Generated RequestId
120             String requestId = requestContext.getProperty("requestId").toString();
121             MsoLogger.setLogContext(requestId, null);
122             logger.info("{} {}", MessageEnum.APIH_GENERATED_REQUEST_ID.toString(), requestId);
123             
124             // set APIH status, this is the main entry point
125             rsp.setApih(HealthcheckStatus.UP.toString());
126             // set BPMN
127             rsp.setBpmn(querySubsystemHealth(MsoSubsystems.BPMN));
128             // set SDNCAdapter
129             rsp.setSdncAdapter(querySubsystemHealth(MsoSubsystems.SDNC));        
130             // set ASDCController
131             rsp.setAsdcController(querySubsystemHealth(MsoSubsystems.ASDC));        
132             // set CatalogDbAdapter
133             rsp.setCatalogdbAdapter(querySubsystemHealth(MsoSubsystems.CATALOGDB));    
134             // set RequestDbAdapter
135             rsp.setRequestdbAdapter(querySubsystemHealth(MsoSubsystems.REQUESTDB));        
136             // set OpenStackAdapter
137             rsp.setOpenstackAdapter(querySubsystemHealth(MsoSubsystems.OPENSTACK));        
138             // set RequestDbAdapterAttSvc
139             rsp.setRequestdbAdapterAttsvc(querySubsystemHealth(MsoSubsystems.REQUESTDBATT));
140             // set Message
141             rsp.setMessage(String.format("HttpStatus: %s", HttpStatus.SC_OK));
142             logger.info(rsp.toString());
143
144             HEALTH_CHECK_RESPONSE = Response.status (HttpStatus.SC_OK)
145                     .entity (rsp)
146                     .build ();
147             
148         }catch (Exception ex){
149                 logger.error("Exception occurred", ex);
150                 rsp.setMessage(ex.getMessage());
151             HEALTH_CHECK_RESPONSE = Response.status (HttpStatus.SC_INTERNAL_SERVER_ERROR)
152                     .entity (rsp)
153                     .build ();
154         }
155         
156         return HEALTH_CHECK_RESPONSE;
157     }
158     
159     protected HttpEntity<String> buildHttpEntityForRequest(){
160         HttpHeaders headers = new HttpHeaders();
161         headers.setAccept(Collections.singletonList(MediaType.APPLICATION_JSON)); 
162         headers.set("Content-Type", "application/json");
163         HttpEntity<String> entity = new HttpEntity<>("parameters", headers);       
164         return entity;
165     }
166     
167     protected String querySubsystemHealth(MsoSubsystems subsystem){
168         try{
169             // get port number for the subsystem
170                 String ept = getEndpointUrlForSubsystemEnum(subsystem);
171                 
172                 // build final endpoint url
173                         UriBuilder builder = UriBuilder.fromPath(ept).path(actuatorContextPath).path(health);
174                         URI uri = builder.build();
175                 logger.info("Calculated URL: {}", uri.toString());
176             
177             ResponseEntity<SubsystemHealthcheckResponse> result = 
178                         restTemplate.exchange(uri, HttpMethod.GET, buildHttpEntityForRequest(), SubsystemHealthcheckResponse.class);
179             
180                 return processResponseFromSubsystem(result,subsystem);
181                 
182         }catch(Exception ex){
183                         logger.error("Exception occured in GlobalHealthcheckHandler.querySubsystemHealth() ", ex);
184                 return HealthcheckStatus.DOWN.toString();
185         }
186     }
187         protected String processResponseFromSubsystem(ResponseEntity<SubsystemHealthcheckResponse> result, MsoSubsystems subsystem){
188         if(result == null || result.getStatusCodeValue() != HttpStatus.SC_OK){
189                 logger.error(String.format("Globalhealthcheck: checking subsystem: %s failed ! result object is: %s",
190                                 subsystem,
191                                 result == null? "NULL": result));
192                 return HealthcheckStatus.DOWN.toString();
193         }
194         
195         SubsystemHealthcheckResponse body = result.getBody();
196
197                 String status = body.getStatus();
198                 if("UP".equalsIgnoreCase(status)){
199                         return HealthcheckStatus.UP.toString();
200                 }else{
201                         logger.error("{}, query health endpoint did not return UP status!", subsystem);
202                         return HealthcheckStatus.DOWN.toString();
203                 }
204         }
205     
206     
207     protected String getEndpointUrlForSubsystemEnum(MsoSubsystems subsystem){
208         switch (subsystem){
209         case SDNC:
210                 return this.endpointSdnc;
211         case ASDC:
212                 return this.endpointAsdc;
213         case BPMN:
214                 return this.endpointBpmn;
215         case CATALOGDB:
216                 return this.endpointCatalogdb;
217         case OPENSTACK:
218                 return this.endpointOpenstack;
219         case REQUESTDB:
220                 return this.endpointRequestdb;
221         case REQUESTDBATT:
222                 return this.endpointRequestdbAttsvc;
223         default:
224                 return "";
225         }       
226     }
227 }