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

org.apache.river.lookup.entry.LookupAttributes Maven / Gradle / Ivy

The newest version!
/*
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership. The ASF licenses this file
 * to you 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 org.apache.river.lookup.entry;

import net.jini.core.entry.Entry;
import net.jini.lookup.entry.ServiceControlled;
import java.lang.reflect.Field;
import java.lang.reflect.Modifier;
import java.io.IOException;
import java.io.Serializable;
import java.util.Arrays;
import java.util.Comparator;
import net.jini.io.MarshalledInstance;

/**
 * Some simple utilities for manipulating lookup service attributes.
 * These are not high-performance operations; it is expected that
 * they are called relatively infrequently.
 *
 * @author Sun Microsystems, Inc.
 *
 */
public class LookupAttributes {

    private LookupAttributes() {}

    /** Comparator for sorting fields */
    private static final FieldComparator comparator = new FieldComparator();
    private static final Class[] noArg = new Class[0];

    /**
     * Returns a new array containing the elements of the 
     * addAttrSets parameter (that are not duplicates of
     * any of the elements already in the attrSets parameter)
     * added to the elements of attrSets. The parameter
     * arrays are not modified.
     * 

* Note that attribute equality is defined in terms of * MarshalledObject.equals on field values. The * parameter arrays are not modified. *

* Throws an IllegalArgumentException if any element of * addAttrSets is not an instance of a valid * Entry class (the class is not public, or does not have a * no-arg constructor, or has primitive public non-static non-final * fields). * @param attrSets original Entry attributes * @param addAttrSets Entry attributes to added if not already contained in attrSets * @return new array of Entry attributes, containing no duplicates. */ public static Entry[] add(Entry[] attrSets, Entry[] addAttrSets) { return add(attrSets, addAttrSets, false); } /** * Returns a new array containing the elements of the * addAttrSets parameter (that are not duplicates of * any of the elements already in the attrSets parameter) * added to the elements of attrSets. The parameter * arrays are not modified. *

* Note that attribute equality is defined in terms of * MarshalledInstance.equals on field values. The * parameter arrays are not modified. *

* If the checkSC parameter is true, * then a SecurityException is thrown if any elements * of the addAttrSets parameter are instanceof * ServiceControlled. *

* Throws an IllegalArgumentException if any element of * addAttrSets is not an instance of a valid * Entry class (the class is not public, or does not have a * no-arg constructor, or has primitive public non-static non-final * fields). * @param attrSets original Entry attributes * @param addAttrSets Entry attributes to added if not already contained in attrSets * @param checkSC if true checks for any elements of addAttrSets are instances * of ServiceControlled. * @return new array of Entry attributes, containing no duplicates. */ public static Entry[] add(Entry[] attrSets, Entry[] addAttrSets, boolean checkSC) { check(addAttrSets, false); Entry[] newSets = concat(attrSets, addAttrSets); for (int i = newSets.length; --i >= attrSets.length; ) { if (checkSC) check(newSets[i]); if (isDup(newSets, i)) newSets = delete(newSets, i); } return newSets; } /** * Returns a new array that contains copies of the attributes in the * attrSets parameter, modified according to the contents * of both the attrSetTmpls parameter and the * modAttrSets parameter. The parameter arrays and * their Entry instances are not modified. *

* Throws an IllegalArgumentException if any element of * attrSetTmpls or modAttrSets is not an * instance of a valid Entry class (the class is not public, * or does not have a no-arg constructor, or has primitive public * non-static non-final fields). * @param attrSets Entry attributes * @param attrSetTmpls Entry attribute templates * @param modAttrSets Entry modified attributes * @return a new array of Entry attributes. */ public static Entry[] modify(Entry[] attrSets, Entry[] attrSetTmpls, Entry[] modAttrSets) { return modify(attrSets, attrSetTmpls, modAttrSets, false); } /** * Returns a new array that contains copies of the attributes in the * attrSets parameter, modified according to the contents * of both the attrSetTmpls parameter and the * modAttrSets parameter. The parameter arrays and * their Entry instances are not modified. *

* If the checkSC parameter is true, then a * SecurityException is thrown if any elements of the * attrSets parameter that would be deleted or modified * are instanceof ServiceControlled. *

* Throws an IllegalArgumentException if any element of * attrSetTmpls or modAttrSets is not an * instance of a valid Entry class (the class is not public, * or does not have a no-arg constructor, or has primitive public * non-static non-final fields). * @param attrSets Entry attributes * @param attrSetTmpls Entry attribute templates * @param modAttrSets Entry modified attributes * @param checkSC if true checks for any elements of addAttrSets are instances * of ServiceControlled. * @return a new array of Entry attributes. */ public static Entry[] modify(Entry[] attrSets, Entry[] attrSetTmpls, Entry[] modAttrSets, boolean checkSC) { if (attrSetTmpls.length != modAttrSets.length) throw new IllegalArgumentException( "attribute set length mismatch"); for (int i = modAttrSets.length; --i >= 0; ) { if (modAttrSets[i] != null && !isAssignableFrom(modAttrSets[i].getClass(), attrSetTmpls[i].getClass())) throw new IllegalArgumentException( "attribute set type mismatch"); } check(attrSetTmpls, false); check(modAttrSets, true); attrSets = (Entry[])attrSets.clone(); for (int i = attrSets.length; --i >= 0; ) { Entry pre = attrSets[i]; for (int j = attrSetTmpls.length; --j >= 0; ) { if (matches(attrSetTmpls[j], pre)) { if (checkSC) check(pre); Entry mods = modAttrSets[j]; if (mods == null) { attrSets = delete(attrSets, i); break; } else { attrSets[i] = update(attrSets[i], mods); } } } } for (int i = attrSets.length; --i >= 0; ) { if (isDup(attrSets, i)) attrSets = delete(attrSets, i); } return attrSets; } /** * Test that two entries are the same type, with the same * public fields. Attribute equality is defined in terms of * MarshalledObject.equals on field values. * @param e1 an Entry * @param e2 A second Entry * @return true if Entries are the same type. */ public static boolean equal(Entry e1, Entry e2) { if (!equal(e1.getClass(), e2.getClass())) return false; Field[] fields1 = getFields(e1); Field[] fields2 = getFields(e2, e1, fields1); try { for (int i = fields1.length; --i >= 0; ) { if (!equal(fields1[i].get(e1), fields2[i].get(e2))) return false; } } catch (IllegalAccessException ex) { throw new IllegalArgumentException( "unexpected IllegalAccessException"); } return true; } /** Tests that two Entry[] arrays are the same. * @param attrSet1 first Entry array. * @param attrSet2 second Entry array * @return true if both arrays contain the same number of elements with * equal entries */ public static boolean equal(Entry[] attrSet1, Entry[] attrSet2) { return contains(attrSet1, attrSet2) && contains(attrSet2, attrSet1); } /** * Test if the parameter tmpl is the same class as, or a * superclass of, the parameter e, and that every * non-null public field of tmpl is the * same as the corresponding field of e. Attribute equality * is defined in terms of MarshalledObject.equals on * field values. * @param tmpl Entry template. * @param e Entry to be checked for template matching. * @return true if Entry matches template. */ public static boolean matches(Entry tmpl, Entry e) { if (!isAssignableFrom(tmpl.getClass(), e.getClass())) return false; Field[] tfields = getFields(tmpl); Field[] efields = getFields(e, tmpl, tfields); try { for (int i = tfields.length; --i >= 0; ) { Object val = tfields[i].get(tmpl); if (val != null && !equal(val, efields[i].get(e))) return false; } } catch (IllegalAccessException ex) { throw new IllegalArgumentException ("unexpected IllegalAccessException"); } return true; } /** * Throws an IllegalArgumentException if any element of * the array is not an instance of a valid Entry class * (the class is not public, or does not have a no-arg constructor, or * has primitive public non-static non-final fields). If * nullOK is false, and any element of the * array is null, a NullPointerException * is thrown. * @param attrs to be checked if valid Entry classes. * @param nullOK true if array can contain null elements. */ public static void check(Entry[] attrs, boolean nullOK) { for (int i = attrs.length; --i >= 0; ) { Entry e = attrs[i]; if (e == null ){ if (nullOK) continue; throw new NullPointerException("null Entry not ok"); } Class c = e.getClass(); if (!Modifier.isPublic(c.getModifiers())) throw new IllegalArgumentException("entry class " + c.getName() + " is not public"); try { c.getConstructor(noArg); } catch (NoSuchMethodException ex) { throw new IllegalArgumentException("entry class " + c.getName() + " does not have a public no-arg constructor"); } Field[] fields = c.getFields(); for (int j = fields.length; --j >= 0; ) { if ((fields[j].getModifiers() & (Modifier.STATIC|Modifier.FINAL|Modifier.TRANSIENT)) == 0 && fields[j].getType().isPrimitive()) throw new IllegalArgumentException("entry class " + c.getName() + " has a primitive field"); } } } /** * Throws a SecurityException if parameter e * is instanceof ServiceControlled. */ private static void check(Entry e) { if (e instanceof ServiceControlled) throw new SecurityException ("attempt to add or modify a ServiceControlled attribute set"); } /** * Test if the set at the given index is equal to any * other set earlier in the Entry[] array parameter. */ private static boolean isDup(Entry[] attrs, int index) { Entry set = attrs[index]; for (int i = index; --i >= 0; ) { if (equal(set, attrs[i])) return true; } return false; } /** * Return a new entry that, for each non-null field of * the parameter mods, has the same field value as * mods, else the same field value as the parameter * e. */ private static Entry update(Entry e, Entry mods) { try { Entry ec = (Entry)e.getClass().newInstance(); Field[] mfields = getFields(mods); Field[] efields = getFields(e, mods, mfields); for (int i = efields.length; --i >= 0; ) { efields[i].set(ec, efields[i].get(e)); } for (int i = mfields.length; --i >= 0; ) { Object val = mfields[i].get(mods); if (val != null) efields[i].set(ec, val); } return ec; } catch (InstantiationException ex) { throw new IllegalArgumentException( "unexpected InstantiationException"); } catch (IllegalAccessException ex) { throw new IllegalArgumentException( "unexpected IllegalAccessException"); } } /** * Returns true if the two input objects are the same in * MarshalledInstance form, false otherwise. */ private static boolean equal(Object o1, Object o2) { if (o1 == o2) return true; if (o1 == null || o2 == null) return false; Class c = o1.getClass(); if (c == String.class || c == Integer.class || c == Boolean.class || c == Character.class || c == Long.class || c == Float.class || c == Double.class || c == Byte.class || c == Short.class) return o1.equals(o2); try { return new MarshalledInstance(o1).equals(new MarshalledInstance(o2)); } catch (IOException ex) { throw new IllegalArgumentException("unexpected IOException"); } } /** * Tests if two classes are equal, using the class equivalence * semantics of the lookup service: same name. */ private static boolean equal(Class c1, Class c2) { return c1.equals(c2) || c1.getName().equals(c2.getName()); } /** * Tests if class c1 is equal to, or a superclass of, * class c2, using the class equivalence semantics of * the lookup service: same name. */ private static boolean isAssignableFrom(Class c1, Class c2) { if (c1.isAssignableFrom(c2)) return true; String n1 = c1.getName(); for (Class sup = c2; sup != null; sup = sup.getSuperclass()) { if (n1.equals(sup.getName())) return true; } return false; } /** * Returns public fields, in super to subclass order, sorted * alphabetically within a given class. */ private static Field[] getFields(Entry e) { Field[] fields = e.getClass().getFields(); Arrays.sort(fields, comparator); int len = 0; for (int i = 0; i < fields.length; i++) { if ((fields[i].getModifiers() & (Modifier.STATIC|Modifier.FINAL|Modifier.TRANSIENT)) == 0) fields[len++] = fields[i]; } if (len < fields.length) { Field[] nfields = new Field[len]; System.arraycopy(fields, 0, nfields, 0, len); fields = nfields; } return fields; } /** Comparator for sorting fields. */ private static class FieldComparator implements Comparator, Serializable { private static final long serialVersionUID = 1L; public FieldComparator() {} /** * Sorts superclass fields before subclass fields, and sorts * fields alphabetically within a given class. */ public int compare(Object o1, Object o2) { Field f1 = (Field)o1; Field f2 = (Field)o2; if (f1 == f2) return 0; if (f1.getDeclaringClass() == f2.getDeclaringClass()) return f1.getName().compareTo(f2.getName()); if (f1.getDeclaringClass().isAssignableFrom( f2.getDeclaringClass())) return -1; return 1; } } /** * Returns the public fields of the parameter e. If * e and parameter oe have the same class, * then returns parameter ofields, otherwise ensures that * e has at least as many fields as does parameter * ofields. */ private static Field[] getFields(Entry e, Entry oe, Field[] ofields) { if (e.getClass().equals(oe.getClass())) return ofields; Field[] fields = getFields(e); if (fields.length < ofields.length) throw new IllegalArgumentException("type mismatch"); return fields; } /** Return a concatenation of the two arrays. */ private static Entry[] concat(Entry[] attrs1, Entry[] attrs2) { Entry[] nattrs = new Entry[attrs1.length + attrs2.length]; System.arraycopy(attrs1, 0, nattrs, 0, attrs1.length); System.arraycopy(attrs2, 0, nattrs, attrs1.length, attrs2.length); return nattrs; } /** Return a new array containing all but the given element. */ private static Entry[] delete(Entry[] attrs, int i) { int len = attrs.length - 1; Entry[] nattrs = new Entry[len]; System.arraycopy(attrs, 0, nattrs, 0, i); System.arraycopy(attrs, i + 1, nattrs, i, len - i); return nattrs; } /** * Returns true if the Entry parameter * e is an element of the Entry[] array * parameter eSet; returns false otherwise. */ private static boolean contains(Entry[] eSet, Entry e) { for(int i=0; itrue if the Entry[] array parameter * eSet1 contains the Entry[] array parameter * eSet2; returns false otherwise. That is, * this method determines if eSet2 is a subset of * eSet1. */ private static boolean contains(Entry[] eSet1, Entry[] eSet2) { int len1=0, len2=0; if(eSet1 != null) len1 = eSet1.length; if(eSet2 != null) len2 = eSet2.length; if(len1 < len2) return false; for(int i=0; i





© 2015 - 2025 Weber Informatics LLC | Privacy Policy