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

src.main.java.com.mebigfatguy.fbcontrib.utils.SerialVersionCalc Maven / Gradle / Ivy

/*
 * fb-contrib - Auxiliary detectors for Java programs
 * Copyright (C) 2005-2019 Dave Brosius
 *
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 2.1 of the License, or (at your option) any later version.
 *
 * This library 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
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with this library; if not, write to the Free Software
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
 */
package com.mebigfatguy.fbcontrib.utils;

import java.io.ByteArrayOutputStream;
import java.io.DataOutput;
import java.io.DataOutputStream;
import java.io.IOException;
import java.io.Serializable;
import java.nio.ByteBuffer;
import java.nio.ByteOrder;
import java.security.MessageDigest;
import java.security.NoSuchAlgorithmException;
import java.util.Arrays;
import java.util.Comparator;

import org.apache.bcel.Constants;
import org.apache.bcel.classfile.Field;
import org.apache.bcel.classfile.JavaClass;
import org.apache.bcel.classfile.Method;

public class SerialVersionCalc {

    enum ModifierType {
        CLASS, METHOD, FIELD
    }

    public static long uuid(JavaClass cls) throws IOException {

        if (cls.isEnum()) {
            return 0;
        }

        try (ByteArrayOutputStream baos = new ByteArrayOutputStream();
             DataOutputStream dos = new DataOutputStream(baos)) {

            dos.writeUTF(cls.getClassName());
            dos.writeInt(filterModifiers(cls.getModifiers(), ModifierType.CLASS));

            writeInterfaces(dos, cls);
            writeFields(dos, cls);
            writeMethods(dos, cls);

            dos.flush();
            MessageDigest digest = MessageDigest.getInstance("SHA-1");
            byte[] shaBytes = digest.digest(baos.toByteArray());

            ByteBuffer bb = ByteBuffer.wrap(shaBytes, 0, 8);
            bb.order(ByteOrder.LITTLE_ENDIAN);
            return bb.getLong();

        } catch (NoSuchAlgorithmException e) {
            return 0;
        }
    }
    
    private static void writeInterfaces(DataOutput out, JavaClass cls) throws IOException {
        String[] infs = cls.getInterfaceNames();
        if (infs.length > 0) {
            infs = infs.clone();
            Arrays.sort(infs);
            for (String inf : infs) {
                out.writeUTF(inf);
            }
        }
    }

    private static void writeFields(DataOutput out, JavaClass cls) throws IOException {
        Field[] fields = cls.getFields();
        if (fields.length > 0) {
            fields = fields.clone();
            Arrays.sort(fields, new FieldSorter());
            for (Field field : fields) {
                if (!field.isPrivate() || (!field.isStatic() && !field.isTransient())) {
                    out.writeUTF(field.getName());
                    out.writeInt(filterModifiers(field.getModifiers(), ModifierType.FIELD));
                    out.writeUTF(field.getSignature());
                }
            }
        }
    }

    private static void writeMethods(DataOutput out, JavaClass cls) throws IOException {
        Method[] methods = cls.getMethods();
        if (methods.length > 0) {
            methods = methods.clone();
            Arrays.sort(methods, new MethodSorter());

            for (Method sinit : methods) {
                if ("".equals(sinit.getName())) {
                    out.writeUTF("");
                    out.writeInt(Constants.ACC_STATIC);
                    out.writeUTF("()V");
                    break;
                }
            }

            for (Method init : methods) {
                if ("".equals(init.getName()) && !init.isPrivate()) {
                    out.writeUTF("");
                    out.writeInt(filterModifiers(init.getModifiers(), ModifierType.METHOD));
                    out.writeUTF(init.getSignature().replace('/', '.')); // how bazaar
                }
            }

            for (Method method : methods) {
                if (!"".equals(method.getName()) && !"".equals(method.getName()) && !method.isPrivate()) {
                    out.writeUTF(method.getName());
                    out.writeInt(filterModifiers(method.getModifiers(), ModifierType.METHOD));
                    out.writeUTF(method.getSignature().replace('/', '.')); // how bazaar
                }
            }
        }
    }
    
    private static int filterModifiers(int modifier, ModifierType type) {

        switch (type) {
        case CLASS:
            return modifier
                    & (Constants.ACC_PUBLIC | Constants.ACC_FINAL | Constants.ACC_INTERFACE | Constants.ACC_ABSTRACT);

        case METHOD:
            return modifier & (Constants.ACC_PUBLIC | Constants.ACC_PRIVATE | Constants.ACC_PROTECTED
                    | Constants.ACC_STATIC | Constants.ACC_FINAL | Constants.ACC_SYNCHRONIZED | Constants.ACC_NATIVE
                    | Constants.ACC_ABSTRACT | Constants.ACC_STRICT);

        case FIELD:
            return modifier & (Constants.ACC_PUBLIC | Constants.ACC_PRIVATE | Constants.ACC_PROTECTED
                    | Constants.ACC_STATIC | Constants.ACC_FINAL | Constants.ACC_VOLATILE | Constants.ACC_TRANSIENT);

        default:
            return 0;
        }

    }

    static class FieldSorter implements Comparator, Serializable {
        private static final long serialVersionUID = 1L;

        @Override
        public int compare(Field f1, Field f2) {
            return f1.getName().compareTo(f2.getName());
        }
    }

    static class MethodSorter implements Comparator, Serializable {
        private static final long serialVersionUID = 1L;

        @Override
        public int compare(Method m1, Method m2) {
            int cmp = m1.getName().compareTo(m2.getName());
            if (cmp != 0) {
                return cmp;
            }

            return m1.getSignature().compareTo(m2.getSignature());
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy