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

com.aoindustries.aoserv.client.backup.MysqlReplication Maven / Gradle / Ivy

There is a newer version: 1.92.0
Show newest version
/*
 * aoserv-client - Java client for the AOServ Platform.
 * Copyright (C) 2003-2009, 2014, 2016, 2017, 2018, 2019  AO Industries, Inc.
 *     [email protected]
 *     7262 Bull Pen Cir
 *     Mobile, AL 36695
 *
 * This file is part of aoserv-client.
 *
 * aoserv-client 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 3 of the License, or
 * (at your option) any later version.
 *
 * aoserv-client 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 aoserv-client.  If not, see .
 */
package com.aoindustries.aoserv.client.backup;

import com.aoindustries.aoserv.client.AOServConnector;
import com.aoindustries.aoserv.client.CachedObjectIntegerKey;
import com.aoindustries.aoserv.client.monitoring.AlertLevel;
import com.aoindustries.aoserv.client.mysql.Server;
import com.aoindustries.aoserv.client.net.Host;
import com.aoindustries.aoserv.client.schema.AoservProtocol;
import com.aoindustries.aoserv.client.schema.Table;
import com.aoindustries.io.stream.StreamableInput;
import com.aoindustries.io.stream.StreamableOutput;
import java.io.IOException;
import java.sql.ResultSet;
import java.sql.SQLException;

/**
 * Represents MySQL replication for one a FailoverFileReplication or {@link Server}.
 *
 * @author  AO Industries, Inc.
 */
final public class MysqlReplication extends CachedObjectIntegerKey {

	static final int
		COLUMN_PKEY=0,
		COLUMN_AO_SERVER=1,
		COLUMN_REPLICATION=2,
		COLUMN_MYSQL_SERVER=3
	;
	static final String COLUMN_AO_SERVER_name = "ao_server";
	static final String COLUMN_REPLICATION_name = "replication";
	static final String COLUMN_MYSQL_SERVER_name = "mysql_server";

	int ao_server;
	int replication;
	private int mysql_server;
	private int monitoring_seconds_behind_low;
	private int monitoring_seconds_behind_medium;
	private int monitoring_seconds_behind_high;
	private int monitoring_seconds_behind_critical;
	private AlertLevel maxAlertLevel;

	@Override
	protected Object getColumnImpl(int i) {
		switch(i) {
			case COLUMN_PKEY : return pkey;
			case COLUMN_AO_SERVER : return ao_server==-1 ? null : ao_server;
			case COLUMN_REPLICATION : return replication==-1 ? null : replication;
			case COLUMN_MYSQL_SERVER : return mysql_server;
			case 4 : return monitoring_seconds_behind_low==-1 ? null : monitoring_seconds_behind_low;
			case 5 : return monitoring_seconds_behind_medium==-1 ? null : monitoring_seconds_behind_medium;
			case 6 : return monitoring_seconds_behind_high==-1 ? null : monitoring_seconds_behind_high;
			case 7 : return monitoring_seconds_behind_critical==-1 ? null : monitoring_seconds_behind_critical;
			case 8 : return maxAlertLevel.name();
			default: throw new IllegalArgumentException("Invalid index: " + i);
		}
	}

	public com.aoindustries.aoserv.client.linux.Server getLinuxServer() throws SQLException, IOException {
		if(ao_server==-1) return null;
		com.aoindustries.aoserv.client.linux.Server ao=table.getConnector().getLinux().getServer().get(ao_server);
		if(ao==null) throw new SQLException("Unable to find linux.Server: "+ao_server);
		return ao;
	}

	public FileReplication getFailoverFileReplication() throws SQLException, IOException {
		if(replication==-1) return null;
		FileReplication ffr=table.getConnector().getBackup().getFileReplication().get(replication);
		if(ffr==null) throw new SQLException("Unable to find FailoverFileReplication: "+replication);
		return ffr;
	}

	public Server getMySQLServer() throws IOException, SQLException {
		Server ms=table.getConnector().getMysql().getServer().get(mysql_server);
		if(ms==null) throw new SQLException("Unable to find MySQLServer: "+mysql_server);
		return ms;
	}

	public int getMonitoringSecondsBehindLow() {
		return monitoring_seconds_behind_low;
	}

	public int getMonitoringSecondsBehindMedium() {
		return monitoring_seconds_behind_medium;
	}

	public int getMonitoringSecondsBehindHigh() {
		return monitoring_seconds_behind_high;
	}

	public int getMonitoringSecondsBehindCritical() {
		return monitoring_seconds_behind_critical;
	}

	/**
	 * Determines if monitoring is enabled for this replication.
	 * This is based on {@link Host#isMonitoringEnabled()} derived from either
	 * {@link #getLinuxServer()} or {@link #getFailoverFileReplication()}.
	 */
	public boolean isMonitoringEnabled() throws SQLException, IOException {
		com.aoindustries.aoserv.client.linux.Server linuxServer =
			(ao_server != -1)
			? getLinuxServer()
			: getFailoverFileReplication().getBackupPartition().getLinuxServer();
		return linuxServer.getHost().isMonitoringEnabled();
	}

	public AlertLevel getMaxAlertLevel() {
		return maxAlertLevel;
	}

	@Override
	public Table.TableID getTableID() {
		return Table.TableID.FAILOVER_MYSQL_REPLICATIONS;
	}

	@Override
	public void init(ResultSet result) throws SQLException {
		int pos = 1;
		pkey=result.getInt(pos++);
		ao_server=result.getInt(pos++);
		if(result.wasNull()) ao_server = -1;
		replication=result.getInt(pos++);
		if(result.wasNull()) replication = -1;
		mysql_server=result.getInt(pos++);
		monitoring_seconds_behind_low = result.getInt(pos++);
		if(result.wasNull()) monitoring_seconds_behind_low = -1;
		monitoring_seconds_behind_medium = result.getInt(pos++);
		if(result.wasNull()) monitoring_seconds_behind_medium = -1;
		monitoring_seconds_behind_high = result.getInt(pos++);
		if(result.wasNull()) monitoring_seconds_behind_high = -1;
		monitoring_seconds_behind_critical = result.getInt(pos++);
		if(result.wasNull()) monitoring_seconds_behind_critical = -1;
		maxAlertLevel = AlertLevel.valueOf(result.getString(pos++));
	}

	@Override
	public void read(StreamableInput in, AoservProtocol.Version protocolVersion) throws IOException {
		pkey = in.readCompressedInt();
		ao_server = in.readCompressedInt();
		replication = in.readCompressedInt();
		mysql_server = in.readCompressedInt();
		monitoring_seconds_behind_low = in.readCompressedInt();
		monitoring_seconds_behind_medium = in.readCompressedInt();
		monitoring_seconds_behind_high = in.readCompressedInt();
		monitoring_seconds_behind_critical = in.readCompressedInt();
		maxAlertLevel = AlertLevel.valueOf(in.readCompressedUTF());
	}

	@Override
	public String toStringImpl() throws IOException, SQLException {
		if(ao_server!=-1) return getMySQLServer().toStringImpl()+"->"+getLinuxServer().toStringImpl();
		else return getMySQLServer().toStringImpl()+"->"+getFailoverFileReplication().toStringImpl();
	}

	@Override
	public void write(StreamableOutput out, AoservProtocol.Version protocolVersion) throws IOException {
		out.writeCompressedInt(pkey);
		if(protocolVersion.compareTo(AoservProtocol.Version.VERSION_1_59)>=0) out.writeCompressedInt(ao_server);
		out.writeCompressedInt(replication);
		out.writeCompressedInt(mysql_server);
		if(protocolVersion.compareTo(AoservProtocol.Version.VERSION_1_56)>=0) {
			out.writeCompressedInt(monitoring_seconds_behind_low);
			out.writeCompressedInt(monitoring_seconds_behind_medium);
			out.writeCompressedInt(monitoring_seconds_behind_high);
			out.writeCompressedInt(monitoring_seconds_behind_critical);
		}
		if(protocolVersion.compareTo(AoservProtocol.Version.VERSION_1_74) >= 0) {
			out.writeCompressedUTF(maxAlertLevel.name());
		}
	}

	final public static class SlaveStatus {

		final private String slaveIOState;
		final private String masterLogFile;
		final private String readMasterLogPos;
		final private String relayLogFile;
		final private String relayLogPos;
		final private String relayMasterLogFile;
		final private String slaveIORunning;
		final private String slaveSQLRunning;
		final private String lastErrno;
		final private String lastError;
		final private String skipCounter;
		final private String execMasterLogPos;
		final private String relayLogSpace;
		final private String secondsBehindMaster;

		public SlaveStatus(
			String slaveIOState,
			String masterLogFile,
			String readMasterLogPos,
			String relayLogFile,
			String relayLogPos,
			String relayMasterLogFile,
			String slaveIORunning,
			String slaveSQLRunning,
			String lastErrno,
			String lastError,
			String skipCounter,
			String execMasterLogPos,
			String relayLogSpace,
			String secondsBehindMaster
		) {
			this.slaveIOState=slaveIOState;
			this.masterLogFile=masterLogFile;
			this.readMasterLogPos=readMasterLogPos;
			this.relayLogFile=relayLogFile;
			this.relayLogPos=relayLogPos;
			this.relayMasterLogFile=relayMasterLogFile;
			this.slaveIORunning=slaveIORunning;
			this.slaveSQLRunning=slaveSQLRunning;
			this.lastErrno=lastErrno;
			this.lastError=lastError;
			this.skipCounter=skipCounter;
			this.execMasterLogPos=execMasterLogPos;
			this.relayLogSpace=relayLogSpace;
			this.secondsBehindMaster=secondsBehindMaster;
		}

		public String getSlaveIOState() {
			return slaveIOState;
		}

		public String getMasterLogFile() {
			return masterLogFile;
		}

		public String getReadMasterLogPos() {
			return readMasterLogPos;
		}

		public String getRelayLogFile() {
			return relayLogFile;
		}

		public String getRelayLogPos() {
			return relayLogPos;
		}

		public String getRelayMasterLogFile() {
			return relayMasterLogFile;
		}

		public String getSlaveIORunning() {
			return slaveIORunning;
		}

		public String getSlaveSQLRunning() {
			return slaveSQLRunning;
		}

		public String getLastErrno() {
			return lastErrno;
		}

		public String getLastError() {
			return lastError;
		}

		public String getSkipCounter() {
			return skipCounter;
		}

		public String getExecMasterLogPos() {
			return execMasterLogPos;
		}

		public String getRelayLogSpace() {
			return relayLogSpace;
		}

		public String getSecondsBehindMaster() {
			return secondsBehindMaster;
		}
	}

	/**
	 * Gets the slave status or {@code null} if no slave status provided by MySQL.  If any error occurs, throws either
	 * IOException or SQLException.
	 */
	public SlaveStatus getSlaveStatus() throws IOException, SQLException {
		return table.getConnector().requestResult(
			true,
			AoservProtocol.CommandID.GET_MYSQL_SLAVE_STATUS,
			new AOServConnector.ResultRequest() {
				SlaveStatus result;

				@Override
				public void writeRequest(StreamableOutput out) throws IOException {
					out.writeCompressedInt(pkey);
				}

				@Override
				public void readResponse(StreamableInput in) throws IOException, SQLException {
					int code=in.readByte();
					if(code==AoservProtocol.NEXT) {
						result = new SlaveStatus(
							in.readNullUTF(),
							in.readNullUTF(),
							in.readNullUTF(),
							in.readNullUTF(),
							in.readNullUTF(),
							in.readNullUTF(),
							in.readNullUTF(),
							in.readNullUTF(),
							in.readNullUTF(),
							in.readNullUTF(),
							in.readNullUTF(),
							in.readNullUTF(),
							in.readNullUTF(),
							in.readNullUTF()
						);
					} else if(code==AoservProtocol.DONE) {
						result = null;
					} else {
						AoservProtocol.checkResult(code, in);
						throw new IOException("Unexpected response code: "+code);
					}
				}

				@Override
				public SlaveStatus afterRelease() {
					return result;
				}
			}
		);
	}
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy