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

org.firebirdsql.gds.ClumpletReader Maven / Gradle / Ivy

There is a newer version: 6.0.0-beta-1
Show newest version
/*
 * Firebird Open Source JDBC Driver
 *
 * Distributable under LGPL license.
 * You may obtain a copy of the License at http://www.gnu.org/copyleft/lgpl.html
 *
 * 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
 * LGPL License for more details.
 *
 * This file was created by members of the firebird development team.
 * All individual contributions remain the Copyright (C) of those
 * individuals.  Contributors to this file are either listed here or
 * can be obtained from a source control history command.
 *
 * All rights reserved.
 */
package org.firebirdsql.gds;

import org.firebirdsql.encodings.Encoding;
import org.firebirdsql.gds.ng.FbExceptionBuilder;
import org.firebirdsql.jaybird.fb.constants.SpbItems;
import org.firebirdsql.jaybird.fb.constants.TpbItems;

import java.nio.charset.Charset;
import java.sql.SQLException;
import java.util.Arrays;

import static org.firebirdsql.gds.JaybirdErrorCodes.jb_clumpletReaderUsageError;
import static org.firebirdsql.gds.JaybirdErrorCodes.jb_invalidClumpletStructure;

/**
 * Reader for clumplets, similar to the implementation {@code ClumpletReader.cpp}.
 *
 * @author Mark Rotteveel
 * @since 3.0
 */
public class ClumpletReader {

    private final Kind kind;
    private final byte[] buffer;
    private int spbState; // Reflects state of spb parser/writer
    private int position;

    public ClumpletReader(Kind kind, byte[] buffer) {
        this.kind = kind;
        this.buffer = buffer;
    }

    public boolean isTagged() {
        switch (kind) {
        case Tpb:
        case Tagged:
        case WideTagged:
        case SpbAttach:
            return true;
        }

        return false;
    }

    public int getBufferTag() throws SQLException {
        switch (kind) {
        case Tpb:
        case Tagged:
        case WideTagged:
            if (buffer.length == 0) {
                throw invalidStructure("empty buffer");
            }
            return buffer[0];
        case SpbStart:
        case UnTagged:
        case WideUnTagged:
        case SpbSendItems:
        case SpbReceiveItems:
            throw usageMistake("buffer is not tagged");
        case SpbAttach:
            if (buffer.length == 0) {
                throw invalidStructure("empty buffer");
            }
            switch (buffer[0]) {
            case ISCConstants.isc_spb_version1:
                // This is old SPB format, it's almost like DPB -
                // buffer's tag is the first byte.
                return buffer[0];
            case ISCConstants.isc_spb_version:
                // Buffer's tag is the second byte
                if (buffer.length == 1) {
                    throw invalidStructure("buffer too short (1 byte)");
                }
                return buffer[1];
            case ISCConstants.isc_spb_version3:
                // This is wide SPB attach format - buffer's tag is the first byte.
                return buffer[0];
            default:
                throw invalidStructure("spb in service attach should begin with isc_spb_version1 or isc_spb_version");
            }
        default:
            throw new SQLException("Unexpected clumplet kind: " + kind);
        }
    }

    public ClumpletType getClumpletType(byte tag) throws SQLException {
        switch (kind) {
        case Tagged:
        case UnTagged:
        case SpbAttach:
            return ClumpletType.TraditionalDpb;
        case WideTagged:
        case WideUnTagged:
            return ClumpletType.Wide;
        case Tpb:
            switch (tag) {
            case TpbItems.isc_tpb_lock_write:
            case TpbItems.isc_tpb_lock_read:
            case TpbItems.isc_tpb_lock_timeout:
                return ClumpletType.TraditionalDpb;
            default:
                return ClumpletType.SingleTpb;
            }
        case SpbSendItems:
            switch (tag) {
            case ISCConstants.isc_info_svc_auth_block:
                return ClumpletType.Wide;
            case ISCConstants.isc_info_end:
            case ISCConstants.isc_info_truncated:
            case ISCConstants.isc_info_error:
            case ISCConstants.isc_info_data_not_ready:
            case ISCConstants.isc_info_length:
            case ISCConstants.isc_info_flag_end:
                return ClumpletType.SingleTpb;
            default:
                return ClumpletType.StringSpb;
            }
        case SpbReceiveItems:
            return ClumpletType.SingleTpb;
        case SpbStart:
            switch (tag) {
            case SpbItems.isc_spb_auth_block:
            case SpbItems.isc_spb_trusted_auth:
            case SpbItems.isc_spb_auth_plugin_name:
            case SpbItems.isc_spb_auth_plugin_list:
                return ClumpletType.Wide;
            }
            switch (spbState) {
            case 0:
                return ClumpletType.SingleTpb;
            case ISCConstants.isc_action_svc_backup:
            case ISCConstants.isc_action_svc_restore:
                switch (tag) {
                case ISCConstants.isc_spb_bkp_file:
                case SpbItems.isc_spb_dbname:
                case ISCConstants.isc_spb_res_fix_fss_data:
                case ISCConstants.isc_spb_res_fix_fss_metadata:
                case ISCConstants.isc_spb_bkp_stat:
                    return ClumpletType.StringSpb;
                case ISCConstants.isc_spb_bkp_factor:
                case ISCConstants.isc_spb_bkp_length:
                case ISCConstants.isc_spb_res_length:
                case ISCConstants.isc_spb_res_buffers:
                case ISCConstants.isc_spb_res_page_size:
                case SpbItems.isc_spb_options:
                case SpbItems.isc_spb_verbint:
                    return ClumpletType.IntSpb;
                case SpbItems.isc_spb_verbose:
                    return ClumpletType.SingleTpb;
                case ISCConstants.isc_spb_res_access_mode:
                    return ClumpletType.ByteSpb;
                default:
                    throw invalidStructure("unknown parameter for backup/restore");
                }
            case ISCConstants.isc_action_svc_repair:
                switch (tag) {
                case SpbItems.isc_spb_dbname:
                    return ClumpletType.StringSpb;
                case SpbItems.isc_spb_options:
                case ISCConstants.isc_spb_rpr_commit_trans:
                case ISCConstants.isc_spb_rpr_rollback_trans:
                case ISCConstants.isc_spb_rpr_recover_two_phase:
                    return ClumpletType.IntSpb;
                case ISCConstants.isc_spb_rpr_commit_trans_64:
                case ISCConstants.isc_spb_rpr_rollback_trans_64:
                case ISCConstants.isc_spb_rpr_recover_two_phase_64:
                    return ClumpletType.BigIntSpb;
                default:
                    throw invalidStructure("unknown parameter for repair");
                }
            case ISCConstants.isc_action_svc_add_user:
            case ISCConstants.isc_action_svc_delete_user:
            case ISCConstants.isc_action_svc_modify_user:
            case ISCConstants.isc_action_svc_display_user:
            case ISCConstants.isc_action_svc_display_user_adm:
            case ISCConstants.isc_action_svc_set_mapping:
            case ISCConstants.isc_action_svc_drop_mapping:
                switch (tag) {
                case SpbItems.isc_spb_dbname:
                case SpbItems.isc_spb_sql_role_name:
                case ISCConstants.isc_spb_sec_username:
                case ISCConstants.isc_spb_sec_password:
                case ISCConstants.isc_spb_sec_groupname:
                case ISCConstants.isc_spb_sec_firstname:
                case ISCConstants.isc_spb_sec_middlename:
                case ISCConstants.isc_spb_sec_lastname:
                    return ClumpletType.StringSpb;
                case ISCConstants.isc_spb_sec_userid:
                case ISCConstants.isc_spb_sec_groupid:
                case ISCConstants.isc_spb_sec_admin:
                    return ClumpletType.IntSpb;
                default:
                    throw invalidStructure("unknown parameter for security database operation");
                }
            case ISCConstants.isc_action_svc_properties:
                switch (tag) {
                case SpbItems.isc_spb_dbname:
                    return ClumpletType.StringSpb;
                case ISCConstants.isc_spb_prp_page_buffers:
                case ISCConstants.isc_spb_prp_sweep_interval:
                case ISCConstants.isc_spb_prp_shutdown_db:
                case ISCConstants.isc_spb_prp_deny_new_attachments:
                case ISCConstants.isc_spb_prp_deny_new_transactions:
                case ISCConstants.isc_spb_prp_set_sql_dialect:
                case SpbItems.isc_spb_options:
                case ISCConstants.isc_spb_prp_force_shutdown:
                case ISCConstants.isc_spb_prp_attachments_shutdown:
                case ISCConstants.isc_spb_prp_transactions_shutdown:
                    return ClumpletType.IntSpb;
                case ISCConstants.isc_spb_prp_reserve_space:
                case ISCConstants.isc_spb_prp_write_mode:
                case ISCConstants.isc_spb_prp_access_mode:
                case ISCConstants.isc_spb_prp_shutdown_mode:
                case ISCConstants.isc_spb_prp_online_mode:
                    return ClumpletType.ByteSpb;
                default:
                    throw invalidStructure("unknown parameter for setting database properties");
                }
            case ISCConstants.isc_action_svc_db_stats:
                switch (tag) {
                case SpbItems.isc_spb_dbname:
                case SpbItems.isc_spb_command_line:
                case ISCConstants.isc_spb_sts_table:
                    return ClumpletType.StringSpb;
                case SpbItems.isc_spb_options:
                    return ClumpletType.IntSpb;
                default:
                    throw invalidStructure("unknown parameter for getting statistics");
                }
            case ISCConstants.isc_action_svc_get_ib_log:
                throw invalidStructure("unknown parameter for getting log");
            case ISCConstants.isc_action_svc_nbak:
            case ISCConstants.isc_action_svc_nrest:
                switch (tag) {
                case ISCConstants.isc_spb_nbk_file:
                case ISCConstants.isc_spb_nbk_direct:
                case SpbItems.isc_spb_dbname:
                case ISCConstants.isc_spb_nbk_guid:
                    return ClumpletType.StringSpb;
                case ISCConstants.isc_spb_nbk_level:
                case SpbItems.isc_spb_options:
                case ISCConstants.isc_spb_nbk_keep_days:
                case ISCConstants.isc_spb_nbk_keep_rows:
                    return ClumpletType.IntSpb;
                case ISCConstants.isc_spb_nbk_clean_history:
                    return ClumpletType.SingleTpb;
                default:
                    throw invalidStructure("unknown parameter for nbackup");
                }
            case ISCConstants.isc_action_svc_nfix:
                switch (tag) {
                case SpbItems.isc_spb_dbname:
                    return ClumpletType.StringSpb;
                case SpbItems.isc_spb_options:
                    return ClumpletType.IntSpb;
                default:
                    throw invalidStructure("unknown parameter for nbackup");
                }
            case ISCConstants.isc_action_svc_trace_start:
            case ISCConstants.isc_action_svc_trace_stop:
            case ISCConstants.isc_action_svc_trace_suspend:
            case ISCConstants.isc_action_svc_trace_resume:
                switch (tag) {
                case ISCConstants.isc_spb_trc_cfg:
                case ISCConstants.isc_spb_trc_name:
                    return ClumpletType.StringSpb;
                case ISCConstants.isc_spb_trc_id:
                    return ClumpletType.IntSpb;
                }
                break;
            case ISCConstants.isc_action_svc_validate:
                switch (tag) {
                case ISCConstants.isc_spb_val_tab_incl:
                case ISCConstants.isc_spb_val_tab_excl:
                case ISCConstants.isc_spb_val_idx_incl:
                case ISCConstants.isc_spb_val_idx_excl:
                case SpbItems.isc_spb_dbname:
                    return ClumpletType.StringSpb;
                case ISCConstants.isc_spb_val_lock_timeout:
                    return ClumpletType.IntSpb;
                }
                break;
            }
            throw invalidStructure("wrong spb state");
        }
        throw invalidStructure("unknown reason");
    }

    public void adjustSpbState() throws SQLException {
        if (kind == Kind.SpbStart
                && spbState == 0
                && getClumpletSize(true, true, true) == 1) {
            spbState = getClumpTag();
        }
    }

    public int getClumpletSize(boolean wTag, boolean wLength, boolean wData) throws SQLException {
        // Check for EOF
        if (position >= buffer.length) {
            throw usageMistake("read past EOF");
        }

        int rc = wTag ? 1 : 0;
        int lengthSize = 0;
        int dataSize = 0;

        switch (getClumpletType(buffer[position])) {
        // This form allows clumplets of virtually any size
        case Wide:
            // Check did we receive length component for clumplet
            if (buffer.length - position < 5) {
                throw invalidStructure("buffer end before end of clumplet - no length component");
            }
            lengthSize = 4;
            dataSize = VaxEncoding.iscVaxInteger(buffer, position + 1, 4);
            break;

        // This is the most widely used form
        case TraditionalDpb:
            // Check did we receive length component for clumplet
            if (buffer.length - position < 2) {
                throw invalidStructure("buffer end before end of clumplet - no length component");
            }
            lengthSize = 1;
            dataSize = buffer[position + 1] & 0xFF;
            break;

        // Almost all TPB parameters are single bytes
        case SingleTpb:
            break;

        // Used in SPB for long strings
        case StringSpb:
            // Check did we receive length component for clumplet
            if (buffer.length - position < 3) {
                throw invalidStructure("buffer end before end of clumplet - no length component");
            }
            lengthSize = 2;
            dataSize = VaxEncoding.iscVaxInteger2(buffer, position + 1);
            break;

        // Used in SPB for 4-byte integers
        case IntSpb:
            dataSize = 4;
            break;

        // Used in SPB for 8-byte integers
        case BigIntSpb:
            dataSize = 8;
            break;

        // Used in SPB for single byte
        case ByteSpb:
            dataSize = 1;
            break;
        }

        int total = 1 + lengthSize + dataSize;
        if (position + total > buffer.length) {
            throw invalidStructure("buffer end before end of clumplet - clumplet too long");
        }

        if (wLength) {
            rc += lengthSize;
        }
        if (wData) {
            rc += dataSize;
        }
        return rc;
    }

    public void moveNext() throws SQLException {
        if (isEof()) {
            return; // no need to raise useless exceptions
        }
        int cs = getClumpletSize(true, true, true);
        adjustSpbState();
        position += cs;
    }

    public void rewind() throws SQLException {
        if (buffer == null || buffer.length == 0) {
            position = 0;
            spbState = 0;
            return;
        }
        switch (kind) {
        case UnTagged:
        case WideUnTagged:
        case SpbStart:
        case SpbSendItems:
        case SpbReceiveItems:
            position = 0;
            break;
        default:
            if (kind == Kind.SpbAttach && getBufferLength() > 0 && buffer[0] != ISCConstants.isc_spb_version1) {
                position = 2;
            } else {
                position = 1;
            }
        }
        spbState = 0;
    }

    public boolean find(int tag) throws SQLException {
        final int markPosition = position;
        for (rewind(); !isEof(); moveNext()) {
            if (tag == getClumpTag()) {
                return true;
            }
        }
        position = markPosition;
        return false;
    }

    /**
     * Finds the next {@code tag} in the reader, skipping tags to find it.
     * 

* If {@code tag} is not found, the original position of this reader is retained. *

* * @param tag * Tag to find * @return {@code true} if {@code tag} was found and this reader is positioned to read it, {@code false} otherwise * @throws SQLException * For errors positioning * @see #directNext(int) */ public boolean next(int tag) throws SQLException { if (!isEof()) { final int markPosition = position; if (tag == getClumpTag()) { moveNext(); } for (; !isEof(); moveNext()) { if (tag == getClumpTag()) { return true; } } position = markPosition; } return false; } /** * Checks if the next tag in this reader is {@code tag}. *

* If the next tag is not {@code tag}, the original position of this reader is retained. *

* * @param tag * Tag to find * @return {@code true} if the next tag is {@code tag} and this reader is positioned to read it, {@code false} * otherwise * @throws SQLException * For errors positioning * @see #next(int) * @since 5 */ public boolean directNext(int tag) throws SQLException { if (!isEof()) { final int markPosition = position; moveNext(); if (!isEof() && tag == getClumpTag()) { return true; } position = markPosition; } return false; } // Methods which work with currently selected clumplet public int getClumpTag() throws SQLException { // Check for EOF if (position >= buffer.length) { throw usageMistake("read past EOF"); } return buffer[position] & 0xFF; } public int getClumpLength() throws SQLException { return getClumpletSize(false, false, true); } public byte[] getBytes() throws SQLException { final int from = position + getClumpletSize(true, true, false); final int to = from + getClumpLength(); return Arrays.copyOfRange(buffer, from, to); } public int getInt() throws SQLException { final int length = getClumpLength(); if (length > 4) { throw invalidStructure("length of integer exceeds 4 bytes"); } final int from = position + getClumpletSize(true, true, false); return VaxEncoding.iscVaxInteger(buffer, from, length); } public long getLong() throws SQLException { final int length = getClumpLength(); if (length > 8) { throw invalidStructure("length of long exceeds 8 bytes"); } final int from = position + getClumpletSize(true, true, false); return VaxEncoding.iscVaxLong(buffer, from, length); } public String getString(Encoding encoding) throws SQLException { final int length = getClumpLength(); final int from = position + getClumpletSize(true, true, false); return encoding.decodeFromCharset(buffer, from, length); } public String getString(Charset charset) throws SQLException { final int length = getClumpLength(); final int from = position + getClumpletSize(true, true, false); return new String(buffer, from, length, charset); } // TODO: Other types in ClumpletReader.cpp not copied public boolean isEof() { return position >= getBufferLength(); } private int getBufferLength() { if (buffer.length == 1 && kind != Kind.UnTagged && kind != Kind.SpbStart && kind != Kind.WideUnTagged && kind != Kind.SpbSendItems && kind != Kind.SpbReceiveItems) { return 0; } return buffer.length; } private static SQLException invalidStructure(String message) { return FbExceptionBuilder.forException(jb_invalidClumpletStructure).messageParameter(message).toSQLException(); } private static SQLException usageMistake(String message) { return FbExceptionBuilder.forException(jb_clumpletReaderUsageError).messageParameter(message).toSQLException(); } @SuppressWarnings("java:S115") public enum Kind { EndOfList, Tagged, UnTagged, SpbAttach, SpbStart, Tpb, WideTagged, WideUnTagged, SpbSendItems, SpbReceiveItems } @SuppressWarnings("java:S115") public enum ClumpletType { TraditionalDpb, SingleTpb, StringSpb, IntSpb, BigIntSpb, ByteSpb, Wide } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy