package org.gcube.portlets.user.uriresolvermanager.readers; import static org.gcube.resources.discovery.icclient.ICFactory.client; import java.io.StringReader; import java.util.HashMap; import java.util.List; import java.util.Map; import javax.xml.parsers.DocumentBuilder; import javax.xml.parsers.DocumentBuilderFactory; import org.gcube.common.resources.gcore.utils.XPathHelper; import org.gcube.common.scope.api.ScopeProvider; import org.gcube.portlets.user.uriresolvermanager.util.ScopeUtil; import org.gcube.resources.discovery.client.api.DiscoveryClient; import org.gcube.resources.discovery.client.queries.api.Query; import org.gcube.resources.discovery.client.queries.impl.QueryBox; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.w3c.dom.Node; import org.xml.sax.InputSource; /** * * @author Francesco Mangiacrapa francesco.mangiacrapa@isti.cnr.it * @Oct 13, 2014 * */ public class UriResolverMapReader { /** * */ private static final String URIRESOLVERMAP = "UriResolverMap"; /** * */ private static final String URI_RESOLVER_MAP = "Uri-Resolver-Map"; private Logger logger = LoggerFactory.getLogger(UriResolverMapReader.class); private String secondaryType; private String scope; private String resourceName; private Map applicationTypes; //A map ApplicationType - Resource Name /** * * @param scope - the scope to be searched * @throws Exception */ public UriResolverMapReader(String scope) throws Exception { this.scope = scope; if(scope==null){ throw new Exception("Scope is null, set scope provider!"); } this.resourceName = URI_RESOLVER_MAP; this.secondaryType = URIRESOLVERMAP; readProfileFromInfrastrucure(); } /** * this method looks up the generic resource among the ones available in the infrastructure using scope provider {@link ScopeProvider.instance.get()} * resource name {@value #URI_RESOLVER_MAP} and secondaryType {@value #URIRESOLVERMAP} * @param portletClassName your servlet class name will be used ad unique identifier for your applicationProfile * @return the applicationProfile profile * @throws Exception */ private void readProfileFromInfrastrucure() throws Exception { String queryString = getGcubeGenericQueryString(secondaryType, resourceName); logger.info("Trying to fetch applicationProfile profile from the infrastructure for " + secondaryType + " scope: " + scope); logger.info(queryString); try { String infra = ScopeUtil.getInfrastructureNameFromScope(this.scope); ScopeProvider.instance.set(infra); logger.info("scope provider set instance: "+infra); Query q = new QueryBox(queryString); DiscoveryClient client = client(); List appUriResolverMap = client.submit(q); if (appUriResolverMap == null || appUriResolverMap.size() == 0) throw new ApplicationProfileException("Your applicationProfile with secondaryType: "+secondaryType+" and name "+resourceName+" is not registered in the infrastructure"); else { String elem = appUriResolverMap.get(0); DocumentBuilder docBuilder = DocumentBuilderFactory.newInstance().newDocumentBuilder(); Node node = docBuilder.parse(new InputSource(new StringReader(elem))).getDocumentElement(); XPathHelper helper = new XPathHelper(node); List currValue = null; currValue = helper.evaluate("/Resource/Profile/Body/access_point/application_type/text()"); if (currValue != null && currValue.size() > 0) { logger.info("Application Types are: "+currValue.size()); applicationTypes = new HashMap(currValue.size()); // List appTypes = currValue; //FOR EACH APPLICATION TYPE for (String at : currValue) { logger.info("Application Type "+at); // currValue = helper.evaluate("/Resource/Profile/Body/EndPoint[Scope='"+scope.toString()+"']/Scope/text()"); List resources = helper.evaluate("/Resource/Profile/Body/access_point[application_type='"+at+"']/resource/text()"); if(resources!=null && resources.size()>0){ applicationTypes.put(at, resources.get(0)); logger.info("Stored: "+at +" -> Resource: "+ resources.get(0)); }else logger.warn("Skipping Type "+at+" generic resource not found!"); } } } } catch (Exception e) { logger.error("Error while trying to fetch applicationProfile with secondaryType: "+secondaryType+" and name "+resourceName+" from the infrastructure", e); throw new ApplicationProfileException("Error while trying to fetch applicationProfile with secondaryType: "+secondaryType+" and name "+resourceName+" from the infrastructure"); } } public static String getGcubeGenericQueryString(String secondaryType, String name){ return "for $profile in collection('/db/Profiles/GenericResource')//Resource " + "where $profile/Profile/SecondaryType/string() eq '"+secondaryType+"' and $profile/Profile/Name/string() " + " eq '" + name + "'" + "return $profile"; } public String getSecondaryType() { return secondaryType; } public String getScope() { return scope; } public Map getApplicationTypes() { return applicationTypes; } public String getResourceName() { return resourceName; } @Override public String toString() { StringBuilder builder = new StringBuilder(); builder.append("UriResolverMapReader [secondaryType="); builder.append(secondaryType); builder.append(", scope="); builder.append(scope); builder.append(", resourceName="); builder.append(resourceName); builder.append(", applicationTypes="); builder.append(applicationTypes); builder.append("]"); return builder.toString(); } /*public static void main(String[] args) { String scope ="/gcube"; try { UriResolverMapReader resolver = new UriResolverMapReader(scope); } catch (Exception e) { // TODO Auto-generated catch block e.printStackTrace(); } }*/ }