Thursday, February 4, 2016

Customize ATG Web Service to include Security

Customize ATG Web Service to include Security
ATG by default do not provide to include SOAP header security in Webservices. Thus we need to customize ATG SOAP webservice to read header. Also it is not possible to read headers from request in our custom webservice component. Thus we need to customize below components and Read the Headers from Web Service calls.
·         Override WebserviceGenerator component, create a custom class and Override method addWSDLImportFilter to add out custom filter.
public class CustomWebserviceGenerator extends WebServiceGeneratorImpl {
           
           
           
            protected void addWSDLImportFilter(String pWebAppFactoryName,
                                    WebApp pWebApp, String pServletName) throws JAXBException {
                       
                       
                        List filters = pWebApp.getFilter();

                        boolean found = false;
                        Iterator filterIter = filters.iterator();
                        while (filterIter.hasNext()) {
                                    FilterType filter = (FilterType) filterIter.next();
                                    FilterClassType filterClass = filter.getFilterClass();
                                    if (filterClass.getValue()
                                                            .equals("com.custom.filter. CustomWSDLImportFilter")) {
                                                found = true;
                                    }
                        }

                        if (!(found)) {
                                    appendFilterToWebAppDD(pWebAppFactoryName, pWebApp,
                                                            "com.custom.filter.CustomWSDLImportFilter", " customWSDLImportFilter");
                        }

                        String mapping = "/" + pServletName + "/*";
                        appendFilterMappingToWebAppDD(pWebAppFactoryName, pWebApp,
                                                "CustomWSDLImportFilter", mapping);
            }
}
·         Re-Create WebService EAR file. Now the web.xml in the custom module will include our custom filter.
·         Write CustomWSDLImportFilter class to read header with the help of MultiReadHttpServletRequest
public class CustomWSDLImportFilter extends WSDLImportFilter {
            public CustomWSDLImportFilter() {
                        super();
            }
           
            public void doFilter(ServletRequest paramServletRequest,
                                    ServletResponse paramServletResponse, FilterChain paramFilterChain)
                                    throws IOException, ServletException {

                       
                        RepositoryItem userProfileItem = null;
                        SOAPMessageContext messageContext = new SOAPMessageContext();

                        MimeHeaders headers = getHeaders((HttpServletRequest) paramServletRequest);
                        MultiReadHttpServletRequest multiReadRequest = new MultiReadHttpServletRequest(
                                                (HttpServletRequest) paramServletRequest);

                        try {

                                    javax.naming.InitialContext ic = new javax.naming.InitialContext();
                                    ProfileTools profileTools = (atg.userprofiling.ProfileTools) ic
                                                            .lookup(EatonConstants.WS_JNDI_PROFILE_TOOLS);
                                    if (profileTools.isWebServiceHeaderRequired()) {
                                                SOAPMessage message = getSOAPMessageFromRequest(
                                                                        (HttpServletRequest) multiReadRequest, headers,
                                                                        messageContext);
                                                if (message.getSOAPHeader() != null) {
                                                            Iterator iter = message.getSOAPHeader()
                                                                                    .extractAllHeaderElements();

                                                            while (iter.hasNext()) {
                                                                        SOAPHeaderElement headerEl = (SOAPHeaderElement) iter
                                                                                                .next();
                                                                        if (headerEl.getElementsByTagName(
                                                                                                EatonConstants.WS_USERNAME).item(0) != null) {
                                                                                    userName = headerEl
                                                                                                            .getElementsByTagName(
                                                                                                                                    EatonConstants.WS_USERNAME).item(0)
                                                                                                            .getTextContent();
                                                                        }
                                                                        if (headerEl.getElementsByTagName(
                                                                                                EatonConstants.WS_PASSWORD).item(0) != null) {
                                                                                    password = headerEl
                                                                                                            .getElementsByTagName(
                                                                                                                                    EatonConstants.WS_PASSWORD).item(0)
                                                                                                            .getTextContent();
                                                                        }
                                                            }

                                                            System.out.println(“username : ” + userName);
System.out.println(“password: ” + password);
                                                } else {
                                                            throw new SecurityException(
                                                                                    EatonConstants.WS_HEADER_MISSING);
                                                }
                                    }
                        } catch (SOAPException soapExeception) {
                                    // return;

                        } catch (NamingException NamingException) {
                                    return;

                        }

                        super.doFilter(multiReadRequest, paramServletResponse, paramFilterChain);
            }

            protected static MimeHeaders getHeaders(HttpServletRequest req) {
                        Enumeration enums = req.getHeaderNames();
                        MimeHeaders headers = new MimeHeaders();

                        while (enums.hasMoreElements()) {
                                    String headerName = (String) enums.nextElement();
                                    String headerValue = req.getHeader(headerName);
                                    headers.addHeader(headerName, headerValue);
                        }

                        return headers;
            }

            protected SOAPMessage getSOAPMessageFromRequest(HttpServletRequest request,
                                    MimeHeaders headers, SOAPMessageContext messageContext)
                                    throws IOException {
                        SOAPMessage message = null;

                        InputStream is = request.getInputStream();

                        byte[] bytes = readFully(is);

                        int length = (request.getContentLength() == -1) ? bytes.length
                                                : request.getContentLength();

                        ByteInputStream in = new ByteInputStream(bytes, length);
                        message = messageContext.createMessage(headers, in);

                        return message;
            }

            protected static byte[] readFully(InputStream istream) throws IOException {
                        ByteArrayOutputStream bout = new ByteArrayOutputStream();
                        byte[] buf = new byte[1024];
                        int num = 0;

                        while ((istream != null) && ((num = istream.read(buf)) != -1)) {
                                    bout.write(buf, 0, num);
                        }

                        byte[] ret = bout.toByteArray();
                        return ret;
            }

}


·         MultiReadHttpServletRequest implementation to read headers.
public class MultiReadHttpServletRequest extends HttpServletRequestWrapper {
              private ByteArrayOutputStream cachedBytes;

              public MultiReadHttpServletRequest(HttpServletRequest request) {
                super(request);
              }

              @Override
              public ServletInputStream getInputStream() throws IOException {
                if (cachedBytes == null)
                  cacheInputStream();

                  return new CachedServletInputStream();
              }

              @Override
              public BufferedReader getReader() throws IOException{
                return new BufferedReader(new InputStreamReader(getInputStream()));
              }

              private void cacheInputStream() throws IOException {
                /* Cache the inputstream in order to read it multiple times. For
                 * convenience, I use apache.commons IOUtils
                 */
                cachedBytes = new ByteArrayOutputStream();
               
                IOUtils.copy(super.getInputStream(), cachedBytes);
              }

              /* An inputstream which reads the cached request body */
              public class CachedServletInputStream extends ServletInputStream {
                private ByteArrayInputStream input;

                public CachedServletInputStream() {
                  /* create a new input stream from the cached request body */
                  input = new ByteArrayInputStream(cachedBytes.toByteArray());
                }

                @Override
                public int read() throws IOException {
                  return input.read();
                }
              }
            }


No comments:

Post a Comment