package org.gcube.informationsystem.icproxy.resources; import static org.gcube.resources.discovery.icclient.ICFactory.client; import static org.gcube.resources.discovery.icclient.ICFactory.clientFor; import static org.gcube.resources.discovery.icclient.ICFactory.queryFor; import java.util.ArrayList; import java.util.List; import java.util.Objects; import javax.validation.constraints.NotNull; import javax.ws.rs.*; import javax.ws.rs.core.MediaType; import lombok.extern.slf4j.Slf4j; import org.gcube.common.authorization.library.provider.AccessTokenProvider; import org.gcube.common.keycloak.model.ModelUtils; import org.gcube.common.resources.gcore.*; import org.gcube.common.resources.gcore.utils.Group; import org.gcube.common.scope.api.ScopeProvider; import org.gcube.resources.discovery.client.api.DiscoveryClient; import org.gcube.resources.discovery.client.queries.api.SimpleQuery; import org.gcube.common.encryption.StringEncrypter; @Slf4j @Path("ServiceEndpoint") public class ServiceEndpointResource { @GET @Path("/{category}") @Produces(MediaType.APPLICATION_XML) public List retrieve(@NotNull @PathParam("category") String resourceCategory) { log.info("ServiceEndpoint called with category {} in context {}",resourceCategory, ScopeProvider.instance.get()); DiscoveryClient client = clientFor(ServiceEndpoint.class); List endpoints = client.submit(getQuery(resourceCategory)); log.debug("retrieved resources are "+endpoints.size()); return endpoints; } @GET @Path("/{category}/{name}") @Produces(MediaType.APPLICATION_XML) public List retrieve(@NotNull @PathParam("name") String resourceName, @NotNull @PathParam("category") String resourceCategory, @QueryParam("decrypt") boolean isDecrypt) { log.info("ServiceEndpoint called with category {} and name {} in scope {}",resourceCategory, resourceName, ScopeProvider.instance.get()); DiscoveryClient client = clientFor(ServiceEndpoint.class); List endpoints = client.submit(getQuery(resourceName, resourceCategory)); if(Objects.nonNull(endpoints)) { log.debug("retrieved resources are "+endpoints.size()); return getServiceEndpoints(isDecrypt, endpoints); }else{ log.error("ServiceEndpoint called with category {} and name {} in scope {}, return NULL",resourceCategory, resourceName, ScopeProvider.instance.get()); return null; } } @GET @Path("/{category}/{name}/Result/{result:([^$\\?]+)}") @Produces(MediaType.TEXT_XML) public String retrieveCustom(@NotNull @PathParam("name") String resourceName, @NotNull @PathParam("category") String resourceCategory, @NotNull @PathParam("result") String resultXPath) { log.info("ServiceEndpoint called with category {} and name {} and result {} in scope {}" ,resourceCategory, resourceName, resultXPath, ScopeProvider.instance.get()); SimpleQuery query = getQuery(resourceName, resourceCategory); if (resultXPath.startsWith("/")) query.setResult("$resource"+resultXPath); else query.setResult("$resource/"+resultXPath); DiscoveryClient client = client(); List endpoints = client.submit(query); StringBuilder builder = new StringBuilder(""); for (String single: endpoints) builder.append("").append(single.replaceAll("\n", "")).append(""); builder.append(""); log.debug("retrieved resources are "+endpoints.size()); return builder.toString(); } private SimpleQuery getQuery(String resourceName, String resourceCategory){ SimpleQuery query = queryFor(ServiceEndpoint.class); query.addCondition(String.format("$resource/Profile/Name/text() eq '%s'",resourceName)); query.addCondition(String.format("$resource/Profile/Category/text() eq '%s'",resourceCategory)); return query; } private SimpleQuery getQuery(String resourceCategory){ SimpleQuery query = queryFor(ServiceEndpoint.class); query.addCondition(String.format("$resource/Profile/Category/text() eq '%s'",resourceCategory)); return query; } private ServiceEndpoint decryptResource(ServiceEndpoint resource) { Group aps=resource.profile().accessPoints(); for (ServiceEndpoint.AccessPoint ap : aps){ String decrypted =decryptString(ap.password()); String user= ap.username(); ap.credentials(decrypted, user); } return resource; } private static String decryptString(String toDecrypt){ try{ return StringEncrypter.getEncrypter().decrypt(toDecrypt); }catch(Exception e) { throw new RuntimeException("Unable to decrypt : "+toDecrypt,e); } } private List getServiceEndpoints(boolean isDecrypt, List endpoints) { if (isDecrypt) { if (isRoleEnabled()){ List ses = new ArrayList<>(endpoints.size()); for (ServiceEndpoint resource : endpoints) { ses.add(decryptResource(resource)); } return ses; }else{ log.warn("user not enabled to see the resource free to air, sorry"); } } return endpoints; } private boolean isRoleEnabled(){ String at= AccessTokenProvider.instance.get(); try{ if (ModelUtils.getAccessTokenFrom(at).getRealmAccess().getRoles().contains("service-endpoint-key" )) { log.info("The client is authorized to see the resource as 'free-to-air'"); return true; } }catch (Exception e){ log.error("token not retrieved properly: "+e.getMessage()); e.printStackTrace(); } log.info("user not authorized, sorry"); return false; } }