All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.mobicents.as7.deployment.SipAnnotationDeploymentProcessor Maven / Gradle / Ivy

There is a newer version: 4.0.12
Show newest version
/*
 * TeleStax, Open Source Cloud Communications
 * Copyright 2011-2014, Telestax Inc and individual contributors
 * by the @authors tag.
 *
 * This program is free software: you can redistribute it and/or modify
 * under the terms of the GNU Affero General Public License as
 * published by the Free Software Foundation; either version 3 of
 * the License, or (at your option) any later version.
 *
 * This program 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 Affero General Public License for more details.
 *
 * You should have received a copy of the GNU Affero General Public License
 * along with this program.  If not, see 
 */

package org.mobicents.as7.deployment;

import java.lang.reflect.Modifier;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import java.util.Map.Entry;

import javax.servlet.sip.SipServletRequest;
import javax.servlet.sip.annotation.SipApplication;
import javax.servlet.sip.annotation.SipApplicationKey;
import javax.servlet.sip.annotation.SipListener;
import javax.servlet.sip.annotation.SipServlet;
import org.mobicents.servlet.sip.annotation.ConcurrencyControl;
import org.mobicents.servlet.sip.annotation.ConcurrencyControlMode;

import org.jboss.annotation.javaee.Descriptions;
import org.jboss.annotation.javaee.DisplayNames;
import org.jboss.annotation.javaee.Icons;
import org.jboss.as.ee.structure.DeploymentType;
import org.jboss.as.ee.structure.DeploymentTypeMarker;
import org.jboss.as.server.deployment.Attachments;
import org.jboss.as.server.deployment.DeploymentPhaseContext;
import org.jboss.as.server.deployment.DeploymentUnit;
import org.jboss.as.server.deployment.DeploymentUnitProcessingException;
import org.jboss.as.server.deployment.DeploymentUnitProcessor;
import org.jboss.as.server.deployment.annotation.AnnotationIndexUtils;
import org.jboss.as.server.deployment.module.ResourceRoot;
import org.jboss.jandex.AnnotationInstance;
import org.jboss.jandex.AnnotationTarget;
import org.jboss.jandex.AnnotationValue;
import org.jboss.jandex.ClassInfo;
import org.jboss.jandex.DotName;
import org.jboss.jandex.Index;
import org.jboss.jandex.MethodInfo;
import org.jboss.jandex.Type;
import org.jboss.logging.Logger;
import org.jboss.metadata.javaee.spec.DescriptionGroupMetaData;
import org.jboss.metadata.javaee.spec.DescriptionImpl;
import org.jboss.metadata.javaee.spec.DescriptionsImpl;
import org.jboss.metadata.javaee.spec.DisplayNameImpl;
import org.jboss.metadata.javaee.spec.DisplayNamesImpl;
import org.jboss.metadata.javaee.spec.IconImpl;
import org.jboss.metadata.javaee.spec.IconsImpl;
import org.jboss.metadata.web.spec.ListenerMetaData;
import org.jboss.metadata.web.spec.ServletMetaData;
import org.jboss.metadata.web.spec.SessionConfigMetaData;
import org.jboss.vfs.VirtualFile;
import org.mobicents.metadata.sip.spec.ProxyConfigMetaData;
import org.mobicents.metadata.sip.spec.Sip11MetaData;
import org.mobicents.metadata.sip.spec.SipAnnotationMetaData;
import org.mobicents.metadata.sip.spec.SipApplicationKeyMethodInfo;
import org.mobicents.metadata.sip.spec.SipMetaData;
import org.mobicents.metadata.sip.spec.SipServletSelectionMetaData;
import org.mobicents.metadata.sip.spec.SipServletsMetaData;

/**
 * Web annotation deployment processor.
 *
 * @author Emanuel Muckenhuber
 * @author Remy Maucherat
 * @author [email protected]
 */
public class SipAnnotationDeploymentProcessor implements DeploymentUnitProcessor {

    private static final Logger logger = Logger.getLogger(SipAnnotationDeploymentProcessor.class);

//    // @RunAs
//    addTypeProcessor(new RunAsProcessor(finder));
//    // @DeclareRoles
//    addTypeProcessor(new DeclareRolesProcessor(finder));
//    // @SipServlet
//    addTypeProcessor(new SipServletProcessor(finder));
//    // @SipListener
//    addTypeProcessor(new SipListenerProcessor(finder));
//    // @ConcurrencyControl
//    addTypeProcessor(new ConcurrencyControlProcessor(finder));
//    // @SipApplicationKey
//    addMethodProcessor(new SipApplicationKeyProcessor(finder));

//    protected static final DotName webFilter = DotName.createSimple(WebFilter.class.getName());
    protected static final DotName sipListener = DotName.createSimple(SipListener.class.getName());
    protected static final DotName sipServlet = DotName.createSimple(SipServlet.class.getName());
    protected static final DotName sipApplication = DotName.createSimple(SipApplication.class.getName());
    protected static final DotName sipApplicationKey = DotName.createSimple(SipApplicationKey.class.getName());
    protected static final DotName concurrencyControl = DotName.createSimple(ConcurrencyControl.class.getName());
//    protected static final DotName webServlet = DotName.createSimple(WebServlet.class.getName());
//    protected static final DotName runAs = DotName.createSimple(RunAs.class.getName());
//    protected static final DotName declareRoles = DotName.createSimple(DeclareRoles.class.getName());
//    protected static final DotName multipartConfig = DotName.createSimple(MultipartConfig.class.getName());
//    protected static final DotName servletSecurity = DotName.createSimple(ServletSecurity.class.getName());

    /**
     * Process web annotations.
     */
    @Override
    public void deploy(final DeploymentPhaseContext phaseContext) throws DeploymentUnitProcessingException {
        final DeploymentUnit deploymentUnit = phaseContext.getDeploymentUnit();
        // Commented for http://code.google.com/p/sipservlets/issues/detail?id=168
    	// When no sip.xml but annotations only, Application is not recognized as SIP App by AS7
//        SipMetaData sipMetaData = deploymentUnit.getAttachment(SipMetaData.ATTACHMENT_KEY);
//        if (sipMetaData == null) {
//            return;
//        }
        SipAnnotationMetaData sipAnnotationsMetaData = deploymentUnit.getAttachment(SipAnnotationMetaData.ATTACHMENT_KEY);
        if (sipAnnotationsMetaData == null) {
            sipAnnotationsMetaData = new SipAnnotationMetaData();
            deploymentUnit.putAttachment(SipAnnotationMetaData.ATTACHMENT_KEY, sipAnnotationsMetaData);
        }
        Map indexes = AnnotationIndexUtils.getAnnotationIndexes(deploymentUnit);

        // Process components
        for (final Entry entry : indexes.entrySet()) {
            if (logger.isDebugEnabled()) logger.debug("doDeploy(): processing annotations from " + entry.getKey().getRootName());
            final Index jarIndex = entry.getValue();
            SipMetaData sipMetaData = processAnnotations(sipAnnotationsMetaData, jarIndex);
            if(sipMetaData != null) {
            	// https://github.com/Mobicents/sip-servlets/issues/68 Adding only the ones that actually have annotations 
            	// to save memory and computing time later on
            	sipAnnotationsMetaData.put(entry.getKey().getRootName(), sipMetaData);
            }
        }
    }

    /**
     * Process a single index.
     *
     * @param index the annotation index
     * @throws DeploymentUnitProcessingException
     */
    protected SipMetaData processAnnotations(SipAnnotationMetaData sipAnotationsMetaData, Index index) throws DeploymentUnitProcessingException {
        Sip11MetaData sipMetaData = null;
        // @SipListener
        final List sipListenerAnnotations = index.getAnnotations(sipListener);
        if (sipListenerAnnotations != null && sipListenerAnnotations.size() > 0) {
        	sipMetaData = new Sip11MetaData();
            List listeners = new ArrayList();
            for (final AnnotationInstance annotation : sipListenerAnnotations) {
                if (logger.isDebugEnabled()) logger.debug("processAnnotations(): @SipListener: " + annotation);
                ListenerMetaData listener = new ListenerMetaData();
                AnnotationTarget target = annotation.target();
                if (!(target instanceof ClassInfo)) {
                    throw new DeploymentUnitProcessingException("@SipListener is only allowed at class level " + target);
                }
                ClassInfo classInfo = ClassInfo.class.cast(target);
                listener.setListenerClass(classInfo.toString());
                AnnotationValue descriptionValue = annotation.value();
                if (descriptionValue != null) {
                    DescriptionGroupMetaData descriptionGroup = getDescriptionGroup(descriptionValue.asString());
                    if (descriptionGroup != null) {
                        listener.setDescriptionGroup(descriptionGroup);
                    }
                }
                for (AnnotationValue value: annotation.values()) {
                    if (value.name().compareTo("name") == 0) {
                        // listener name not supported
                        //listener.setName(value.asString());
                    }
                    else if (value.name().compareTo("applicationName") == 0) {
                        if (sipMetaData.getApplicationName() == null) {
                            sipMetaData.setApplicationName(value.asString());
                        }
                        else if ((sipMetaData.getApplicationName() != null) && (sipMetaData.getApplicationName().compareTo(value.asString()) != 0)) {
                            throw (new DeploymentUnitProcessingException("Sip Application Name mismatch: already loaded: " + sipMetaData.getApplicationName() + " - from @SipListener annotation (" + listener.getListenerClass() + "): " + value.asString()));
                        }
                    }
                }
                if (logger.isDebugEnabled()) logger.debug("processAnnotations(): sipListener added " + listener);
                listeners.add(listener);
            }
            if (logger.isDebugEnabled()) logger.debug("processAnnotations(): " + listeners.size() + " sipListeners added");
            sipMetaData.setListeners(listeners);
        }

        // @SipServlet
        final List sipServletAnnotations = index.getAnnotations(sipServlet);
        if (sipServletAnnotations != null && sipServletAnnotations.size() > 0) {
        	if(sipMetaData == null) {
        		sipMetaData = new Sip11MetaData();
        	}
            SipServletsMetaData sipServlets = new SipServletsMetaData();
            // @SipApplication
            final List sipApplicationAnnotations = index.getAnnotations(sipApplication);
            boolean sipApplicationPresent = false;
            String parsedAnnotatedPackage = null;
            if (sipApplicationAnnotations != null && sipApplicationAnnotations.size() > 0) {
                sipApplicationPresent = true;
            }
            for (final AnnotationInstance annotation : sipServletAnnotations) {
                if (logger.isDebugEnabled()) logger.debug("processAnnotations(): @SipServlet: " + annotation);
                ServletMetaData servlet = new ServletMetaData();
                AnnotationTarget target = annotation.target();
                if (!(target instanceof ClassInfo)) {
                    throw new DeploymentUnitProcessingException("@SipServlet is only allowed at class level " + target);
                }
                ClassInfo classInfo = ClassInfo.class.cast(target);
                servlet.setServletClass(classInfo.toString());
                AnnotationValue descriptionValue = annotation.value();
                if (descriptionValue != null) {
                    DescriptionGroupMetaData descriptionGroup = getDescriptionGroup(descriptionValue.asString());
                    if (descriptionGroup != null) {
                        servlet.setDescriptionGroup(descriptionGroup);
                    }
                }
                for (AnnotationValue value: annotation.values()) {
                    if (value.name().compareTo("name") == 0) {
                        servlet.setName(value.asString());
                    }
                    else if (value.name().compareTo("loadOnStartup") == 0) {
                        servlet.setLoadOnStartup(value.asString());
                    }
                    else if (value.name().compareTo("applicationName") == 0) {
                        if (sipMetaData.getApplicationName() == null) {
                            sipMetaData.setApplicationName(value.asString());
                        }
                        else if ((sipMetaData.getApplicationName() != null) && (sipMetaData.getApplicationName().compareTo(value.asString()) != 0)) {
                            throw (new DeploymentUnitProcessingException("Sip Application Name mismatch: already loaded: " + sipMetaData.getApplicationName() + " - from @SipServlet annotation (" + servlet.getServletClass() + "): " + value.asString()));
                        }
                    }
                }
                if (servlet.getName() == null) {
                    // must have a name
                    servlet.setName(servlet.getServletClass());
                }
                if (sipApplicationPresent) {
                	sipAnotationsMetaData.setSipApplicationAnnotationPresent(true);
                    AnnotationInstance sipAppAnnotation = sipApplicationAnnotations.get(0);
                    if (logger.isDebugEnabled()) logger.debug("processAnnotations(): @SipAnnotation: " + annotation);
                    AnnotationTarget sipAppTarget = sipAppAnnotation.target();
                    if (!(target instanceof ClassInfo)) {
                        throw new DeploymentUnitProcessingException("@SipAnnotation is only allowed at class level " + target);
                    }
                    ClassInfo sipAppClassInfo = ClassInfo.class.cast(sipAppTarget);
                    String packageName = sipAppClassInfo.toString().substring(0,sipAppClassInfo.toString().lastIndexOf('.'));
                    if (parsedAnnotatedPackage != null && !parsedAnnotatedPackage.equals(packageName)) {
                        throw new DeploymentUnitProcessingException("Cant have two different applications in a single context - " + packageName + " and " + parsedAnnotatedPackage);
                    }
                    if (parsedAnnotatedPackage == null) {
                        parsedAnnotatedPackage = packageName;
                        parseSipApplication(sipMetaData,sipAppAnnotation, packageName);
                    }
                }
                if (logger.isDebugEnabled()) logger.debug("processAnnotations(): sipServlet added " + servlet);
                sipServlets.add(servlet);
            }
            if (logger.isDebugEnabled()) logger.debug("processAnnotations(): " + sipServlets.size() + " sipServlets added");
            sipMetaData.setSipServlets(sipServlets);
        }

        //@SipApplicationKey
        final List sipApplicationKeyAnnotations = index.getAnnotations(sipApplicationKey);
        if (sipApplicationKeyAnnotations != null && sipApplicationKeyAnnotations.size() > 0) {
        	if(sipMetaData == null) {
        		sipMetaData = new Sip11MetaData();
        	}
            for (final AnnotationInstance annotation : sipApplicationKeyAnnotations) {
                if (logger.isDebugEnabled()) logger.debug("processAnnotations(): @SipApplicationKey: " + annotation);
                final AnnotationTarget target = annotation.target();
                if (!(target instanceof MethodInfo)) {
                    throw new DeploymentUnitProcessingException("@SipApplicationKey is only allowed at method level " + target);
                }
                final MethodInfo methodInfo = MethodInfo.class.cast(target);
                if(!Modifier.isStatic(methodInfo.flags()) || !Modifier.isPublic(methodInfo.flags())) {
                	throw new DeploymentUnitProcessingException("A method annotated with the @SipApplicationKey annotation MUST be public and static");
                }
                if(!methodInfo.returnType().name().toString().equals(String.class.getName())) {
                	throw new DeploymentUnitProcessingException("A method annotated with the @SipApplicationKey annotation MUST return a String");
                }
                final Type[] types = methodInfo.args();
                if(types.length != 1 || !types[0].name().toString().equals(SipServletRequest.class.getName())) {
                	throw new DeploymentUnitProcessingException("A method annotated with the @SipApplicationKey annotation MUST have a single argument of type SipServletRequest");
                }
                if(sipMetaData.getSipApplicationKeyMethodInfo() != null && !sipMetaData.getSipApplicationKeyMethodInfo().equals(methodInfo)) {
                    // https://code.google.com/p/sipservlets/issues/detail?id=37
                	throw new DeploymentUnitProcessingException("More than one SipApplicationKey annotated method is not allowed.");
                }
                final SipApplicationKeyMethodInfo sipApplicationKeyMethodInfo = new SipApplicationKeyMethodInfo(methodInfo.declaringClass().name().toString(), methodInfo.name());
                sipMetaData.setSipApplicationKeyMethodInfo(sipApplicationKeyMethodInfo);
            	if (logger.isDebugEnabled()) logger.debug("processAnnotations(): added " + sipApplicationKeyMethodInfo + " as @SipApplicationKey method");
            }
        }
        //@ConcurrencyControl
        final List concurrencyControlAnnotations = index.getAnnotations(concurrencyControl);
        if (concurrencyControlAnnotations != null && concurrencyControlAnnotations.size() > 0) {
        	if(sipMetaData == null) {
        		sipMetaData = new Sip11MetaData();
        	}
            for (final AnnotationInstance annotation : concurrencyControlAnnotations) {
                if (logger.isDebugEnabled()) logger.debug("processAnnotations(): @ConcurrencyControl: " + annotation);
                final AnnotationTarget target = annotation.target();
                if (!(target instanceof ClassInfo)) {
                    throw new DeploymentUnitProcessingException("@ConcurrencyControl is only allowed at package level " + target);
                }
                ConcurrencyControlMode mode = ConcurrencyControlMode.valueOf(annotation.value("mode").asString());
                sipMetaData.setConcurrencyControlMode(mode);
            	if (logger.isDebugEnabled()) logger.debug("processAnnotations(): ConcurrencyControl set to " + mode);
            }
        }
        return sipMetaData;
    }

    protected Descriptions getDescription(String description) {
        DescriptionsImpl descriptions = null;
        if (description.length() > 0) {
            DescriptionImpl di = new DescriptionImpl();
            di.setDescription(description);
            descriptions = new DescriptionsImpl();
            descriptions.add(di);
        }
        return descriptions;
    }

    protected DisplayNames getDisplayName(String displayName) {
        DisplayNamesImpl displayNames = null;
        if (displayName.length() > 0) {
            DisplayNameImpl dn = new DisplayNameImpl();
            dn.setDisplayName(displayName);
            displayNames = new DisplayNamesImpl();
            displayNames.add(dn);
        }
        return displayNames;
    }

    protected Icons getIcons(String smallIcon, String largeIcon) {
        IconsImpl icons = null;
        if (smallIcon.length() > 0 || largeIcon.length() > 0) {
            IconImpl i = new IconImpl();
            i.setSmallIcon(smallIcon);
            i.setLargeIcon(largeIcon);
            icons = new IconsImpl();
            icons.add(i);
        }
        return icons;
    }

    protected DescriptionGroupMetaData getDescriptionGroup(String description) {
        DescriptionGroupMetaData dg = null;
        if (description.length() > 0) {
            dg = new DescriptionGroupMetaData();
            Descriptions descriptions = getDescription(description);
            dg.setDescriptions(descriptions);
        }
        return dg;
    }

    protected DescriptionGroupMetaData getDescriptionGroup(String description, String displayName, String smallIcon,
            String largeIcon) {
        DescriptionGroupMetaData dg = null;
        if (description.length() > 0 || displayName.length() > 0 || smallIcon.length() > 0 || largeIcon.length() > 0) {
            dg = new DescriptionGroupMetaData();
            Descriptions descriptions = getDescription(description);
            if (descriptions != null)
                dg.setDescriptions(descriptions);
            DisplayNames displayNames = getDisplayName(displayName);
            if (displayNames != null)
                dg.setDisplayNames(displayNames);
            Icons icons = getIcons(smallIcon, largeIcon);
            if (icons != null)
                dg.setIcons(icons);
        }
        return dg;
    }

    private static SipApplication getApplicationAnnotation(Package pack) {
        if(pack == null) return null;
        SipApplication sipApp = (SipApplication) pack.getAnnotation(SipApplication.class);
        if(sipApp != null) {
            return sipApp;
        }
        return null;
    }

    private void parseSipApplication(SipMetaData sipMetaData, AnnotationInstance sipAppAnnInstance, String packageName) throws DeploymentUnitProcessingException {
        String description = null;
        String displayName = null;
        String largeIcon = null;
        String smallIcon = null;

        for (AnnotationValue value: sipAppAnnInstance.values()) {
            // application name
            if (value.name().compareTo("name") == 0) {
                if (sipMetaData.getApplicationName() == null) {
                	if (logger.isDebugEnabled()) logger.debug("parseSipApplication(): @SipApplication: " + value.asString());
                    sipMetaData.setApplicationName(value.asString());
                }
                else if ((sipMetaData.getApplicationName() != null) && (sipMetaData.getApplicationName().compareTo(value.asString()) != 0)) {
                    throw (new DeploymentUnitProcessingException("Sip Application Name mismatch! Already defined: " + sipMetaData.getApplicationName() + " - from @SipAnnotation (package " + packageName + "): " + value.asString()));
                }
            }
            // description
            else if (value.name().compareTo("description") == 0) {
                DescriptionImpl di = new DescriptionImpl();
                di.setDescription(value.asString());
                getOrCreateDs(getOrCreateDG(sipMetaData)).add(di);
            }
            // display name
            else if (value.name().compareTo("displayName") == 0) {
                DisplayNameImpl dn = new DisplayNameImpl();
                dn.setDisplayName(value.asString());
                getOrCreateDn(getOrCreateDG(sipMetaData)).add(dn);
            }
            // distributable
            else if (value.name().compareTo("distributable") == 0) {
                throw (new DeploymentUnitProcessingException("Distributable not supported yet"));
                // sipMetaData.setDistributable(new EmptyMetaData());
            }
            // large icon
            else if (value.name().compareTo("largeIcon") == 0) {
                largeIcon = value.asString();
            }
            // small icon
            else if (value.name().compareTo("smallIcon") == 0) {
                smallIcon = value.asString();
            }
            else if (value.name().compareTo("mainServlet") == 0) {
                if (sipMetaData.getServletSelection() == null) {
                    sipMetaData.setServletSelection(new SipServletSelectionMetaData());
                }
                sipMetaData.getServletSelection().setMainServlet(value.asString());
            }
            else if (value.name().compareTo("proxyTimeout") == 0) {
                if(sipMetaData.getProxyConfig() == null) {
                    sipMetaData.setProxyConfig(new ProxyConfigMetaData());
                }
                sipMetaData.getProxyConfig().setProxyTimeout(value.asInt());
            }
            else if (value.name().compareTo("sessionTimeout") == 0) {
                if(sipMetaData.getSessionConfig() == null) {
                    sipMetaData.setSessionConfig(new SessionConfigMetaData());
                }
                sipMetaData.getSessionConfig().setSessionTimeout(value.asInt());
            }
        }
        if (smallIcon != null || largeIcon != null) {
            IconImpl icon = new IconImpl();
            if (smallIcon != null)
                icon.setSmallIcon(smallIcon);
            if (largeIcon != null)
                icon.setLargeIcon(largeIcon);
            getOrCreateIs(getOrCreateDG(sipMetaData)).add(icon);
        }
        if (sipMetaData.getApplicationName() == null) {
        	if (logger.isDebugEnabled()) logger.debug("parseSipApplication(): @SipApplication: " + packageName);
            sipMetaData.setApplicationName(packageName);
        }
    }

    private static DescriptionGroupMetaData getOrCreateDG(SipMetaData sipMetaData) {
        DescriptionGroupMetaData dg = sipMetaData.getDescriptionGroup();
        if (dg == null) {
            dg = new DescriptionGroupMetaData();
            sipMetaData.setDescriptionGroup(dg);
        }
        return dg;
    }

    private static DescriptionsImpl getOrCreateDs(DescriptionGroupMetaData dg) {
        DescriptionsImpl ds = (DescriptionsImpl) dg.getDescriptions();
        if (ds == null) {
            ds = new DescriptionsImpl();
            dg.setDescriptions(ds);
        }
        return ds;
    }

    private static DisplayNamesImpl getOrCreateDn(DescriptionGroupMetaData dg) {
        DisplayNamesImpl dn = (DisplayNamesImpl) dg.getDisplayNames();
        if (dn == null) {
            dn = new DisplayNamesImpl();
            dg.setDisplayNames(dn);
        }
        return dn;
    }

    private static IconsImpl getOrCreateIs(DescriptionGroupMetaData dg) {
        IconsImpl ii = (IconsImpl) dg.getIcons();
        if (ii == null) {
            ii = new IconsImpl();
            dg.setIcons(ii);
        }
        return ii;
    }


    static final String SIP_XML = "WEB-INF/sip.xml";

    protected boolean canHandle(DeploymentUnit deploymentUnit) {
        //return DeploymentTypeMarker.isType(DeploymentType.WAR, deploymentUnit);
        final ResourceRoot deploymentRoot = deploymentUnit.getAttachment(Attachments.DEPLOYMENT_ROOT);
        final VirtualFile sipXml = deploymentRoot.getRoot().getChild(SIP_XML);
        if (DeploymentTypeMarker.isType(DeploymentType.WAR, deploymentUnit) && sipXml.exists()) {
            return true;
        }
        return false;
    }

    @Override
    public void undeploy(DeploymentUnit context) {
        // TODO Auto-generated method stub

    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy