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

com.uber.hoodie.io.storage.HoodieWrapperFileSystem Maven / Gradle / Ivy

There is a newer version: 0.4.7
Show newest version
/*
 * Copyright (c) 2016 Uber Technologies, Inc. ([email protected])
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *          http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package com.uber.hoodie.io.storage;

import org.apache.hadoop.conf.Configuration;
import org.apache.hadoop.fs.*;
import org.apache.hadoop.fs.permission.AclEntry;
import org.apache.hadoop.fs.permission.AclStatus;
import org.apache.hadoop.fs.permission.FsAction;
import org.apache.hadoop.fs.permission.FsPermission;
import org.apache.hadoop.security.AccessControlException;
import org.apache.hadoop.security.Credentials;
import org.apache.hadoop.security.token.Token;
import org.apache.hadoop.util.Progressable;

import java.io.FileNotFoundException;
import java.io.IOException;
import java.net.URI;
import java.net.URISyntaxException;
import java.util.EnumSet;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;

/**
 * HoodieWrapperFileSystem wraps the default file system.
 * It holds state about the open streams in the file system to support getting the
 * written size to each of the open streams.
 */
public class HoodieWrapperFileSystem extends FileSystem {
    private static final Set SUPPORT_SCHEMES;
    public static final String HOODIE_SCHEME_PREFIX = "hoodie-";

    static {
        SUPPORT_SCHEMES = new HashSet<>();
        SUPPORT_SCHEMES.add("file");
        SUPPORT_SCHEMES.add("hdfs");
        SUPPORT_SCHEMES.add("s3");

        // Hoodie currently relies on underlying object store being fully
        // consistent so only regional buckets should be used.
        SUPPORT_SCHEMES.add("gs");
        SUPPORT_SCHEMES.add("viewfs");
    }

    private ConcurrentMap openStreams =
        new ConcurrentHashMap<>();
    private FileSystem fileSystem;
    private URI uri;

    @Override public void initialize(URI uri, Configuration conf) throws IOException {
        // Get the default filesystem to decorate
        fileSystem = FileSystem.get(conf);
        // Do not need to explicitly initialize the default filesystem, its done already in the above FileSystem.get
        // fileSystem.initialize(FileSystem.getDefaultUri(conf), conf);
        // fileSystem.setConf(conf);
        this.uri = uri;
    }

    @Override public URI getUri() {
        return uri;
    }

    @Override public FSDataInputStream open(Path f, int bufferSize) throws IOException {
        return fileSystem.open(convertToDefaultPath(f), bufferSize);
    }

    @Override public FSDataOutputStream create(Path f, FsPermission permission, boolean overwrite,
        int bufferSize, short replication, long blockSize, Progressable progress)
        throws IOException {
        final Path translatedPath = convertToDefaultPath(f);
        return wrapOutputStream(f, fileSystem
            .create(translatedPath, permission, overwrite, bufferSize, replication, blockSize,
                progress));
    }

    private FSDataOutputStream wrapOutputStream(final Path path,
        FSDataOutputStream fsDataOutputStream) throws IOException {
        if (fsDataOutputStream instanceof SizeAwareFSDataOutputStream) {
            return fsDataOutputStream;
        }

        SizeAwareFSDataOutputStream os =
            new SizeAwareFSDataOutputStream(fsDataOutputStream, new Runnable() {
                @Override public void run() {
                    openStreams.remove(path.getName());
                }
            });
        openStreams.put(path.getName(), os);
        return os;
    }

    @Override public FSDataOutputStream create(Path f, boolean overwrite) throws IOException {
        return wrapOutputStream(f, fileSystem.create(convertToDefaultPath(f), overwrite));
    }

    @Override public FSDataOutputStream create(Path f) throws IOException {
        return wrapOutputStream(f, fileSystem.create(convertToDefaultPath(f)));
    }

    @Override public FSDataOutputStream create(Path f, Progressable progress) throws IOException {
        return fileSystem.create(convertToDefaultPath(f), progress);
    }

    @Override public FSDataOutputStream create(Path f, short replication) throws IOException {
        return fileSystem.create(convertToDefaultPath(f), replication);
    }

    @Override public FSDataOutputStream create(Path f, short replication, Progressable progress)
        throws IOException {
        return fileSystem.create(convertToDefaultPath(f), replication, progress);
    }

    @Override public FSDataOutputStream create(Path f, boolean overwrite, int bufferSize)
        throws IOException {
        return fileSystem.create(convertToDefaultPath(f), overwrite, bufferSize);
    }

    @Override public FSDataOutputStream create(Path f, boolean overwrite, int bufferSize,
        Progressable progress) throws IOException {
        return fileSystem.create(convertToDefaultPath(f), overwrite, bufferSize, progress);
    }

    @Override
    public FSDataOutputStream create(Path f, boolean overwrite, int bufferSize, short replication,
        long blockSize, Progressable progress) throws IOException {
        return fileSystem
            .create(convertToDefaultPath(f), overwrite, bufferSize, replication, blockSize,
                progress);
    }

    @Override
    public FSDataOutputStream create(Path f, FsPermission permission, EnumSet flags,
        int bufferSize, short replication, long blockSize, Progressable progress)
        throws IOException {
        return fileSystem
            .create(convertToDefaultPath(f), permission, flags, bufferSize, replication, blockSize,
                progress);
    }

    @Override
    public FSDataOutputStream create(Path f, FsPermission permission, EnumSet flags,
        int bufferSize, short replication, long blockSize, Progressable progress,
        Options.ChecksumOpt checksumOpt) throws IOException {
        return fileSystem
            .create(convertToDefaultPath(f), permission, flags, bufferSize, replication, blockSize,
                progress, checksumOpt);
    }


    @Override
    public FSDataOutputStream create(Path f, boolean overwrite, int bufferSize, short replication,
        long blockSize) throws IOException {
        return fileSystem
            .create(convertToDefaultPath(f), overwrite, bufferSize, replication, blockSize);
    }


    @Override public FSDataOutputStream append(Path f, int bufferSize, Progressable progress)
        throws IOException {
        return fileSystem.append(convertToDefaultPath(f), bufferSize, progress);
    }

    @Override public boolean rename(Path src, Path dst) throws IOException {
        return fileSystem.rename(convertToDefaultPath(src), convertToDefaultPath(dst));
    }

    @Override public boolean delete(Path f, boolean recursive) throws IOException {
        return fileSystem.delete(convertToDefaultPath(f), recursive);
    }

    @Override public FileStatus[] listStatus(Path f) throws FileNotFoundException, IOException {
        return fileSystem.listStatus(convertToDefaultPath(f));
    }

    @Override public void setWorkingDirectory(Path new_dir) {
        fileSystem.setWorkingDirectory(convertToDefaultPath(new_dir));
    }

    @Override public Path getWorkingDirectory() {
        return convertToHoodiePath(fileSystem.getWorkingDirectory());
    }

    @Override public boolean mkdirs(Path f, FsPermission permission) throws IOException {
        return fileSystem.mkdirs(convertToDefaultPath(f), permission);
    }

    @Override public FileStatus getFileStatus(Path f) throws IOException {
        return fileSystem.getFileStatus(convertToDefaultPath(f));
    }

    @Override public String getScheme() {
        return uri.getScheme();
    }

    @Override public String getCanonicalServiceName() {
        return fileSystem.getCanonicalServiceName();
    }

    @Override public String getName() {
        return fileSystem.getName();
    }

    @Override public Path makeQualified(Path path) {
        return convertToHoodiePath(fileSystem.makeQualified(convertToDefaultPath(path)));
    }

    @Override public Token getDelegationToken(String renewer) throws IOException {
        return fileSystem.getDelegationToken(renewer);
    }

    @Override public Token[] addDelegationTokens(String renewer, Credentials credentials)
        throws IOException {
        return fileSystem.addDelegationTokens(renewer, credentials);
    }

    @Override public FileSystem[] getChildFileSystems() {
        return fileSystem.getChildFileSystems();
    }

    @Override public BlockLocation[] getFileBlockLocations(FileStatus file, long start, long len)
        throws IOException {
        return fileSystem.getFileBlockLocations(file, start, len);
    }

    @Override public BlockLocation[] getFileBlockLocations(Path p, long start, long len)
        throws IOException {
        return fileSystem.getFileBlockLocations(convertToDefaultPath(p), start, len);
    }

    @Override public FsServerDefaults getServerDefaults() throws IOException {
        return fileSystem.getServerDefaults();
    }

    @Override public FsServerDefaults getServerDefaults(Path p) throws IOException {
        return fileSystem.getServerDefaults(convertToDefaultPath(p));
    }

    @Override public Path resolvePath(Path p) throws IOException {
        return convertToHoodiePath(fileSystem.resolvePath(convertToDefaultPath(p)));
    }

    @Override public FSDataInputStream open(Path f) throws IOException {
        return fileSystem.open(convertToDefaultPath(f));
    }

    @Override
    public FSDataOutputStream createNonRecursive(Path f, boolean overwrite, int bufferSize,
        short replication, long blockSize, Progressable progress) throws IOException {
        return fileSystem
            .createNonRecursive(convertToDefaultPath(f), overwrite, bufferSize, replication,
                blockSize, progress);
    }

    @Override
    public FSDataOutputStream createNonRecursive(Path f, FsPermission permission, boolean overwrite,
        int bufferSize, short replication, long blockSize, Progressable progress)
        throws IOException {
        return fileSystem
            .createNonRecursive(convertToDefaultPath(f), permission, overwrite, bufferSize,
                replication, blockSize, progress);
    }

    @Override public FSDataOutputStream createNonRecursive(Path f, FsPermission permission,
        EnumSet flags, int bufferSize, short replication, long blockSize,
        Progressable progress) throws IOException {
        return fileSystem
            .createNonRecursive(convertToDefaultPath(f), permission, flags, bufferSize, replication,
                blockSize, progress);
    }

    @Override public boolean createNewFile(Path f) throws IOException {
        return fileSystem.createNewFile(convertToDefaultPath(f));
    }

    @Override public FSDataOutputStream append(Path f) throws IOException {
        return fileSystem.append(convertToDefaultPath(f));
    }

    @Override public FSDataOutputStream append(Path f, int bufferSize) throws IOException {
        return fileSystem.append(convertToDefaultPath(f), bufferSize);
    }

    @Override public void concat(Path trg, Path[] psrcs) throws IOException {
        Path[] psrcsNew = convertDefaults(psrcs);
        fileSystem.concat(convertToDefaultPath(trg), psrcsNew);
    }

    @Override public short getReplication(Path src) throws IOException {
        return fileSystem.getReplication(convertToDefaultPath(src));
    }

    @Override public boolean setReplication(Path src, short replication) throws IOException {
        return fileSystem.setReplication(convertToDefaultPath(src), replication);
    }

    @Override public boolean delete(Path f) throws IOException {
        return fileSystem.delete(convertToDefaultPath(f));
    }

    @Override public boolean deleteOnExit(Path f) throws IOException {
        return fileSystem.deleteOnExit(convertToDefaultPath(f));
    }

    @Override public boolean cancelDeleteOnExit(Path f) {
        return fileSystem.cancelDeleteOnExit(convertToDefaultPath(f));
    }

    @Override public boolean exists(Path f) throws IOException {
        return fileSystem.exists(convertToDefaultPath(f));
    }

    @Override public boolean isDirectory(Path f) throws IOException {
        return fileSystem.isDirectory(convertToDefaultPath(f));
    }

    @Override public boolean isFile(Path f) throws IOException {
        return fileSystem.isFile(convertToDefaultPath(f));
    }

    @Override public long getLength(Path f) throws IOException {
        return fileSystem.getLength(convertToDefaultPath(f));
    }

    @Override public ContentSummary getContentSummary(Path f) throws IOException {
        return fileSystem.getContentSummary(convertToDefaultPath(f));
    }

    @Override public RemoteIterator listCorruptFileBlocks(Path path) throws IOException {
        return fileSystem.listCorruptFileBlocks(convertToDefaultPath(path));
    }

    @Override public FileStatus[] listStatus(Path f, PathFilter filter)
        throws FileNotFoundException, IOException {
        return fileSystem.listStatus(convertToDefaultPath(f), filter);
    }

    @Override public FileStatus[] listStatus(Path[] files)
        throws FileNotFoundException, IOException {
        return fileSystem.listStatus(convertDefaults(files));
    }

    @Override public FileStatus[] listStatus(Path[] files, PathFilter filter)
        throws FileNotFoundException, IOException {
        return fileSystem.listStatus(convertDefaults(files), filter);
    }

    @Override public FileStatus[] globStatus(Path pathPattern) throws IOException {
        return fileSystem.globStatus(convertToDefaultPath(pathPattern));
    }

    @Override public FileStatus[] globStatus(Path pathPattern, PathFilter filter)
        throws IOException {
        return fileSystem.globStatus(convertToDefaultPath(pathPattern), filter);
    }

    @Override public RemoteIterator listLocatedStatus(Path f)
        throws FileNotFoundException, IOException {
        return fileSystem.listLocatedStatus(convertToDefaultPath(f));
    }

    @Override public RemoteIterator listFiles(Path f, boolean recursive)
        throws FileNotFoundException, IOException {
        return fileSystem.listFiles(convertToDefaultPath(f), recursive);
    }

    @Override public Path getHomeDirectory() {
        return convertToHoodiePath(fileSystem.getHomeDirectory());
    }

    @Override public boolean mkdirs(Path f) throws IOException {
        return fileSystem.mkdirs(convertToDefaultPath(f));
    }

    @Override public void copyFromLocalFile(Path src, Path dst) throws IOException {
        fileSystem.copyFromLocalFile(convertToDefaultPath(src), convertToDefaultPath(dst));
    }

    @Override public void moveFromLocalFile(Path[] srcs, Path dst) throws IOException {
        fileSystem.moveFromLocalFile(convertDefaults(srcs), convertToDefaultPath(dst));
    }

    @Override public void moveFromLocalFile(Path src, Path dst) throws IOException {
        fileSystem.moveFromLocalFile(convertToDefaultPath(src), convertToDefaultPath(dst));
    }

    @Override public void copyFromLocalFile(boolean delSrc, Path src, Path dst) throws IOException {
        fileSystem.copyFromLocalFile(delSrc, convertToDefaultPath(src), convertToDefaultPath(dst));
    }

    @Override
    public void copyFromLocalFile(boolean delSrc, boolean overwrite, Path[] srcs, Path dst)
        throws IOException {
        fileSystem
            .copyFromLocalFile(delSrc, overwrite, convertDefaults(srcs), convertToDefaultPath(dst));
    }

    @Override public void copyFromLocalFile(boolean delSrc, boolean overwrite, Path src, Path dst)
        throws IOException {
        fileSystem.copyFromLocalFile(delSrc, overwrite, convertToDefaultPath(src),
            convertToDefaultPath(dst));
    }

    @Override public void copyToLocalFile(Path src, Path dst) throws IOException {
        fileSystem.copyToLocalFile(convertToDefaultPath(src), convertToDefaultPath(dst));
    }

    @Override public void moveToLocalFile(Path src, Path dst) throws IOException {
        fileSystem.moveToLocalFile(convertToDefaultPath(src), convertToDefaultPath(dst));
    }

    @Override public void copyToLocalFile(boolean delSrc, Path src, Path dst) throws IOException {
        fileSystem.copyToLocalFile(delSrc, convertToDefaultPath(src), convertToDefaultPath(dst));
    }

    @Override
    public void copyToLocalFile(boolean delSrc, Path src, Path dst, boolean useRawLocalFileSystem)
        throws IOException {
        fileSystem.copyToLocalFile(delSrc, convertToDefaultPath(src), convertToDefaultPath(dst),
            useRawLocalFileSystem);
    }

    @Override public Path startLocalOutput(Path fsOutputFile, Path tmpLocalFile)
        throws IOException {
        return convertToHoodiePath(fileSystem.startLocalOutput(convertToDefaultPath(fsOutputFile),
            convertToDefaultPath(tmpLocalFile)));
    }

    @Override public void completeLocalOutput(Path fsOutputFile, Path tmpLocalFile)
        throws IOException {
        fileSystem.completeLocalOutput(convertToDefaultPath(fsOutputFile),
            convertToDefaultPath(tmpLocalFile));
    }

    @Override public void close() throws IOException {
        fileSystem.close();
    }

    @Override public long getUsed() throws IOException {
        return fileSystem.getUsed();
    }

    @Override public long getBlockSize(Path f) throws IOException {
        return fileSystem.getBlockSize(convertToDefaultPath(f));
    }

    @Override public long getDefaultBlockSize() {
        return fileSystem.getDefaultBlockSize();
    }

    @Override public long getDefaultBlockSize(Path f) {
        return fileSystem.getDefaultBlockSize(convertToDefaultPath(f));
    }

    @Override public short getDefaultReplication() {
        return fileSystem.getDefaultReplication();
    }

    @Override public short getDefaultReplication(Path path) {
        return fileSystem.getDefaultReplication(convertToDefaultPath(path));
    }

    @Override public void access(Path path, FsAction mode)
        throws AccessControlException, FileNotFoundException, IOException {
        fileSystem.access(convertToDefaultPath(path), mode);
    }

    @Override public void createSymlink(Path target, Path link, boolean createParent)
        throws AccessControlException, FileAlreadyExistsException, FileNotFoundException,
        ParentNotDirectoryException, UnsupportedFileSystemException, IOException {
        fileSystem
            .createSymlink(convertToDefaultPath(target), convertToDefaultPath(link), createParent);
    }

    @Override public FileStatus getFileLinkStatus(Path f)
        throws AccessControlException, FileNotFoundException, UnsupportedFileSystemException,
        IOException {
        return fileSystem.getFileLinkStatus(convertToDefaultPath(f));
    }

    @Override public boolean supportsSymlinks() {
        return fileSystem.supportsSymlinks();
    }

    @Override public Path getLinkTarget(Path f) throws IOException {
        return convertToHoodiePath(fileSystem.getLinkTarget(convertToDefaultPath(f)));
    }

    @Override public FileChecksum getFileChecksum(Path f) throws IOException {
        return fileSystem.getFileChecksum(convertToDefaultPath(f));
    }

    @Override public FileChecksum getFileChecksum(Path f, long length) throws IOException {
        return fileSystem.getFileChecksum(convertToDefaultPath(f), length);
    }

    @Override public void setVerifyChecksum(boolean verifyChecksum) {
        fileSystem.setVerifyChecksum(verifyChecksum);
    }

    @Override public void setWriteChecksum(boolean writeChecksum) {
        fileSystem.setWriteChecksum(writeChecksum);
    }

    @Override public FsStatus getStatus() throws IOException {
        return fileSystem.getStatus();
    }

    @Override public FsStatus getStatus(Path p) throws IOException {
        return fileSystem.getStatus(convertToDefaultPath(p));
    }

    @Override public void setPermission(Path p, FsPermission permission) throws IOException {
        fileSystem.setPermission(convertToDefaultPath(p), permission);
    }

    @Override public void setOwner(Path p, String username, String groupname) throws IOException {
        fileSystem.setOwner(convertToDefaultPath(p), username, groupname);
    }

    @Override public void setTimes(Path p, long mtime, long atime) throws IOException {
        fileSystem.setTimes(convertToDefaultPath(p), mtime, atime);
    }

    @Override public Path createSnapshot(Path path, String snapshotName) throws IOException {
        return convertToHoodiePath(
            fileSystem.createSnapshot(convertToDefaultPath(path), snapshotName));
    }

    @Override public void renameSnapshot(Path path, String snapshotOldName, String snapshotNewName)
        throws IOException {
        fileSystem.renameSnapshot(convertToDefaultPath(path), snapshotOldName, snapshotNewName);
    }

    @Override public void deleteSnapshot(Path path, String snapshotName) throws IOException {
        fileSystem.deleteSnapshot(convertToDefaultPath(path), snapshotName);
    }

    @Override public void modifyAclEntries(Path path, List aclSpec) throws IOException {
        fileSystem.modifyAclEntries(convertToDefaultPath(path), aclSpec);
    }

    @Override public void removeAclEntries(Path path, List aclSpec) throws IOException {
        fileSystem.removeAclEntries(convertToDefaultPath(path), aclSpec);
    }

    @Override public void removeDefaultAcl(Path path) throws IOException {
        fileSystem.removeDefaultAcl(convertToDefaultPath(path));
    }

    @Override public void removeAcl(Path path) throws IOException {
        fileSystem.removeAcl(convertToDefaultPath(path));
    }

    @Override public void setAcl(Path path, List aclSpec) throws IOException {
        fileSystem.setAcl(convertToDefaultPath(path), aclSpec);
    }

    @Override public AclStatus getAclStatus(Path path) throws IOException {
        return fileSystem.getAclStatus(convertToDefaultPath(path));
    }

    @Override public void setXAttr(Path path, String name, byte[] value) throws IOException {
        fileSystem.setXAttr(convertToDefaultPath(path), name, value);
    }

    @Override public void setXAttr(Path path, String name, byte[] value, EnumSet flag)
        throws IOException {
        fileSystem.setXAttr(convertToDefaultPath(path), name, value, flag);
    }

    @Override public byte[] getXAttr(Path path, String name) throws IOException {
        return fileSystem.getXAttr(convertToDefaultPath(path), name);
    }

    @Override public Map getXAttrs(Path path) throws IOException {
        return fileSystem.getXAttrs(convertToDefaultPath(path));
    }

    @Override public Map getXAttrs(Path path, List names)
        throws IOException {
        return fileSystem.getXAttrs(convertToDefaultPath(path), names);
    }

    @Override public List listXAttrs(Path path) throws IOException {
        return fileSystem.listXAttrs(convertToDefaultPath(path));
    }

    @Override public void removeXAttr(Path path, String name) throws IOException {
        fileSystem.removeXAttr(convertToDefaultPath(path), name);
    }

    @Override public void setConf(Configuration conf) {
        // ignore this. we will set conf on init
    }

    @Override public Configuration getConf() {
        return fileSystem.getConf();
    }

    @Override public int hashCode() {
        return fileSystem.hashCode();
    }

    @Override public boolean equals(Object obj) {
        return fileSystem.equals(obj);
    }

    @Override public String toString() {
        return fileSystem.toString();
    }

    public Path convertToHoodiePath(Path oldPath) {
        return convertPathWithScheme(oldPath, getHoodieScheme(fileSystem.getScheme()));
    }

    public static Path convertToHoodiePath(Path file, Configuration conf) {
        String scheme = FileSystem.getDefaultUri(conf).getScheme();
        return convertPathWithScheme(file, getHoodieScheme(scheme));
    }

    private Path convertToDefaultPath(Path oldPath) {
        return convertPathWithScheme(oldPath, fileSystem.getScheme());
    }

    private Path[] convertDefaults(Path[] psrcs) {
        Path[] psrcsNew = new Path[psrcs.length];
        for (int i = 0; i < psrcs.length; i++) {
            psrcsNew[i] = convertToDefaultPath(psrcs[i]);
        }
        return psrcsNew;
    }

    private static Path convertPathWithScheme(Path oldPath, String newScheme) {
        URI oldURI = oldPath.toUri();
        URI newURI;
        try {
            newURI = new URI(newScheme, oldURI.getUserInfo(), oldURI.getHost(), oldURI.getPort(),
                oldURI.getPath(), oldURI.getQuery(), oldURI.getFragment());
            return new Path(newURI);
        } catch (URISyntaxException e) {
            // TODO - Better Exception handling
            throw new RuntimeException(e);
        }
    }

    public static String getHoodieScheme(String scheme) {
        String newScheme;
        if (SUPPORT_SCHEMES.contains(scheme)) {
            newScheme = HOODIE_SCHEME_PREFIX + scheme;
        } else {
            throw new IllegalArgumentException(
                "BlockAlignedAvroParquetWriter does not support scheme " + scheme);
        }
        return newScheme;
    }

    public long getBytesWritten(Path file) {
        if (openStreams.containsKey(file.getName())) {
            return openStreams.get(file.getName()).getBytesWritten();
        }
        // When the file is first written, we do not have a track of it
        throw new IllegalArgumentException(file.toString()
            + " does not have a open stream. Cannot get the bytes written on the stream");
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy