data:image/s3,"s3://crabby-images/02ace/02ace956f9868cf2a1a780bd2c0a517cd3a46077" alt="JAR search and dependency download from the Maven repository"
com.larswerkman.boxer.Boxer Maven / Gradle / Ivy
/*
* Copyright 2014 Lars Werkman
*
* 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.larswerkman.boxer;
import com.larswerkman.boxer.internal.BoxerProcessor;
import com.larswerkman.boxer.wrappers.android.BundleWrapper;
import com.larswerkman.boxer.wrappers.android.DataMapWrapper;
import com.larswerkman.boxer.wrappers.android.ParcelWrapper;
import com.larswerkman.boxer.wrappers.android.SQLiteWrapper;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.util.*;
/**
* Boxer class used for serialization.
*
*
* Use {@link #from(Object)} to retrieve the correct Wrapper for serialization
*
*/
public abstract class Boxer {
private static HashMap> wrappers = new HashMap>();
private static HashMap> defaultWrappers = new HashMap>();
static {
defaultWrappers.put("android.os.Bundle", BundleWrapper.class);
defaultWrappers.put("android.os.Parcel", ParcelWrapper.class);
defaultWrappers.put("com.google.android.gms.wearable.DataMap", DataMapWrapper.class);
defaultWrappers.put("android.database.sqlite.SQLiteDatabase", SQLiteWrapper.class);
}
/**
* Empty constructor, Can't be a generic type because of ClassNotFoundException
* @param object Serialization object
*/
public Boxer(Object object) {
//Ensure the wrapper classes have a default constructor
}
/**
* Retrieve correct wrapper of object for serialization
* @param object Object to find wrapper for
*
* @return instance of {@link com.larswerkman.boxer.Boxer}
* or Null if there's no wrapper class known.
*/
public static Boxer from(Object object) {
for(String target : defaultWrappers.keySet()){
try{
if(Class.forName(target).isAssignableFrom(object.getClass())){
return defaultWrappers.get(target)
.getDeclaredConstructor(Object.class).newInstance(object);
}
} catch (Exception e){/*Do nothing*/}
}
try {
Class extends Boxer> wrapper = null;
if(wrappers.containsKey(object.getClass())){
wrapper = wrappers.get(object.getClass());
} else {
for(Class wrapperClass : wrappers.keySet()){
if(wrapperClass.isAssignableFrom(object.getClass())){
wrapper = wrappers.get(object.getClass());
break;
}
}
}
if(wrapper != null) {
return wrapper.getDeclaredConstructor(Object.class).newInstance(object);
}
} catch (Exception e){/*Do nothing*/}
return null;
}
/**
* Returns the fields of a specific class which will be serialized.
*
* @param clazz The boxbale class you want to inspect
* @return An {@link java.util.List} of fields, will return an empty
* {@link java.util.List} when there are no fields to be serialized
*/
public static List getBoxableFields(Class extends Boxable> clazz){
List fields = new ArrayList();
for(Field field : clazz.getDeclaredFields()){
if(!Modifier.isTransient(field.getModifiers())){
fields.add(field);
}
}
Class superClass = clazz.getSuperclass();
while(superClass != null){
for(Field field : superClass.getDeclaredFields()){
if(!fields.contains(field) && !Modifier.isTransient(field.getModifiers())){
fields.add(field);
}
}
superClass = superClass.getSuperclass();
}
return fields;
}
/**
* Globally register {@link com.larswerkman.boxer.Boxer} implementations
* to a specified {@code Class}.
*
* @param wrapper The class definition of the wrapper to be registered
* @param clazz The class definition ot which the wrapper will be bound.
*/
public static void registerWrapper(Class extends Boxer> wrapper, Class clazz){
if(wrapper == null){
throw new IllegalArgumentException("When adding a wrapper, the wrapper can't be null");
} else if(clazz == null){
throw new IllegalArgumentException("When adding a wrapper, the class can't be null");
}
wrappers.put(clazz, wrapper);
}
/**
* Globally remove a registered wrapper of the same type.
*
* @param clazz The class definition of the wrapper that needs to be removed.
* @return The class definition to which the wrapper was registered,
* will return null if no wrapper is removed
*/
public static Class removeWrapper(Class extends Boxer> clazz){
if(wrappers.containsValue(clazz)){
for(Map.Entry> entry : wrappers.entrySet()){
if(entry.getValue() == clazz) {
wrappers.remove(entry.getKey());
return entry.getKey();
}
}
}
return null;
}
/**
* Globally remove a registered wrapper that is registered for a certain type.
*
* @param clazz The class definition to which the wrapper is bound that needs to be removed
* @return The class definition of the wrapper that has been removed;
*/
public static Class extends Boxer> removeWrapperForType(Class clazz){
return wrappers.remove(clazz);
}
/**
* Globally clears all wrappers.
*/
public static void clearWrappers(){
wrappers.clear();
}
/**
* Stores a filled {@link com.larswerkman.boxer.Boxable} object.
*
* @param sub Subclass of the abstract {@link com.larswerkman.boxer.Boxer} class
* @param boxable object to be stored
* @param object Object to store the boxable in
* @param Subclass of {@link com.larswerkman.boxer.Boxer}
* @param Should implement the {@link com.larswerkman.boxer.Boxable} interface
* @param Generic serialization object
* @return A the filled instance of the {@link T} object that is given
*/
protected T storeBoxable(Class sub, B boxable, T object){
try {
Class boxer = boxableClass(boxable.getClass());
Method method = boxer.getMethod(BoxerProcessor.METHOD_SERIALIZE, boxable.getClass(), Boxer.class);
A wrapper = sub.getDeclaredConstructor(Object.class).newInstance(object);
method.invoke(null, boxable, wrapper);
} catch (Exception e){}
return object;
}
/**
* Stores a filled {@link com.larswerkman.boxer.Boxable} object.
*
* @param wrapper subclass of the abstract {@link com.larswerkman.boxer.Boxer} class
* @param boxable object to be stored
* @param Subclass of {@link com.larswerkman.boxer.Boxer}
* @param Should implement the {@link com.larswerkman.boxer.Boxable} interface
*/
protected void storeBoxable(A wrapper, B boxable){
try {
Class boxer = boxableClass(boxable.getClass());
Method method = boxer.getMethod(BoxerProcessor.METHOD_SERIALIZE, boxable.getClass(), Boxer.class);
method.invoke(null, boxable, wrapper);
} catch (Exception e){}
}
/**
* Retrieves a filled {@link com.larswerkman.boxer.Boxable} object.
*
* @param sub Subclass of the abstract {@link com.larswerkman.boxer.Boxer} class
* @param boxable Type of stored boxable
* @param object Object to retrieve data from.
* @param Subclass of {@link com.larswerkman.boxer.Boxer}
* @param Should implement the {@link com.larswerkman.boxer.Boxable} interface
* @param Generic serialization object
* @return A restored {@link com.larswerkman.boxer.Boxable} object of type {@link A}
*/
protected B retrieveBoxable(Class sub, Class boxable, T object){
B value = null;
try {
Class boxer = boxableClass(boxable);
Method method = boxer.getMethod(BoxerProcessor.METHOD_DESERIALIZE, Boxer.class);
A wrapper = sub.getDeclaredConstructor(Object.class).newInstance(object);
value = (B) method.invoke(null, wrapper);
} catch (Exception e){};
return value;
}
/**
* Retrieves a filled {@link com.larswerkman.boxer.Boxable} object.
*
* @param wrapper Subclass of the abstract {@link com.larswerkman.boxer.Boxer} class
* @param boxable Type of stored boxable
* @param Subclass of {@link com.larswerkman.boxer.Boxer}
* @param Should implement the {@link com.larswerkman.boxer.Boxable} interface
* @return A restored {@link com.larswerkman.boxer.Boxable} object of the type {@link A}
*/
protected B retrieveBoxable(A wrapper, Class boxable){
B value = null;
try{
Class boxer = boxableClass(boxable);
Method method = boxer.getMethod(BoxerProcessor.METHOD_DESERIALIZE, Boxer.class);
value = (B) method.invoke(null, wrapper);
} catch (Exception e){}
return value;
}
private static Class> boxableClass(Class tClass) throws ClassNotFoundException {
return Class.forName(tClass.getCanonicalName() + BoxerProcessor.CLASS_EXTENSION);
}
/*
Add methods
*/
/**
* Inserts a Boxable value into the mapping of the Boxer,
* replacing any existing value for the given key.
*
* @param key a String
* @param value a Boxable object, or null
* @param should extend a Boxable
*/
public abstract void addBoxable(String key, T value);
/**
* Inserts a Boxable List value into the mapping of the Boxer,
* replacing any existing value for the given key.
*
* @param key a String
* @param value a Boxable List object, or null
* @param should implement a Boxable interface
*/
public abstract void addBoxableList(String key, List value);
/**
* Inserts a Boxable array value into the mapping of the Boxer,
* replacing any existing value for the given key.
*
* @param key a String
* @param value a Boxable array object, or null
* @param should implement a Boxable interface
*/
public abstract void addBoxableArray(String key, T[] value);
/**
* Inserts an Enum value into the mapping of the Boxer,
* replacing any existing value for the given key.
*
* @param key a String
* @param value an Enum object, or null
*/
public abstract void addEnum(String key, Enum value);
/**
* Inserts an Enum array value into the mapping of the Boxer,
* replacing any existing value for the given key.
*
* @param key a String
* @param value an Enum array object, or null
*/
public abstract void addEnumArray(String key, Enum[] value);
/**
* Inserts an Enum List value into the mapping of the Boxer,
* replacing any existing value for the given key.
*
* @param key a String
* @param value an Enum List object, or null
*/
public abstract void addEnumList(String key, List extends Enum> value);
/**
* Inserts an String value into the mapping of the Boxer,
* replacing any existing value for the given key.
*
* @param key a String
* @param value an String object, or null
*/
public abstract void addString(String key, String value);
/**
* Inserts a String array value into the mapping of the Boxer,
* replacing any existing value for the given key.
*
* @param key a String
* @param value a String array object, or null
*/
public abstract void addStringArray(String key, String[] value);
/**
* Inserts a String List value into the mapping of the Boxer,
* replacing any existing value for the given key.
*
* @param key a String
* @param value a String List object, or null
*/
public abstract void addStringList(String key, List value);
/**
* Inserts a Boolean value into the mapping of the Boxer,
* replacing any existing value for the given key.
*
* @param key a String
* @param value a Boolean, or null
*/
public abstract void addBoolean(String key, boolean value);
/**
* Inserts a Boolean array value into the mapping of the Boxer,
* replacing any existing value for the given key.
*
* @param key a String
* @param value a Boolean array object, or null
*/
public abstract void addBooleanArray(String key, boolean[] value);
/**
* Inserts a Boolean List value into the mapping of the Boxer,
* replacing any existing value for the given key.
*
* @param key a String
* @param value a Boolean List object, or null
*/
public abstract void addBooleanList(String key, List value);
/**
* Inserts a Byte value into the mapping of the Boxer,
* replacing any existing value for the given key.
*
* @param key a String
* @param value a Byte, or null
*/
public abstract void addByte(String key, byte value);
/**
* Inserts a Byte array value into the mapping of the Boxer,
* replacing any existing value for the given key.
*
* @param key a String
* @param value a Byte array object, or null
*/
public abstract void addByteArray(String key, byte[] value);
/**
* Inserts a Byte List value into the mapping of the Boxer,
* replacing any existing value for the given key.
*
* @param key a String
* @param value a Byte List object, or null
*/
public abstract void addByteList(String key, List value);
/**
* Inserts a Character value into the mapping of the Boxer,
* replacing any existing value for the given key.
*
* @param key a String
* @param value a Character, or null
*/
public abstract void addChar(String key, char value);
/**
* Inserts a Character array value into the mapping of the Boxer,
* replacing any existing value for the given key.
*
* @param key a String
* @param value a Character array object, or null
*/
public abstract void addCharArray(String key, char[] value);
/**
* Inserts a Character List value into the mapping of the Boxer,
* replacing any existing value for the given key.
*
* @param key a String
* @param value a Character List object, or null
*/
public abstract void addCharList(String key, List value);
/**
* Inserts a Short value into the mapping of the Boxer,
* replacing any existing value for the given key.
*
* @param key a String
* @param value a Short, or null
*/
public abstract void addShort(String key, short value);
/**
* Inserts a Short array value into the mapping of the Boxer,
* replacing any existing value for the given key.
*
* @param key a String
* @param value a Short array object, or null
*/
public abstract void addShortArray(String key, short[] value);
/**
* Inserts a Short List value into the mapping of the Boxer,
* replacing any existing value for the given key.
*
* @param key a String
* @param value a Short List object, or null
*/
public abstract void addShortList(String key, List value);
/**
* Inserts an Integer value into the mapping of the Boxer,
* replacing any existing value for the given key.
*
* @param key a String
* @param value an Integer, or null
*/
public abstract void addInt(String key, int value);
/**
* Inserts an Integer array value into the mapping of the Boxer,
* replacing any existing value for the given key.
*
* @param key a String
* @param value an Integer array object, or null
*/
public abstract void addIntArray(String key, int[] value);
/**
* Inserts an Integer List value into the mapping of the Boxer,
* replacing any existing value for the given key.
*
* @param key a String
* @param value an Integer List object, or null
*/
public abstract void addIntList(String key, List value);
/**
* Inserts a Long value into the mapping of the Boxer,
* replacing any existing value for the given key.
*
* @param key a String
* @param value a Long, or null
*/
public abstract void addLong(String key, long value);
/**
* Inserts a Long array value into the mapping of the Boxer,
* replacing any existing value for the given key.
*
* @param key a String
* @param value a Long array object, or null
*/
public abstract void addLongArray(String key, long[] value);
/**
* Inserts a Long List value into the mapping of the Boxer,
* replacing any existing value for the given key.
*
* @param key a String
* @param value a Long List object, or null
*/
public abstract void addLongList(String key, List value);
/**
* Inserts a Double value into the mapping of the Boxer,
* replacing any existing value for the given key.
*
* @param key a String
* @param value a Double, or null
*/
public abstract void addDouble(String key, double value);
/**
* Inserts a Double array value into the mapping of the Boxer,
* replacing any existing value for the given key.
*
* @param key a String
* @param value a Double array object, or null
*/
public abstract void addDoubleArray(String key, double[] value);
/**
* Inserts a Double List value into the mapping of the Boxer,
* replacing any existing value for the given key.
*
* @param key a String
* @param value a Double List object, or null
*/
public abstract void addDoubleList(String key, List value);
/**
* Inserts a Float value into the mapping of the Boxer,
* replacing any existing value for the given key.
*
* @param key a String
* @param value a Float, or null
*/
public abstract void addFloat(String key, float value);
/**
* Inserts a Float array value into the mapping of the Boxer,
* replacing any existing value for the given key.
*
* @param key a String
* @param value a Float array object, or null
*/
public abstract void addFloatArray(String key, float[] value);
/**
* Inserts a Float List value into the mapping of the Boxer,
* replacing any existing value for the given key.
*
* @param key a String
* @param value a Float List object, or null
*/
public abstract void addFloatList(String key, List value);
/*
Get methods
*/
/**
* Returns the value associated with the given key, or null if
* no mapping of the desired type exists for the given key or a null
* value is explicitly associated with the key.
*
* @param should implement a Boxable interface
* @param key a String
* @param clazz type of Boxable expected class
* @return a Boxable value, or null
*/
public abstract T getBoxable(String key, Class clazz);
/**
* Returns the value associated with the given key, or null if
* no mapping of the desired type exists for the given key or a null
* value is explicitly associated with the key.
*
* @param key a String
* @param clazz type of Boxable expected class
* @param should implement a Boxable interface
* @return a Boxable[] value, or null
*/
public abstract T[] getBoxableArray(String key, Class clazz);
/**
* Returns the value associated with the given key, or null if
* no mapping of the desired type exists for the given key or a null
* value is explicitly associated with the key.
*
* @param key a String
* @param clazz type of Boxable expected class
* @param listtype type of expected List, should have a no-args constructor.
* @param should implement a Boxable interface
* @param Type of the List that will be instantiated
* @return a List value, or null
*/
public abstract > E getBoxableList(String key, Class clazz, Class listtype);
/**
* Returns the value associated with the given key, or null if
* no mapping of the desired type exists for the given key or a null
* value is explicitly associated with the key.
*
* @param key a String
* @param clazz type of Enum expected class
* @param should extend the Enum class
* @return an Enum value, or null
*/
public abstract T getEnum(String key, Class clazz);
/**
* Returns the value associated with the given key, or null if
* no mapping of the desired type exists for the given key or a null
* value is explicitly associated with the key.
*
* @param key a String
* @param clazz type of Enum expected class
* @param should extend the Enum class
* @return an Enum[] value, or null
*/
public abstract T[] getEnumArray(String key, Class clazz);
/**
* Returns the value associated with the given key, or null if
* no mapping of the desired type exists for the given key or a null
* value is explicitly associated with the key.
*
* @param should extend the Enum class
* @param Type of the List that will be instantiated
* @param key a String
* @param clazz type of Enum expected class
* @param listtype type of expected List, should have a no-args constructor.
* @return an List value, or null
*/
public abstract > List getEnumList(String key, Class clazz, Class listtype);
/**
* Returns the value associated with the given key, or null if
* no mapping of the desired type exists for the given key or a null
* value is explicitly associated with the key.
*
* @param key a String
* @return a String value, or null
*/
public abstract String getString(String key);
/**
* Returns the value associated with the given key, or null if
* no mapping of the desired type exists for the given key or a null
* value is explicitly associated with the key.
*
* @param key a String
* @return a String[] value, or null
*/
public abstract String[] getStringArray(String key);
/**
* Returns the value associated with the given key, or null if
* no mapping of the desired type exists for the given key or a null
* value is explicitly associated with the key.
*
* @param key a String
* @param listtype type of expected List, should have a no-args constructor.
* @param Type of the List that will be instantiated
* @return a List value, or null
*/
public abstract > T getStringList(String key, Class listtype);
/**
* Returns the value associated with the given key, or null if
* no mapping of the desired type exists for the given key or a null
* value is explicitly associated with the key.
*
* @param key a String
* @return a Boolean value, or null
*/
public abstract boolean getBoolean(String key);
/**
* Returns the value associated with the given key, or null if
* no mapping of the desired type exists for the given key or a null
* value is explicitly associated with the key.
*
* @param key a String
* @return a Boolean value, or null
*/
public abstract boolean[] getBooleanArray(String key);
/**
* Returns the value associated with the given key, or null if
* no mapping of the desired type exists for the given key or a null
* value is explicitly associated with the key.
*
* @param key a String
* @param listtype type of expected List, should have a no-args constructor.
* @param Type of the List that will be instantiated
* @return a List value, or null
*/
public abstract > T getBooleanList(String key, Class listtype);
/**
* Returns the value associated with the given key, or null if
* no mapping of the desired type exists for the given key or a null
* value is explicitly associated with the key.
*
* @param key a String
* @return a Byte value, or null
*/
public abstract byte getByte(String key);
/**
* Returns the value associated with the given key, or null if
* no mapping of the desired type exists for the given key or a null
* value is explicitly associated with the key.
*
* @param key a String
* @return a Byte[] value, or null
*/
public abstract byte[] getByteArray(String key);
/**
* Returns the value associated with the given key, or null if
* no mapping of the desired type exists for the given key or a null
* value is explicitly associated with the key.
*
* @param key a String
* @param listtype type of expected List, should have a no-args constructor.
* @param Type of the List that will be instantiated
* @return a List value, or null
*/
public abstract > T getByteList(String key, Class listtype);
/**
* Returns the value associated with the given key, or null if
* no mapping of the desired type exists for the given key or a null
* value is explicitly associated with the key.
*
* @param key a String
* @return a Character value, or null
*/
public abstract char getChar(String key);
/**
* Returns the value associated with the given key, or null if
* no mapping of the desired type exists for the given key or a null
* value is explicitly associated with the key.
*
* @param key a String
* @return a Character[] value, or null
*/
public abstract char[] getCharArray(String key);
/**
* Returns the value associated with the given key, or null if
* no mapping of the desired type exists for the given key or a null
* value is explicitly associated with the key.
*
* @param key a String
* @param listtype type of expected List, should have a no-args constructor.
* @param Type of the List that will be instantiated
* @return a List value, or null
*/
public abstract > T getCharList(String key, Class listtype);
/**
* Returns the value associated with the given key, or null if
* no mapping of the desired type exists for the given key or a null
* value is explicitly associated with the key.
*
* @param key a String
* @return a Short value, or null
*/
public abstract short getShort(String key);
/**
* Returns the value associated with the given key, or null if
* no mapping of the desired type exists for the given key or a null
* value is explicitly associated with the key.
*
* @param key a String
* @return a Short[] value, or null
*/
public abstract short[] getShortArray(String key);
/**
* Returns the value associated with the given key, or null if
* no mapping of the desired type exists for the given key or a null
* value is explicitly associated with the key.
*
* @param key a String
* @param listtype type of expected List, should have a no-args constructor.
* @param Type of the List that will be instantiated
* @return a List value, or null
*/
public abstract > T getShortList(String key, Class listtype);
/**
* Returns the value associated with the given key, or null if
* no mapping of the desired type exists for the given key or a null
* value is explicitly associated with the key.
*
* @param key a String
* @return an Integer value, or null
*/
public abstract int getInt(String key);
/**
* Returns the value associated with the given key, or null if
* no mapping of the desired type exists for the given key or a null
* value is explicitly associated with the key.
*
* @param key a String
* @return an Integer[] value, or null
*/
public abstract int[] getIntArray(String key);
/**
* Returns the value associated with the given key, or null if
* no mapping of the desired type exists for the given key or a null
* value is explicitly associated with the key.
*
* @param key a String
* @param listtype type of expected List, should have a no-args constructor.
* @param Type of the List that will be instantiated
* @return a List value, or null
*/
public abstract > T getIntList(String key, Class listtype);
/**
* Returns the value associated with the given key, or null if
* no mapping of the desired type exists for the given key or a null
* value is explicitly associated with the key.
*
* @param key a String
* @return a Long value, or null
*/
public abstract long getLong(String key);
/**
* Returns the value associated with the given key, or null if
* no mapping of the desired type exists for the given key or a null
* value is explicitly associated with the key.
*
* @param key a String
* @return a Long[] value, or null
*/
public abstract long[] getLongArray(String key);
/**
* Returns the value associated with the given key, or null if
* no mapping of the desired type exists for the given key or a null
* value is explicitly associated with the key.
*
* @param key a String
* @param listtype type of expected List, should have a no-args constructor.
* @param Type of the List that will be instantiated
* @return a List value, or null
*/
public abstract > T getLongList(String key, Class listtype);
/**
* Returns the value associated with the given key, or null if
* no mapping of the desired type exists for the given key or a null
* value is explicitly associated with the key.
*
* @param key a String
* @return a Double value, or null
*/
public abstract double getDouble(String key);
/**
* Returns the value associated with the given key, or null if
* no mapping of the desired type exists for the given key or a null
* value is explicitly associated with the key.
*
* @param key a String
* @return a Double[] value, or null
*/
public abstract double[] getDoubleArray(String key);
/**
* Returns the value associated with the given key, or null if
* no mapping of the desired type exists for the given key or a null
* value is explicitly associated with the key.
*
* @param key a String
* @param listtype type of expected List, should have a no-args constructor.
* @param Type of the List that will be instantiated
* @return a List value, or null
*/
public abstract > T getDoubleList(String key, Class listtype);
/**
* Returns the value associated with the given key, or null if
* no mapping of the desired type exists for the given key or a null
* value is explicitly associated with the key.
*
* @param key a String
* @return a Float value, or null
*/
public abstract float getFloat(String key);
/**
* Returns the value associated with the given key, or null if
* no mapping of the desired type exists for the given key or a null
* value is explicitly associated with the key.
*
* @param key a String
* @return a Float[] value, or null
*/
public abstract float[] getFloatArray(String key);
/**
* Returns the value associated with the given key, or null if
* no mapping of the desired type exists for the given key or a null
* value is explicitly associated with the key.
*
* @param key a String
* @param listtype type of expected List, should have a no-args constructor.
* @param Type of the List that will be instantiated
* @return a List value, or null
*/
public abstract > T getFloatList(String key, Class listtype);
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy