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

io.vertx.db2client.impl.codec.InitialHandshakeCommandCodec Maven / Gradle / Ivy

There is a newer version: 5.0.0.CR3
Show newest version
/*
 * Copyright (C) 2019,2020 IBM Corporation
 *
 * 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 io.vertx.db2client.impl.codec;

import io.netty.buffer.ByteBuf;
import io.vertx.db2client.DB2Exception;
import io.vertx.db2client.impl.command.InitialHandshakeCommand;
import io.vertx.db2client.impl.drda.CCSIDConstants;
import io.vertx.db2client.impl.drda.DRDAConnectRequest;
import io.vertx.db2client.impl.drda.DRDAConnectResponse;
import io.vertx.db2client.impl.drda.DRDAConnectResponse.RDBAccessData;
import io.vertx.db2client.impl.drda.DRDAConstants;
import io.vertx.db2client.impl.drda.SQLState;
import io.vertx.db2client.impl.drda.SqlCode;
import io.vertx.sqlclient.internal.Connection;
import io.vertx.sqlclient.internal.command.CommandResponse;

/**
 * InitialHandshakeCommandCodec encodes the packets to get a connection from the database.
 * It sends the EXCSAT and ACCSEC commands in one packet, then once we can determine if we are accessing
 * Db2 LUW or Db2/z it sends the SECCHK and ACCRDB commands in a separate packet. The first two commands are
 * always EBCDIC, but the second two are EBCDIC for Db2 LUW and UTF8 for Db2/z
 *
 */
class InitialHandshakeCommandCodec extends AuthenticationCommandBaseCodec {

  private static enum ConnectionState {
    CONNECTING, AUTHENTICATING, CONNECTED, CONNECT_FAILED
  }

  private static final int TARGET_SECURITY_MEASURE = DRDAConstants.SECMEC_USRIDPWD;

  private ConnectionState status = ConnectionState.CONNECTING;

  InitialHandshakeCommandCodec(InitialHandshakeCommand cmd) {
    super(cmd);
  }

  @Override
  void encode(DB2Encoder encoder) {
    super.encode(encoder);
    encoder.socketConnection.connMetadata.databaseName = cmd.database();
    encoder.socketConnection.closeHandler(h -> {
      if (status == ConnectionState.CONNECTING) {
        // Sometimes DB2 closes the connection when sending an invalid Database name.
        // -4499 = A fatal error occurred that resulted in a disconnect from the data
        // source.
        // 08001 = "The connection was unable to be established"
        cmd.fail(new DB2Exception("The connection was closed by the database server.", SqlCode.CONNECTION_REFUSED,
            SQLState.AUTH_DATABASE_CONNECTION_REFUSED));
      }
    });

    ByteBuf packet = allocateBuffer();
    int packetStartIdx = packet.writerIndex();
    DRDAConnectRequest connectRequest = new DRDAConnectRequest(packet, encoder.socketConnection.connMetadata);
    connectRequest.buildEXCSAT(DRDAConstants.EXTNAM, // externalName,
        0x0A, // targetAgent,
        DRDAConstants.TARGET_SQL_AM, // targetSqlam,
        0x0C, // targetRdb,
        0x0A, // TARGET_SECURITY_MEASURE, //targetSecmgr,
        0, // targetCmntcpip,
        0, // targetCmnappc, (not used)
        0, // targetXamgr,
        0, // targetSyncptmgr,
        0, // targetRsyncmgr,
        CCSIDConstants.TARGET_UNICODE_MGR // targetUnicodemgr
    );
    connectRequest.buildACCSEC(TARGET_SECURITY_MEASURE, this.cmd.database(), null);
    encoder.socketConnection.connMetadata.correlationToken = connectRequest.getCorrelationToken(encoder.socketConnection.socket().localAddress().port());
    connectRequest.completeCommand();

    int lenOfPayload = packet.writerIndex() - packetStartIdx;
    sendPacket(packet, lenOfPayload);
  }

  @Override
  void decodePayload(ByteBuf payload, int payloadLength) {
    DRDAConnectResponse response = new DRDAConnectResponse(payload, encoder.socketConnection.connMetadata);
    switch (status) {
      case CONNECTING:
        response.readExchangeServerAttributes();
        // readAccessSecurity can throw a DB2Exception if there are problems connecting.
        // In that case, we want to catch that exception and
        // make sure to set the status to something other than ST_CONNECTING so we don't
        // try to complete the result twice (when we hit encode)
        try {
          response.readAccessSecurity(TARGET_SECURITY_MEASURE);
        } catch (DB2Exception de) {
          status = ConnectionState.CONNECT_FAILED;
          throw de;
        }

        ByteBuf packet = allocateBuffer();
        int packetStartIdx = packet.writerIndex();
        DRDAConnectRequest connectRequest = new DRDAConnectRequest(packet, encoder.socketConnection.connMetadata);

        connectRequest.buildSECCHK(TARGET_SECURITY_MEASURE, cmd.database(), cmd.username(), cmd.password(), null, // sectkn,
            null); // sectkn2
        connectRequest.buildACCRDB(cmd.database(), false, // readOnly,
            encoder.socketConnection.connMetadata.correlationToken, DRDAConstants.SYSTEM_ASC);
        connectRequest.completeCommand();

        int lenOfPayload = packet.writerIndex() - packetStartIdx;
        sendPacket(packet, lenOfPayload);
        status = ConnectionState.AUTHENTICATING;
        break;

      case AUTHENTICATING:
        response.readSecurityCheck();
        RDBAccessData accData = response.readAccessDatabase();
        if (accData.correlationToken != null) {
          encoder.socketConnection.connMetadata.correlationToken = accData.correlationToken;
        }
        status = ConnectionState.CONNECTED;
        completionHandler.handle(CommandResponse.success(cmd.connection()));
        break;

      default:
        cmd.fail(new DB2Exception("The connection was unable to be established. Invalid connection state.", SqlCode.CONNECTION_REFUSED,
          SQLState.AUTH_DATABASE_CONNECTION_REFUSED));
        break;

    }
  }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy