it.cnr.icar.eric.server.cms.CanonicalXMLCatalogingService.java Source code

Java tutorial

Introduction

Here is the source code for it.cnr.icar.eric.server.cms.CanonicalXMLCatalogingService.java

Source

/*
 * ====================================================================
 * This file is part of the ebXML Registry by Icar Cnr v3.2 
 * ("eRICv32" in the following disclaimer).
 *
 * "eRICv32" is free software: you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *
 * "eRICv32" is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License Version 3
 * along with "eRICv32".  If not, see <http://www.gnu.org/licenses/>.
 *
 * eRICv32 is a forked, derivative work, based on:
 *    - freebXML Registry, a royalty-free, open source implementation of the ebXML Registry standard,
 *      which was published under the "freebxml License, Version 1.1";
 *   - ebXML OMAR v3.2 Edition, published under the GNU GPL v3 by S. Krushe & P. Arwanitis.
 * 
 * All derivative software changes and additions are made under
 *
 * Copyright (C) 2013 Ing. Antonio Messina <messina@pa.icar.cnr.it>
 *
 * This software consists of voluntary contributions made by many
 * individuals on behalf of the freebxml Software Foundation.  For more
 * information on the freebxml Software Foundation, please see
 * "http://www.freebxml.org/".
 *
 * This product includes software developed by the Apache Software
 * Foundation (http://www.apache.org/).
 *
 * ====================================================================
 */
package it.cnr.icar.eric.server.cms;

import it.cnr.icar.eric.common.RepositoryItem;
import it.cnr.icar.eric.server.common.ServerRequestContext;
import it.cnr.icar.eric.server.repository.RepositoryManagerFactory;
import it.cnr.icar.eric.server.util.ServerResourceBundle;

import java.io.BufferedReader;
import java.io.File;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.Reader;

import javax.xml.transform.ErrorListener;
import javax.xml.transform.Transformer;
import javax.xml.transform.TransformerException;
import javax.xml.transform.TransformerFactory;
import javax.xml.transform.stream.StreamResult;
import javax.xml.transform.stream.StreamSource;

import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;

import javax.xml.bind.JAXBElement;
import javax.xml.registry.RegistryException;
import javax.xml.transform.Source;
import javax.xml.transform.TransformerConfigurationException;
import javax.xml.transform.URIResolver;
import org.oasis.ebxml.registry.bindings.rim.ExtrinsicObjectType;
import org.oasis.ebxml.registry.bindings.rim.RegistryObjectListType;
import org.oasis.ebxml.registry.bindings.rim.ServiceType;
import org.oasis.ebxml.registry.bindings.rim.UserType;

/**
 * Canonical XML Content Cataloging Service
 * 
 * @author Tony Graham
 * 
 */
public class CanonicalXMLCatalogingService extends AbstractContentCatalogingService {
    private static final Log log = LogFactory.getLog(CanonicalXMLCatalogingService.class.getName());

    public ServiceOutput invoke(ServerRequestContext context, ServiceInput input, ServiceType service,
            InvocationController invocationController, UserType user) throws RegistryException {
        if (log.isTraceEnabled()) {
            log.trace("CanonicalXMLCatalogingService.invoke()");
        }
        ServiceOutput so = new ServiceOutput();
        so.setOutput(context);
        RepositoryItem repositoryItem = input.getRepositoryItem();
        // The RI is optional per the [ebRS] spec. Return empty ServiceOutput.
        if (repositoryItem != null) {
            @SuppressWarnings("unused")
            String roId = input.getRegistryObject().getId();

            ServerRequestContext outputContext = null;

            try {
                outputContext = context; // new RequestContext(null);
                StreamSource inputSrc = getAsStreamSource((ExtrinsicObjectType) input.getRegistryObject());

                StreamSource invocationControlFileSrc = rm.getAsStreamSource(invocationController.getEoId());

                // dumpStream(invocationControlFileSrc);

                TransformerFactory tFactory = TransformerFactory.newInstance();
                Transformer transformer = initTransformer(tFactory, invocationControlFileSrc);
                // Use CatalogingService URIResolver to resolve RIs submitted in
                // the
                // ServiceInput object
                transformer.setURIResolver(getURIResolver(context));
                // Set respository item as parameter
                transformer.setParameter("repositoryItem", input.getRegistryObject().getId());

                // Create the output file with catalogedMetadata
                File outputFile = File.createTempFile("CanonicalXMLCatalogingService_OutputFile", ".xml");
                outputFile.deleteOnExit();

                log.debug("Tempfile= " + outputFile.getAbsolutePath());

                StreamResult sr = new StreamResult(outputFile);
                transformer.transform(inputSrc, sr);

                @SuppressWarnings("unchecked")
                JAXBElement<RegistryObjectListType> ebRegistryObjectList = (JAXBElement<RegistryObjectListType>) bu
                        .getJAXBContext().createUnmarshaller().unmarshal(outputFile);
                // take ComplexType from Element
                RegistryObjectListType catalogedMetadata = ebRegistryObjectList.getValue();

                //            RegistryObjectListType catalogedMetadata = (RegistryObjectListType) bu.getJAXBContext()
                //                  .createUnmarshaller().unmarshal(outputFile);

                // TODO: User should refer to "Service object for the
                // Content Management Service that generated the
                // Cataloged Content."
                outputContext.setUser(user);

                bu.getObjectRefsAndRegistryObjects(catalogedMetadata,
                        outputContext.getTopLevelRegistryObjectTypeMap(), outputContext.getObjectRefTypeMap());
            } catch (Exception e) {
                if (outputContext != context) {
                    outputContext.rollback();
                }
                throw new RegistryException(e);
            }

            so.setOutput(outputContext);

            // Setting this error list is redundant, but Content Validation
            // Services
            // currently output a Boolean and a RegistryErrorList, so using
            // same mechanism to report errors from Content Cataloging Services.
            so.setErrorList(outputContext.getErrorList());

            if (outputContext != context) {
                outputContext.commit();
            }
        }
        return so;
    }

    /*
     * This method is used to create and initialize a Transformer instance
     */
    private Transformer initTransformer(TransformerFactory tFactory, StreamSource invocationControlFileSrc)
            throws TransformerConfigurationException {
        Transformer transformer = tFactory.newTransformer(invocationControlFileSrc);
        transformer.setErrorListener(new ErrorListener() {
            public void error(TransformerException exception) throws TransformerException {
                log.info(exception);
            }

            public void fatalError(TransformerException exception) throws TransformerException {
                log.error(exception);
                throw exception;
            }

            public void warning(TransformerException exception) throws TransformerException {
                log.info(exception);
            }
        });
        return transformer;
    }

    protected void dumpStream(StreamSource ss) {
        BufferedReader br = null;

        try {
            InputStream inputStream = ss.getInputStream();
            Reader inputReader = ss.getReader();

            if (inputStream != null) {
                br = new BufferedReader(new InputStreamReader(inputStream));
            } else if (inputReader != null) {
                br = new BufferedReader(inputReader);
            }

            if (br == null) {
                System.err.println("No reader for StreamSource");

                return;
            }

            String line;

            while ((line = br.readLine()) != null) {
                System.err.print(line);
            }

            System.err.println();
        } catch (Exception e) {
            br = null;
        }
    }

    /**
     * A URIResolver for RepositoryItem objects
     * 
     */
    class CatalogingServiceURIResolver implements URIResolver {
        private ServerRequestContext context;

        protected CatalogingServiceURIResolver(ServerRequestContext context) {
            this.context = context;
        }

        @SuppressWarnings("unused")
        public Source resolve(String href, String base) throws TransformerException {
            Source source = null;
            try {
                // Should this check that href is UUID URN first?
                RepositoryItem ri = (RepositoryItem) context.getRepositoryItemsMap().get(href);
                if (ri == null) {
                    // if RI is not submitted in the ServiceInput class, use
                    // the StreamSource to create a Source instance using the
                    // href as the URI string.
                    try {
                        source = new StreamSource(href);
                        if (source == null) {
                            source = getSourceFromRepositoryManager(href);
                        }
                    } catch (TransformerException te) {
                        throw te;
                    } catch (Throwable ex) {
                        // if RI not found in the Map and a StreamSource could
                        // not resolve the URI, use the RepositoryManager
                        source = getSourceFromRepositoryManager(href);
                    }
                } else {
                    source = new StreamSource(ri.getDataHandler().getInputStream());
                }
            } catch (TransformerException ex) {
                throw ex;
            } catch (Throwable t) {
                throw new TransformerException(t);
            }
            return source;
        }
    }

    private Source getSourceFromRepositoryManager(String href) throws TransformerException {
        Source source = null;
        try {
            source = RepositoryManagerFactory.getInstance().getRepositoryManager().getAsStreamSource(href);
            if (source == null) {
                // RM could not resolve the URI throw exception
                throw new TransformerException(ServerResourceBundle.getInstance()
                        .getString("message.error.couldNotResolveURI", new Object[] { href }));
            }
        } catch (Throwable t) {
            throw new TransformerException(t);
        }
        return source;
    }

    /**
     * Gets a <code>URIResolver</code> that handles locating repository items
     * 
     * @return an <code>URIResolver</code> value
     */
    public URIResolver getURIResolver(ServerRequestContext context) {
        return new CatalogingServiceURIResolver(context);
    }

}